83 Commits

Author SHA1 Message Date
kojix2
1a5609022f v0.3.1 2020-12-04 13:56:06 +09:00
kojix2
e11b5047af Add yx format
A new format option for barplot and lineplot.
Use when the first column is the label and the second column is the value.
2020-12-02 18:24:22 +09:00
kojix2
9658bfa71c Remove needless space 2020-11-25 17:34:50 +09:00
kojix2
f2bd99ed2e Add the Processing module (tentative) 2020-11-25 17:31:41 +09:00
kojix2
9849898cb1 Change the test directory from uplot to youplot 2020-11-25 17:21:01 +09:00
kojix2
1a3ad9553c Rename Preprocessing to DSVReader
* DSV stands for Delimiter-separated values.
* Preprocessing is too vague.
2020-11-25 17:19:07 +09:00
kojix2
ccf232a742 Pass standard input to standard output first 2020-11-25 15:52:24 +09:00
kojix2
eb13f2583f Support for encodings other than UTF-8 2020-11-23 23:52:14 +09:00
kojix2
1697360b6b Add MacOS to CI 2020-11-23 21:47:53 +09:00
kojix2
7034a83dea Add some basic tests 2020-11-23 21:35:32 +09:00
kojix2
a0c3863b4c Add barplot test 2020-11-23 20:09:32 +09:00
kojix2
0ff8c6a9f0 Start testing the command line interface 2020-11-23 17:14:43 +09:00
kojix2
7a08d6bab9 Provides a mechanism to switch back-end libraries 2020-11-23 15:02:53 +09:00
kojix2
b72f982618 Add youplot as an executable file 2020-11-23 13:46:33 +09:00
kojix2
e831fa93f4 Rename uplot to youplot 2020-11-23 13:23:59 +09:00
kojix2
d85be56521 Removed needless file 2020-11-23 11:50:22 +09:00
kojix2
5c59a77054 Update README
* Build status badge travis -> github actions
2020-11-19 22:58:09 +09:00
kojix2
8c78465ce9 Fix CI 2020-11-19 22:49:15 +09:00
kojix2
648e606ed4 Use Guthub Actions 2020-11-19 22:42:47 +09:00
kojix2
58ba6bb966 v0.2.8 2020-11-15 00:46:45 +09:00
kojix2
b4585b053a Fix comment 2020-11-15 00:43:13 +09:00
kojix2
19c3b0367a Rubocop auto correct 2020-11-15 00:43:00 +09:00
kojix2
159b90998b Improved main help menu 2020-11-15 00:29:20 +09:00
kojix2
a6ff1ebf2e show the help option on usage 2020-11-14 23:41:35 +09:00
kojix2
f8ea11f0d0 Improved help messages 2020-11-14 23:09:49 +09:00
kojix2
406fb80377 Improved README 2020-11-14 22:22:37 +09:00
kojix2
4d761dd0e7 Show barplot with line num as labels if 1 series 2020-11-13 11:22:36 +09:00
kojix2
471f0a907e Removed raw_inputs 2020-11-10 23:09:41 +09:00
kojix2
552756cadf Fix test 2020-11-08 12:22:47 +09:00
kojix2
4b4848438c Make sure the labels are String 2020-11-08 12:21:05 +09:00
kojix2
6f9c77f4fe Rename count -> count_values 2020-11-08 01:41:13 +09:00
kojix2
40304329bc Update README.md 2020-11-06 10:37:29 +09:00
kojix2
2ecaa278c2 Update README
* How to show sub-coomand help
2020-11-06 10:29:17 +09:00
kojix2
3ab02e5a05 Update README.md 2020-11-06 10:23:44 +09:00
kojix2
2181e4a0f7 Fix color command option 2020-11-06 10:21:36 +09:00
kojix2
7fc7c797af Removed file name extensions 2020-11-06 10:00:55 +09:00
kojix2
942705ab23 Update help messages 2020-11-06 09:56:35 +09:00
kojix2
f7a7dcd1d4 Added that the command takes a file as an argument 2020-11-05 10:41:38 +09:00
kojix2
d085828883 Update README.md
* Added -e option to make the echo command work on both Mac and Ubuntu. (#1)
2020-11-04 16:08:34 +09:00
284km
93f8efc60a Output \n as a string in example code
Current output is:

$ echo "from numpy import random;" \
>      "n = random.randn(10000);"  \
>      "print('\n'.join(str(i) for i in n))"
from numpy import random; n = random.randn(10000); print('
'.join(str(i) for i in n))

After this change, I think it is originally expected string:

$ echo "from numpy import random;" \
>      "n = random.randn(10000);"  \
>      "print('\\\n'.join(str(i) for i in n))"
from numpy import random; n = random.randn(10000); print('\n'.join(str(i) for i in n))
2020-11-04 15:49:22 +09:00
kojix2
99e9e28ec9 Update README.md 2020-11-04 15:37:19 +09:00
kojix2
54e1865640 Update README.md 2020-10-13 18:40:56 +09:00
kojix2
00c2ce9b44 v0.2.7 2020-10-12 23:08:32 +09:00
kojix2
84196c197d Revert "Show message when No input data provided"
Uplot should expect cases where data takes a long time to reach the standard input.
This mechanism is not appropriate.

This reverts commit c40c59a21d.
2020-10-12 20:44:31 +09:00
kojix2
c40c59a21d Show message when No input data provided
Thank you csvkit developers
2020-10-12 17:10:25 +09:00
kojix2
c4f21df588 Show subcommand options ahead 2020-10-12 16:45:28 +09:00
kojix2
39166894a3 Improved help banner
Summary width 24
2020-10-12 16:14:51 +09:00
kojix2
7b8213833f Removed 'under development' in README 2020-10-12 14:57:03 +09:00
kojix2
9090bbf51b Fix help banner 2020-10-12 14:52:56 +09:00
kojix2
428d525c5f Add sample codes to README 2020-10-12 14:24:02 +09:00
kojix2
6ebc707c51 Add images to README.md 2020-10-12 14:04:36 +09:00
kojix2
c73da80de6 v0.2.6 2020-10-12 07:45:42 +09:00
kojix2
2e8641ccea Improved descs 2020-10-11 09:23:57 +09:00
kojix2
1b43f7d48f Add comment about pass 2020-10-11 08:29:02 +09:00
kojix2
d8396fecf9 Rubocop auto correct 2020-10-11 08:28:39 +09:00
kojix2
4660c2ab02 Extract case-branching by fmt into another method 2020-10-11 08:02:43 +09:00
kojix2
d7e49f048f Add a comment to params 2020-10-11 07:55:20 +09:00
kojix2
34ae2b5815 v0.2.5 2020-10-11 00:14:18 +09:00
kojix2
7e8dc6190c Fix arg order 2020-10-11 00:11:54 +09:00
kojix2
ba105ab1f3 Improved help desc 2020-10-10 23:58:53 +09:00
kojix2
96a1d1feb9 Fix style 2020-10-10 23:58:53 +09:00
kojix2
2b65dae60c Add a image to README.md 2020-10-10 23:35:26 +09:00
kojix2
943d4e6c44 Use keyword arg for count 2020-10-10 23:21:55 +09:00
kojix2
de33805c56 Drop support for ruby 2.3 2020-10-10 23:18:27 +09:00
kojix2
de3a366d15 Fix debug option 2020-10-10 23:16:15 +09:00
kojix2
4544c0e456 Removed count option for barplot 2020-10-10 23:08:20 +09:00
kojix2
ccfbaa7bde Allows you to specify the output file 2020-09-29 18:16:42 +09:00
kojix2
3aceae9279 Improved command description 2020-09-29 18:14:41 +09:00
kojix2
731daef3f8 v0.2.4 2020-09-25 00:55:00 +09:00
kojix2
78363cd198 Update banner 2020-09-25 00:52:33 +09:00
kojix2
0389f7fc5c create sub-parsers using case instead of Hash 2020-09-23 21:32:11 +09:00
kojix2
a33b0e7628 Rubocop auto correct 2020-09-19 00:08:09 +09:00
kojix2
4d62acea75 Remove get_lim
* Use only comma separators
* A hyphen is indistinguishable from a minus sign
* Colon separators aren't as common as commas
2020-09-19 00:06:03 +09:00
kojix2
e22976c1a2 Fix loading file order 2020-09-17 16:51:58 +09:00
kojix2
385d02d232 Fix color check 2020-09-17 10:45:57 +09:00
kojix2
0ce394a11d Fix style 2020-09-17 10:45:44 +09:00
kojix2
3baada320e Fix variables again 2020-09-17 10:38:59 +09:00
kojix2
b6c3ca9b43 Fix variables 2020-09-17 10:28:01 +09:00
kojix2
975eb95f55 Add parser class 2020-09-17 10:06:31 +09:00
kojix2
bd16c30613 Avoid multi-line chains of blocks... 2020-09-17 09:33:31 +09:00
kojix2
3a8c1e62f3 Fixed command descriptions 2020-09-17 00:24:08 +09:00
kojix2
522a111aa9 Improved help message of sub parsers 2020-09-16 23:50:24 +09:00
kojix2
7020785818 Added description of command line options 2020-09-16 23:14:20 +09:00
37 changed files with 1172 additions and 534 deletions

18
.github/workflows/ci.yml vendored Normal file
View File

@@ -0,0 +1,18 @@
name: test
on: [push, pull_request]
jobs:
build:
name: ${{ matrix.os }} Ruby ${{ matrix.ruby }}
runs-on: ${{ matrix.os }}-latest
strategy:
matrix:
os: ['ubuntu', 'macos']
ruby: [ '2.5', '2.6', '2.7' ]
steps:
- uses: actions/checkout@v2
- uses: actions/setup-ruby@v1
with:
ruby-version: ${{ matrix.ruby }}
- run: gem install bundler
- run: bundle install
- run: bundle exec rake test

View File

@@ -1,7 +0,0 @@
language: ruby
rvm: 2.7
script: bundle exec rake test
notifications:
email:
on_success: never
on_failure: change

View File

@@ -1,4 +1,6 @@
# frozen_string_literal: true
source 'https://rubygems.org'
# Specify your gem's dependencies in uplot.gemspec
# Specify your gem's dependencies in youplot.gemspec
gemspec

147
README.md
View File

@@ -1,23 +1,21 @@
# uplot
# YouPlot
[![Build Status](https://travis-ci.com/kojix2/uplot.svg?branch=master)](https://travis-ci.com/kojix2/uplot)
[![Gem Version](https://badge.fury.io/rb/u-plot.svg)](https://badge.fury.io/rb/u-plot)
[![Docs Latest](https://img.shields.io/badge/docs-latest-blue.svg)](https://rubydoc.info/gems/u-plot)
![Build Status](https://github.com/kojix2/youplot/workflows/test/badge.svg)
[![Gem Version](https://badge.fury.io/rb/youplot.svg)](https://badge.fury.io/rb/youplot)
[![Docs Latest](https://img.shields.io/badge/docs-latest-blue.svg)](https://rubydoc.info/gems/youplot)
[![The MIT License](https://img.shields.io/badge/license-MIT-blue.svg)](LICENSE.txt)
Create ASCII charts on the terminal with data from standard streams in the pipeline.
:bar_chart: Powered by [UnicodePlot](https://github.com/red-data-tools/unicode_plot.rb)
:construction: Under development! :construction:
## Installation
```
gem install u-plot
gem install youplot
```
## Usage
## Screenshots
**histogram**
@@ -26,42 +24,125 @@ ruby -r numo/narray -e "puts Numo::DFloat.new(1000).rand_norm.to_a" \
| uplot hist --nbins 15
```
<img src="https://i.imgur.com/wpsoGJq.png" width="75%" height="75%">
```sh
echo -e "from numpy import random;" \
"n = random.randn(10000);" \
"print('\\\n'.join(str(i) for i in n))" \
| python \
| uplot hist --nbins 20
```
┌ ┐
[-4.5, -4.0) ┤ 1
[-4.0, -3.5) ┤ 0
[-3.5, -3.0) ┤ 1
[-3.0, -2.5) ┤▇▇ 9
[-2.5, -2.0) ┤▇▇▇ 15
[-2.0, -1.5) ┤▇▇▇▇▇▇▇▇▇ 50
[-1.5, -1.0) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 97
[-1.0, -0.5) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 154
[-0.5, 0.0) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 193
[ 0.0, 0.5) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 165
[ 0.5, 1.0) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 152
[ 1.0, 1.5) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 86
[ 1.5, 2.0) ┤▇▇▇▇▇▇▇▇▇ 51
[ 2.0, 2.5) ┤▇▇▇▇ 21
[ 2.5, 3.0) ┤▇ 3
[ 3.0, 3.5) ┤ 2
└ ┘
Frequency
```
<img src="https://i.imgur.com/97R2MQx.png" width="75%" height="75%">
**scatter**
```sh
wget https://raw.githubusercontent.com/uiuc-cse/data-fa14/gh-pages/data/iris.csv -qO - \
| cut -f1-4 -d, \
| uplot scatter -H -d, -t IRIS -m 10
curl -s https://raw.githubusercontent.com/uiuc-cse/data-fa14/gh-pages/data/iris.csv \
| cut -f1-4 -d, \
| uplot scatter -H -d, -t IRIS
```
<img src="https://i.imgur.com/STX7bFT.png" width="75%" height="75%">
**line**
```sh
curl -s https://www.mhlw.go.jp/content/pcr_positive_daily.csv \
| cut -f2 -d, \
| uplot line -w 50 -h 15 -t 'PCR positive tests' --xlabel Date --ylabel number
```
<img src="https://i.imgur.com/PVl5dsa.png" width="75%" height="75%">
**box**
```sh
curl -s https://raw.githubusercontent.com/uiuc-cse/data-fa14/gh-pages/data/iris.csv \
| cut -f1-4 -d, \
| uplot box -H -d, -t IRIS
```
<img src="https://i.imgur.com/sNI4SmN.png" width="75%" height="75%">
**colors**
```sh
uplot colors
```
<img src="https://i.imgur.com/LxyHQsz.png">
## Usage
`uplot --help`
```
Program: YouPlot (Tools for plotting on the terminal)
Version: 0.2.7 (using UnicodePlot 0.0.4)
Source: https://github.com/kojix2/youplot
Usage: uplot <command> [options] <in.tsv>
Commands:
barplot bar
histogram hist
lineplot line
lineplots lines
scatter s
density d
boxplot box
colors show the list of available colors
count c baplot based on the number of occurrences
(slower than `sort | uniq -c | sort -n -k1`)
Options:
-O, --pass [VAL] file to output standard input data to [stdout]
for inserting YouPlot in the middle of Unix pipes
-o, --output VAL file to output results to [stderr]
-d, --delimiter VAL use DELIM instead of TAB for field delimiter
-H, --headers specify that the input has header row
-T, --transpose transpose the axes of the input data
-t, --title VAL print string on the top of plot
-x, --xlabel VAL print string on the bottom of the plot
-y, --ylabel VAL print string on the far left of the plot
-w, --width VAL number of characters per row
-h, --height VAL number of rows
-b, --border VAL specify the style of the bounding box
-m, --margin VAL number of spaces to the left of the plot
-p, --padding VAL space of the left and right of the plot
-c, --color VAL color of the drawing
--[no-]labels hide the labels
--fmt VAL xyxy : header is like x1, y1, x2, y2, x3, y3...
xyy : header is like x, y1, y2, y2, y3...
```
Use `--help` to print command-specific options.
`uplot hist --help`
```
Usage: uplot histogram [options] <in.tsv>
Options for histogram:
--symbol VAL character to be used to plot the bars
--closed VAL
-n, --nbins VAL approximate number of bins
Options:
...
```
## Development
Let's keep it simple.
## Contributing
Bug reports and pull requests are welcome on GitHub at [https://github.com/kojix2/uplot](https://github.com/kojix2/uplot).
Bug reports and pull requests are welcome on GitHub at [https://github.com/kojix2/youplot](https://github.com/kojix2/youplot).
## License
The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
[MIT License](https://opensource.org/licenses/MIT).

View File

@@ -1,3 +1,5 @@
# frozen_string_literal: true
require 'bundler/gem_tasks'
require 'rake/testtask'

View File

@@ -1,5 +1,6 @@
#!/usr/bin/env ruby
# frozen_string_literal: true
require 'uplot'
require 'youplot'
Uplot::Command.new.run
YouPlot::Command.new.run

6
exe/youplot Executable file
View File

@@ -0,0 +1,6 @@
#!/usr/bin/env ruby
# frozen_string_literal: true
require 'youplot'
YouPlot::Command.new.run

View File

@@ -1,8 +0,0 @@
require 'unicode_plot'
require 'uplot/version'
require 'uplot/preprocessing'
require 'uplot/plot'
require 'uplot/command'
module Uplot
end

View File

@@ -1,271 +0,0 @@
require 'optparse'
require_relative 'preprocessing'
require_relative 'command/params'
module Uplot
Data = Struct.new(:headers, :series)
class Command
attr_accessor :params, :command
attr_reader :main_parser, :sub_parsers, :raw_inputs, :data, :fmt
def initialize
@params = Params.new
@command = nil
@headers = nil
@delimiter = "\t"
@transpose = false
@output = false
@count = false
@fmt = 'xyy'
@raw_inputs = []
@debug = false
end
def create_default_parser
OptionParser.new do |opt|
opt.program_name = 'uplot'
opt.version = Uplot::VERSION
opt.on('-o', '--output', TrueClass) do |v|
@output = v
end
.on('-d', '--delimiter VAL', String) do |v|
@delimiter = v
end
.on('-H', '--headers', TrueClass) do |v|
@headers = v
end
.on('-T', '--transpose', TrueClass) do |v|
@transpose = v
end
.on('-t', '--title VAL', String) do |v|
params.title = v
end
.on('-w', '--width VAL', Numeric) do |v|
params.width = v
end
.on('-h', '--height VAL', Numeric) do |v|
params.height = v
end
.on('-b', '--border VAL', Numeric) do |v|
params.border = v
end
.on('-m', '--margin VAL', Numeric) do |v|
params.margin = v
end
.on('-p', '--padding VAL', Numeric) do |v|
params.padding = v
end
.on('-c', '--color VAL', String) do |v|
params.color = v =~ /\A[0-9]+\z/ ? v.to_i : v.to_sym
end
.on('-x', '--xlabel VAL', String) do |v|
params.xlabel = v
end
.on('-y', '--ylabel VAL', String) do |v|
params.ylabel = v
end
.on('-l', '--labels', TrueClass) do |v|
params.labels = v
end
.on('--fmt VAL', String) do |v|
@fmt = v
end
.on('--debug', TrueClass) do |v|
@debug = v
end
yield opt if block_given?
end
end
def create_sub_parsers
parsers = Hash.new { |h, k| h[k] = create_default_parser }
parsers[:barplot] = \
parsers[:bar]
.on('--symbol VAL', String) do |v|
params.symbol = v
end
.on('--xscale VAL', String) do |v|
params.xscale = v
end
.on('--count', TrueClass) do |v|
@count = v
end
parsers[:count] = \
parsers[:c] # barplot -c
.on('--symbol VAL', String) do |v|
params.symbol = v
end
parsers[:histogram] = \
parsers[:hist]
.on('-n', '--nbins VAL', Numeric) do |v|
params.nbins = v
end
.on('--closed VAL', String) do |v|
params.closed = v
end
.on('--symbol VAL', String) do |v|
params.symbol = v
end
parsers[:lineplot] = \
parsers[:line]
.on('--canvas VAL', String) do |v|
params.canvas = v
end
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
.on('--ylim VAL', String) do |v|
params.ylim = get_lim(v)
end
parsers[:lineplots] = \
parsers[:lines]
.on('--canvas VAL', String) do |v|
params.canvas = v
end
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
.on('--ylim VAL', String) do |v|
params.ylim = get_lim(v)
end
parsers[:scatter] = \
parsers[:s]
.on('--canvas VAL', String) do |v|
params.canvas = v
end
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
.on('--ylim VAL', String) do |v|
params.ylim = get_lim(v)
end
parsers[:density] = \
parsers[:d]
.on('--grid', TrueClass) do |v|
params.grid = v
end
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
.on('--ylim VAL', String) do |v|
params.ylim = get_lim(v)
end
parsers[:boxplot] = \
parsers[:box]
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
parsers[:colors]
.on('-n', '--names', TrueClass) do |v|
@color_names = v
end
# Preventing the generation of new sub-commands
parsers.default = nil
parsers
end
def create_main_parser
create_default_parser do |main_parser|
# Usage and help messages
main_parser.banner = \
<<~MSG
Program: uplot (Tools for plotting on the terminal)
Version: #{Uplot::VERSION} (using unicode_plot #{UnicodePlot::VERSION})
Usage: uplot <command> [options]
Command: #{sub_parsers.keys.join(' ')}
Options:
MSG
end
end
def parse_options(argv = ARGV)
@sub_parsers = create_sub_parsers
@main_parser = create_main_parser
begin
main_parser.order!(argv)
rescue OptionParser::ParseError => e
warn "uplot: #{e.message}"
exit 1
end
@command = argv.shift&.to_sym
unless sub_parsers.has_key?(command)
if command.nil?
warn main_parser.help
else
warn "uplot: unrecognized command '#{command}'"
end
exit 1
end
parser = sub_parsers[command]
begin
parser.parse!(argv) unless argv.empty?
rescue OptionParser::ParseError => e
warn "uplot: #{e.message}"
exit 1
end
end
def get_lim(str)
str.split(/-|:|\.\./)[0..1].map(&:to_f)
end
def run
parse_options
if command == :colors
Plot.colors
exit
end
# Sometimes the input file does not end with a newline code.
while input = Kernel.gets(nil)
input.freeze
@raw_inputs << input
@data = Preprocessing.input(input, @delimiter, @headers, @transpose)
pp @data if @debug
case command
when :bar, :barplot
Plot.barplot(data, params, @count)
when :count, :c
Plot.barplot(data, params, count = true)
when :hist, :histogram
Plot.histogram(data, params)
when :line, :lineplot
Plot.line(data, params)
when :lines, :lineplots
Plot.lines(data, params, fmt)
when :scatter, :s
Plot.scatter(data, params, fmt)
when :density, :d
Plot.density(data, params, fmt)
when :box, :boxplot
Plot.boxplot(data, params)
else
raise "unrecognized plot_type: #{command}"
end.render($stderr)
print input if @output
end
end
end
end

View File

@@ -1,157 +0,0 @@
require 'unicode_plot'
module Uplot
# plotting functions.
module Plot
module_function
def barplot(data, params, count = false)
headers = data.headers
series = data.series
if count
series = Preprocessing.count(series[0])
params.title = headers[0] if headers
end
params.title ||= headers[1] if headers
labels = series[0]
values = series[1].map(&:to_f)
UnicodePlot.barplot(labels, values, **params.to_hc)
end
def histogram(data, params)
headers = data.headers
series = data.series
params.title ||= data.headers[0] if headers
values = series[0].map(&:to_f)
UnicodePlot.histogram(values, **params.to_hc)
end
def line(data, params)
headers = data.headers
series = data.series
if series.size == 1
# If there is only one series, it is assumed to be sequential data.
params.ylabel ||= headers[0] if headers
y = series[0].map(&:to_f)
UnicodePlot.lineplot(y, **params.to_hc)
else
# If there are 2 or more series,
# assume that the first 2 series are the x and y series respectively.
if headers
params.xlabel ||= headers[0]
params.ylabel ||= headers[1]
end
x = series[0].map(&:to_f)
y = series[1].map(&:to_f)
UnicodePlot.lineplot(x, y, **params.to_hc)
end
end
def get_method2(method1)
(method1.to_s + '!').to_sym
end
def xyy_plot(data, method1, params)
headers = data.headers
series = data.series
method2 = get_method2(method1)
series.map! { |s| s.map(&:to_f) }
if headers
params.name ||= headers[1]
params.xlabel ||= headers[0]
end
params.ylim ||= series[1..-1].flatten.minmax # why need?
plot = UnicodePlot.public_send(method1, series[0], series[1], **params.to_hc)
2.upto(series.size - 1) do |i|
UnicodePlot.public_send(method2, plot, series[0], series[i], name: headers&.[](i))
end
plot
end
def xyxy_plot(data, method1, params)
headers = data.headers
series = data.series
method2 = get_method2(method1)
series.map! { |s| s.map(&:to_f) }
series = series.each_slice(2).to_a
params.name ||= headers[0] if headers
params.xlim = series.map(&:first).flatten.minmax # why need?
params.ylim = series.map(&:last).flatten.minmax # why need?
x1, y1 = series.shift
plot = UnicodePlot.public_send(method1, x1, y1, **params.to_hc)
series.each_with_index do |(xi, yi), i|
UnicodePlot.public_send(method2, plot, xi, yi, name: headers&.[]((i + 1) * 2))
end
plot
end
def lines(data, params, fmt = 'xyy')
check_series_size(data, fmt)
case fmt
when 'xyy'
xyy_plot(data, :lineplot, params)
when 'xyxy'
xyxy_plot(data, :lineplot, params)
end
end
def scatter(data, params, fmt = 'xyy')
check_series_size(data, fmt)
case fmt
when 'xyy'
xyy_plot(data, :scatterplot, params)
when 'xyxy'
xyxy_plot(data, :scatterplot, params)
end
end
def density(data, params, fmt = 'xyy')
check_series_size(data, fmt)
case fmt
when 'xyy'
xyy_plot(data, :densityplot, params)
when 'xyxy'
xyxy_plot(data, :densityplot, params)
end
end
def boxplot(data, params)
headers = data.headers
series = data.series
headers ||= (1..series.size).map(&:to_s)
series.map! { |s| s.map(&:to_f) }
UnicodePlot.boxplot(headers, series, **params.to_hc)
end
def colors
UnicodePlot::StyledPrinter::TEXT_COLORS.each do |k, v|
print v
print k
print "\t"
print ' ●'
print "\033[0m"
print "\t"
end
puts
end
def check_series_size(data, fmt)
series = data.series
if series.size == 1
warn 'uplot: There is only one series of input data. Please check the delimiter.'
warn ''
warn " Headers: \e[35m#{data.headers.inspect}\e[0m"
warn " The first item is: \e[35m\"#{series[0][0]}\"\e[0m"
warn " The last item is : \e[35m\"#{series[0][-1]}\"\e[0m"
exit 1
end
if fmt == 'xyxy' && series.size.odd?
warn 'uplot: In the xyxy format, the number of series must be even.'
warn ''
warn " Number of series: \e[35m#{series.size}\e[0m"
warn " Headers: \e[35m#{data.headers.inspect}\e[0m"
exit 1
end
end
end
end

View File

@@ -1,3 +0,0 @@
module Uplot
VERSION = '0.2.3'.freeze
end

9
lib/youplot.rb Normal file
View File

@@ -0,0 +1,9 @@
# frozen_string_literal: true
require 'unicode_plot'
require 'youplot/version'
require 'youplot/dsv_reader'
require 'youplot/command'
module YouPlot
end

View File

@@ -0,0 +1,24 @@
# frozen_string_literal: true
module YouPlot
# plotting functions.
module Backends
module Processing
module_function
def count_values(arr)
# tally was added in Ruby 2.7
if Enumerable.method_defined? :tally
arr.tally
else
# https://github.com/marcandre/backports
arr.each_with_object(Hash.new(0)) { |item, res| res[item] += 1 }
.tap { |h| h.default = nil }
end
.sort { |a, b| a[1] <=> b[1] }
.reverse
.transpose
end
end
end
end

View File

@@ -0,0 +1,188 @@
# frozen_string_literal: true
require_relative 'processing'
require 'unicode_plot'
module YouPlot
# plotting functions.
module Backends
module UnicodePlotBackend
module_function
def barplot(data, params, fmt = nil, count: false)
headers = data.headers
series = data.series
# `uplot count`
if count
series = Processing.count_values(series[0])
params.title = headers[0] if headers
end
if series.size == 1
# If there is only one series.use the line number for label.
params.title ||= headers[0] if headers
labels = Array.new(series[0].size) { |i| (i + 1).to_s }
values = series[0].map(&:to_f)
else
# If there are 2 or more series...
if fmt == 'yx'
# assume that the first 2 series are the y and x series respectively.
x_col = 1
y_col = 0
else
# assume that the first 2 series are the x and y series respectively.
x_col = 0
y_col = 1
end
params.title ||= headers[y_col] if headers
labels = series[x_col]
values = series[y_col].map(&:to_f)
end
UnicodePlot.barplot(labels, values, **params.to_hc)
end
def histogram(data, params)
headers = data.headers
series = data.series
params.title ||= data.headers[0] if headers
values = series[0].map(&:to_f)
UnicodePlot.histogram(values, **params.to_hc)
end
def line(data, params, fmt = nil)
headers = data.headers
series = data.series
if series.size == 1
# If there is only one series, it is assumed to be sequential data.
params.ylabel ||= headers[0] if headers
y = series[0].map(&:to_f)
UnicodePlot.lineplot(y, **params.to_hc)
else
# If there are 2 or more series...
if fmt == 'yx'
# assume that the first 2 series are the y and x series respectively.
x_col = 1
y_col = 0
else
# assume that the first 2 series are the x and y series respectively.
x_col = 0
y_col = 1
end
if headers
params.xlabel ||= headers[x_col]
params.ylabel ||= headers[y_col]
end
x = series[x_col].map(&:to_f)
y = series[y_col].map(&:to_f)
UnicodePlot.lineplot(x, y, **params.to_hc)
end
end
def get_method2(method1)
"#{method1}!".to_sym
end
def plot_xyy(data, method1, params)
headers = data.headers
series = data.series
method2 = get_method2(method1)
series.map! { |s| s.map(&:to_f) }
if headers
params.name ||= headers[1]
params.xlabel ||= headers[0]
end
params.ylim ||= series[1..-1].flatten.minmax # why need?
plot = UnicodePlot.public_send(method1, series[0], series[1], **params.to_hc)
2.upto(series.size - 1) do |i|
UnicodePlot.public_send(method2, plot, series[0], series[i], name: headers&.[](i))
end
plot
end
def plot_xyxy(data, method1, params)
headers = data.headers
series = data.series
method2 = get_method2(method1)
series.map! { |s| s.map(&:to_f) }
series = series.each_slice(2).to_a
params.name ||= headers[0] if headers
params.xlim = series.map(&:first).flatten.minmax # why need?
params.ylim = series.map(&:last).flatten.minmax # why need?
x1, y1 = series.shift
plot = UnicodePlot.public_send(method1, x1, y1, **params.to_hc)
series.each_with_index do |(xi, yi), i|
UnicodePlot.public_send(method2, plot, xi, yi, name: headers&.[]((i + 1) * 2))
end
plot
end
def plot_fmt(data, fmt, method1, params)
case fmt
when 'xyy'
plot_xyy(data, method1, params)
when 'xyxy'
plot_xyxy(data, method1, params)
when 'yx'
raise "Incorrect format: #{fmt}"
else
raise "Unknown format: #{fmt}"
end
end
def lines(data, params, fmt = 'xyy')
check_series_size(data, fmt)
plot_fmt(data, fmt, :lineplot, params)
end
def scatter(data, params, fmt = 'xyy')
check_series_size(data, fmt)
plot_fmt(data, fmt, :scatterplot, params)
end
def density(data, params, fmt = 'xyy')
check_series_size(data, fmt)
plot_fmt(data, fmt, :densityplot, params)
end
def boxplot(data, params)
headers = data.headers
series = data.series
headers ||= (1..series.size).map(&:to_s)
series.map! { |s| s.map(&:to_f) }
UnicodePlot.boxplot(headers, series, **params.to_hc)
end
def colors(color_names = false)
UnicodePlot::StyledPrinter::TEXT_COLORS.each do |k, v|
print v
print k
unless color_names
print "\t"
print ' ●'
end
print "\033[0m"
print "\t"
end
puts
end
def check_series_size(data, fmt)
series = data.series
if series.size == 1
warn 'youplot: There is only one series of input data. Please check the delimiter.'
warn ''
warn " Headers: \e[35m#{data.headers.inspect}\e[0m"
warn " The first item is: \e[35m\"#{series[0][0]}\"\e[0m"
warn " The last item is : \e[35m\"#{series[0][-1]}\"\e[0m"
exit 1
end
if fmt == 'xyxy' && series.size.odd?
warn 'YouPlot: In the xyxy format, the number of series must be even.'
warn ''
warn " Number of series: \e[35m#{series.size}\e[0m"
warn " Headers: \e[35m#{data.headers.inspect}\e[0m"
exit 1
end
end
end
end
end

98
lib/youplot/command.rb Normal file
View File

@@ -0,0 +1,98 @@
# frozen_string_literal: true
require_relative 'dsv_reader'
require_relative 'command/parser'
# FIXME
require_relative 'backends/unicode_plot_backend'
module YouPlot
Data = Struct.new(:headers, :series)
class Command
attr_accessor :params
attr_reader :data, :fmt, :parser
def initialize(argv = ARGV)
@argv = argv
@params = Params.new
@parser = Parser.new
@backend = YouPlot::Backends::UnicodePlotBackend
end
def run
parser.parse_options(@argv)
command = parser.command
params = parser.params
delimiter = parser.delimiter
transpose = parser.transpose
headers = parser.headers
pass = parser.pass
output = parser.output
fmt = parser.fmt
@encoding = parser.encoding
@debug = parser.debug
if command == :colors
@backend.colors(parser.color_names)
exit
end
# Sometimes the input file does not end with a newline code.
while (input = Kernel.gets(nil))
# Pass the input to subsequent pipelines
case pass
when IO
pass.print(input)
else
if pass
File.open(pass, 'w') do |f|
f.print(input)
end
end
end
@data = if @encoding
input2 = input.dup.force_encoding(@encoding).encode('utf-8')
DSVReader.input(input2, delimiter, headers, transpose)
else
DSVReader.input(input, delimiter, headers, transpose)
end
pp @data if @debug
plot = case command
when :bar, :barplot
@backend.barplot(data, params, fmt)
when :count, :c
@backend.barplot(data, params, count: true)
when :hist, :histogram
@backend.histogram(data, params)
when :line, :lineplot
@backend.line(data, params, fmt)
when :lines, :lineplots
@backend.lines(data, params, fmt)
when :scatter, :s
@backend.scatter(data, params, fmt)
when :density, :d
@backend.density(data, params, fmt)
when :box, :boxplot
@backend.boxplot(data, params)
else
raise "unrecognized plot_type: #{command}"
end
case output
when IO
plot.render(output)
else
File.open(output, 'w') do |f|
plot.render(f)
end
end
end
end
end
end

View File

@@ -1,5 +1,12 @@
module Uplot
# frozen_string_literal: true
module YouPlot
class Command
# UnicodePlot parameters.
# * Normally in a Ruby program, you might use hash for the parameter object.
# * Here, I use Struct for 2 safety reason.
# * The keys are static in Struct.
# * Struct does not conflict with keyword arguments. Hash dose.
Params = Struct.new(
# Sort me!
:title,

View File

@@ -0,0 +1,277 @@
# frozen_string_literal: true
require 'optparse'
require_relative 'params'
module YouPlot
class Command
class Parser
attr_reader :command, :params,
:delimiter, :transpose, :headers, :pass, :output, :fmt,
:color_names, :encoding, :debug
def initialize
@command = nil
@params = Params.new
@delimiter = "\t"
@transpose = false
@headers = nil
@pass = false
@output = $stderr
@fmt = 'xyy'
@encoding = nil
@debug = false
@color_names = false
end
def create_default_parser
OptionParser.new do |opt|
opt.program_name = 'YouPlot'
opt.version = YouPlot::VERSION
opt.summary_width = 24
opt.on_tail('') # Add a blank line at the end
opt.separator('')
opt.on('Common options:')
opt.on('-O', '--pass [VAL]', 'file to output standard input data to [stdout]',
'for inserting YouPlot in the middle of Unix pipes') do |v|
@pass = v || $stdout
end
opt.on('-o', '--output VAL', 'file to output results to [stderr]') do |v|
@output = v
end
opt.on('-d', '--delimiter VAL', String, 'use DELIM instead of TAB for field delimiter') do |v|
@delimiter = v
end
opt.on('-H', '--headers', TrueClass, 'specify that the input has header row') do |v|
@headers = v
end
opt.on('-T', '--transpose', TrueClass, 'transpose the axes of the input data') do |v|
@transpose = v
end
opt.on('-t', '--title VAL', String, 'print string on the top of plot') do |v|
params.title = v
end
opt.on('-x', '--xlabel VAL', String, 'print string on the bottom of the plot') do |v|
params.xlabel = v
end
opt.on('-y', '--ylabel VAL', String, 'print string on the far left of the plot') do |v|
params.ylabel = v
end
opt.on('-w', '--width VAL', Integer, 'number of characters per row') do |v|
params.width = v
end
opt.on('-h', '--height VAL', Numeric, 'number of rows') do |v|
params.height = v
end
opt.on('-b', '--border VAL', String, 'specify the style of the bounding box') do |v|
params.border = v.to_sym
end
opt.on('-m', '--margin VAL', Numeric, 'number of spaces to the left of the plot') do |v|
params.margin = v
end
opt.on('-p', '--padding VAL', Numeric, 'space of the left and right of the plot') do |v|
params.padding = v
end
opt.on('-c', '--color VAL', String, 'color of the drawing') do |v|
params.color = v =~ /\A[0-9]+\z/ ? v.to_i : v.to_sym
end
opt.on('--[no-]labels', TrueClass, 'hide the labels') do |v|
params.labels = v
end
opt.on('--encoding VAL', String, 'Specify the input encoding') do |v|
@encoding = v
end
# Optparse adds the help option, but it doesn't show up in usage.
# This is why you need the code below.
opt.on('--help', 'print sub-command help menu') do
puts opt.help
exit
end
opt.on('--debug', TrueClass, 'print preprocessed data') do |v|
@debug = v
end
yield opt if block_given?
end
end
def main_parser
@main_parser ||= create_default_parser do |main_parser|
# Here, help message is stored in the banner.
# Because help of main_parser may be referred by `sub_parser`.
main_parser.banner = \
<<~MSG
Program: YouPlot (Tools for plotting on the terminal)
Version: #{YouPlot::VERSION} (using UnicodePlot #{UnicodePlot::VERSION})
Source: https://github.com/kojix2/youplot
Usage: uplot <command> [options] <in.tsv>
Commands:
barplot bar draw a horizontal barplot
histogram hist draw a horizontal histogram
lineplot line draw a line chart
lineplots lines draw a line chart with multiple series
scatter s draw a scatter plot
density d draw a density plot
boxplot box draw a horizontal boxplot
colors show the list of available colors
count c draw a baplot based on the number of
occurrences (slow)
General options:
--help print command specific help menu
--version print the version of YouPlot
MSG
# Actually, main_parser can take common optional arguments.
# However, these options dose not be shown in the help menu.
# I think the main help should be simple.
main_parser.on('--help', 'print sub-command help menu') do
puts main_parser.banner
puts
exit
end
end
end
def sub_parser
@sub_parser ||= create_default_parser do |parser|
parser.banner = <<~MSG
Usage: YouPlot #{command} [options] <in.tsv>
Options for #{command}:
MSG
case command
# If you type only `uplot` in the terminal.
when nil
warn main_parser.banner
warn "\n"
exit 1
when :barplot, :bar
parser.on_head('--symbol VAL', String, 'character to be used to plot the bars') do |v|
params.symbol = v
end
parser.on_head('--xscale VAL', String, 'axis scaling') do |v|
params.xscale = v
end
parser.on_head('--fmt VAL', String, 'xy : header is like x, y...', 'yx : header is like y, x...') do |v|
@fmt = v
end
when :count, :c
parser.on_head('--symbol VAL', String, 'character to be used to plot the bars') do |v|
params.symbol = v
end
when :histogram, :hist
parser.on_head('-n', '--nbins VAL', Numeric, 'approximate number of bins') do |v|
params.nbins = v
end
parser.on_head('--closed VAL', String) do |v|
params.closed = v
end
parser.on_head('--symbol VAL', String, 'character to be used to plot the bars') do |v|
params.symbol = v
end
when :lineplot, :line
parser.on_head('--canvas VAL', String, 'type of canvas') do |v|
params.canvas = v
end
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
parser.on_head('--ylim VAL', Array, 'plotting range for the y coordinate') do |v|
params.ylim = v.take(2)
end
parser.on_head('--fmt VAL', String, 'xy : header is like x, y...', 'yx : header is like y, x...') do |v|
@fmt = v
end
when :lineplots, :lines
parser.on_head('--canvas VAL', String) do |v|
params.canvas = v
end
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
parser.on_head('--ylim VAL', Array, 'plotting range for the y coordinate') do |v|
params.ylim = v.take(2)
end
parser.on_head('--fmt VAL', String, 'xyxy : header is like x1, y1, x2, y2, x3, y3...', 'xyy : header is like x, y1, y2, y2, y3...') do |v|
@fmt = v
end
when :scatter, :s
parser.on_head('--canvas VAL', String) do |v|
params.canvas = v
end
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
parser.on_head('--ylim VAL', Array, 'plotting range for the y coordinate') do |v|
params.ylim = v.take(2)
end
parser.on_head('--fmt VAL', String, 'xyxy : header is like x1, y1, x2, y2, x3, y3...', 'xyy : header is like x, y1, y2, y2, y3...') do |v|
@fmt = v
end
when :density, :d
parser.on_head('--grid', TrueClass) do |v|
params.grid = v
end
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
parser.on_head('--ylim VAL', Array, 'plotting range for the y coordinate') do |v|
params.ylim = v.take(2)
end
parser.on('--fmt VAL', String, 'xyxy : header is like x1, y1, x2, y2, x3, y3...', 'xyy : header is like x, y1, y2, y2, y3...') do |v|
@fmt = v
end
when :boxplot, :box
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
when :colors
parser.on_head('-n', '--names', 'show color names only', TrueClass) do |v|
@color_names = v
end
else
warn "uplot: unrecognized command '#{command}'"
exit 1
end
end
end
def parse_options(argv = ARGV)
begin
main_parser.order!(argv)
rescue OptionParser::ParseError => e
warn "uplot: #{e.message}"
exit 1
end
@command = argv.shift&.to_sym
begin
sub_parser.parse!(argv)
rescue OptionParser::ParseError => e
warn "uplot: #{e.message}"
exit 1
end
end
end
end
end

View File

@@ -1,7 +1,10 @@
# frozen_string_literal: true
require 'csv'
module Uplot
module Preprocessing
module YouPlot
# Read and interpret Delimiter-separated values format file or stream.
module DSVReader
module_function
def input(input, delimiter, headers, transpose)
@@ -60,27 +63,11 @@ module Uplot
else
arr
end
elsif headers
transpose2(arr[1..-1])
else
if headers
transpose2(arr[1..-1])
else
transpose2(arr)
end
transpose2(arr)
end
end
def count(arr)
# tally was added in Ruby 2.7
if Enumerable.method_defined? :tally
arr.tally
else
# https://github.com/marcandre/backports
arr.each_with_object(Hash.new(0)) { |item, res| res[item] += 1 }
.tap { |h| h.default = nil }
end
.sort { |a, b| a[1] <=> b[1] }
.reverse
.transpose
end
end
end

5
lib/youplot/version.rb Normal file
View File

@@ -0,0 +1,5 @@
# frozen_string_literal: true
module YouPlot
VERSION = '0.3.1'
end

1
test/fixtures/colors.txt vendored Normal file

File diff suppressed because one or more lines are too long

153
test/fixtures/iris-barplot.txt vendored Normal file
View File

@@ -0,0 +1,153 @@
IRIS-BARPLOT
┌ ┐
5.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.5
4.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
4.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
4.6 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
5.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.6
5.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.9
4.6 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
5.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
4.4 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
4.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
5.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.7
4.8 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
4.8 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
4.3 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.8 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 4.0
5.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 4.4
5.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.9
5.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.5
5.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.8
5.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.8
5.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
5.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.7
4.6 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.6
5.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.3
4.8 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
5.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
5.2 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.5
5.2 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
4.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
4.8 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
5.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
5.2 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 4.1
5.5 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 4.2
4.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
5.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
5.5 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.5
4.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
4.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
5.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.5
4.5 ┤■■■■■■■■■■■■■■■■■■ 2.3
4.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
5.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.5
5.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.8
4.8 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.8
4.6 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
5.3 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.7
5.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.3
7.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
6.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
6.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
5.5 ┤■■■■■■■■■■■■■■■■■■ 2.3
6.5 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
5.7 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.3 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.3
4.9 ┤■■■■■■■■■■■■■■■■■■■ 2.4
6.6 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
5.2 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
5.0 ┤■■■■■■■■■■■■■■■■ 2.0
5.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.0 ┤■■■■■■■■■■■■■■■■■■ 2.2
6.1 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
5.6 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
6.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
5.6 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.8 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
6.2 ┤■■■■■■■■■■■■■■■■■■ 2.2
5.6 ┤■■■■■■■■■■■■■■■■■■■■ 2.5
5.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
6.1 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.3 ┤■■■■■■■■■■■■■■■■■■■■ 2.5
6.1 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.4 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
6.6 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.8 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.0 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
5.7 ┤■■■■■■■■■■■■■■■■■■■■■ 2.6
5.5 ┤■■■■■■■■■■■■■■■■■■■ 2.4
5.5 ┤■■■■■■■■■■■■■■■■■■■ 2.4
5.8 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
6.0 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
5.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
6.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
6.3 ┤■■■■■■■■■■■■■■■■■■ 2.3
5.6 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.5 ┤■■■■■■■■■■■■■■■■■■■■ 2.5
5.5 ┤■■■■■■■■■■■■■■■■■■■■■ 2.6
6.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.8 ┤■■■■■■■■■■■■■■■■■■■■■ 2.6
5.0 ┤■■■■■■■■■■■■■■■■■■ 2.3
5.6 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
5.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.7 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
6.2 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
5.1 ┤■■■■■■■■■■■■■■■■■■■■ 2.5
5.7 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.3 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.3
5.8 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
7.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.3 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
6.5 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
7.6 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
4.9 ┤■■■■■■■■■■■■■■■■■■■■ 2.5
7.3 ┤■■■■■■■■■■■■■■■■■■■■■■■ 2.9
6.7 ┤■■■■■■■■■■■■■■■■■■■■ 2.5
7.2 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.6
6.5 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
6.4 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
6.8 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
5.7 ┤■■■■■■■■■■■■■■■■■■■■ 2.5
5.8 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
6.5 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
7.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.8
7.7 ┤■■■■■■■■■■■■■■■■■■■■■ 2.6
6.0 ┤■■■■■■■■■■■■■■■■■■ 2.2
6.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
5.6 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
7.7 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.3 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
6.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.3
7.2 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
6.2 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.1 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.4 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
7.2 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
7.4 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
7.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.8
6.4 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.3 ┤■■■■■■■■■■■■■■■■■■■■■■ 2.8
6.1 ┤■■■■■■■■■■■■■■■■■■■■■ 2.6
7.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.3 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
6.4 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
6.0 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
6.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
6.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.1
5.8 ┤■■■■■■■■■■■■■■■■■■■■■ 2.7
6.8 ┤■■■■■■■■■■■■■■■■■■■■■■■■■ 3.2
6.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.3
6.7 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.3 ┤■■■■■■■■■■■■■■■■■■■■ 2.5
6.5 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
6.2 ┤■■■■■■■■■■■■■■■■■■■■■■■■■■■ 3.4
5.9 ┤■■■■■■■■■■■■■■■■■■■■■■■■ 3.0
└ ┘

19
test/fixtures/iris-boxplot.txt vendored Normal file
View File

@@ -0,0 +1,19 @@
IRIS-BOXPLOT
┌ ┐
╷ ┌──┬──┐ ╷
sepal_length ├───┤ │ ├───────┤
╵ └──┴──┘ ╵
╷ ┌┬┐ ╷
sepal_width ├───┤│├─────┤
╵ └┴┘ ╵
╷ ┌─────────────┬───┐ ╷
petal_length ├──┤ │ ├───────┤
╵ └─────────────┴───┘ ╵
╷┌───┬──┐ ╷
petal_width ├┤ │ ├──┤
╵└───┴──┘ ╵
species ┤
└ ┘
0 4 8

20
test/fixtures/iris-density.txt vendored Normal file
View File

@@ -0,0 +1,20 @@
IRIS-DENSITY
┌────────────────────────────────────────┐
6.9 │ │ sepal_width
│ │ petal_length
│ │ petal_width
│ │ species
│ ░ │
│ │
│ ░ │
│ ░ │
│ ░ ░ ░░ ░ │
│ ░ ░ ░ │
│ │
│ ░ ░▒ ░ ░ ░ ░ │
│ ░ ░ ░ │
│ │
0 │ ░░ ▒ ░█ ░ ░▒ ░░ ░ ░░░ ░ │
└────────────────────────────────────────┘
4 8
sepal_length

12
test/fixtures/iris-histogram.txt vendored Normal file
View File

@@ -0,0 +1,12 @@
IRIS-HISTOGRAM
┌ ┐
[4.0, 4.5) ┤▇▇▇▇▇ 4
[4.5, 5.0) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 18
[5.0, 5.5) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 30
[5.5, 6.0) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 31
[6.0, 6.5) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 32
[6.5, 7.0) ┤▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇▇ 22
[7.0, 7.5) ┤▇▇▇▇▇▇▇▇ 7
[7.5, 8.0) ┤▇▇▇▇▇▇▇ 6
└ ┘
Frequency

20
test/fixtures/iris-lineplot.txt vendored Normal file
View File

@@ -0,0 +1,20 @@
IRIS-LINEPLOT
┌────────────────────────────────────────┐
5 │⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⠎⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡔⠒⣽⠋⠀⢸⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢣⣼⠇⡠⠠⠊⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⢀⣀⣀⠤⢤⣼⢿⣿⣿⠯⠛⠉⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⠤⢊⡧⡺⠁│
sepal_width │⠀⠀⠀⠀⠀⠈⢑⠦⣺⣵⣿⣿⡝⢵⡠⠆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡠⠔⣺⠥⢊⠥⠊⠁⡷⠁⠀│
│⠀⠀⠀⠀⢀⡨⡽⣿⣯⢟⣿⡯⠿⠭⠥⠤⠤⢤⠤⠮⢍⣕⣣⣏⣉⣒⣦⣶⣭⣤⣺⣥⠊⠁⠀⠀⡰⡇⠀⠀│
│⠀⠀⠀⡠⣗⣭⣞⣿⣾⣟⡇⠀⠀⠀⢀⠤⢊⣁⣰⣣⣾⣟⣥⣾⣿⣽⢿⣿⡿⡿⠛⠉⠓⠢⠤⣴⣁⡇⠀⠀│
│⠀⠀⠈⠉⡟⠋⠉⡝⠀⠉⠁⠀⠀⠀⠉⢺⣷⣷⢿⣿⣿⣿⣷⣾⣿⣿⣿⣿⣿⣿⣯⢿⡿⢟⡻⠋⠉⡇⠀⠀│
│⠀⠀⠀⠀⢱⠀⡰⠁⠀⠀⠀⢀⢖⣶⣾⣿⣿⣻⡿⣿⣿⣿⣿⢿⣿⠟⠛⢻⢛⡻⠛⠉⠉⠉⠉⠉⠉⡇⠀⠀│
│⠀⠀⠀⠀⢸⢠⠃⠀⠀⢔⣶⡾⡿⣟⡿⢿⠷⣟⡿⢿⡟⢅⡱⢏⠎⠀⠀⠗⢁⣀⣀⠤⠤⠒⠒⠉⠉⠁⠀⠀│
│⠀⠀⠀⠀⠸⠇⠀⠀⠀⠁⢴⠋⡩⠊⠀⠗⠋⠁⠉⢺⣷⣎⡱⠮⠔⠒⠊⠉⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
2 │⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡮⠊⠀⠀⠀⠀⠀⠀⠀⠀⠁⠈⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
└────────────────────────────────────────┘
4 8
sepal_length

20
test/fixtures/iris-lineplots.txt vendored Normal file
View File

@@ -0,0 +1,20 @@
IRIS-LINEPLOTS
┌────────────────────────────────────────┐
6.9 │⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣀⣤⣲⠭⠃⠀⠀│ sepal_width
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡀⠀⠀⠀⢀⣀⣠⣤⣶⣿⣿⣯⣴⣒⠖⠉⠁│ petal_length
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣤⢮⣤⣤⣶⣿⣿⣿⠿⠛⠛⠋⠉⠀⠀⠀⠀⠀⠀│ petal_width
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣀⣤⣴⣷⣿⣿⣿⣿⣿⣿⣟⣟⡂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│ species
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣀⠤⠔⠒⢙⡻⠝⢋⡽⢝⣳⣾⣫⣥⣒⣪⣵⣖⣂⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠁⠀⠀⢀⣉⣟⣻⣿⣻⣿⣿⣿⠿⣿⣿⠿⠟⡢⠒⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣤⣿⣷⣾⣿⡿⣿⡯⠿⠛⠋⠉⠀⣀⠔⠉⠀⠀⠀⠀⠀⠀⠀⠀⢀⣀⣄⣀⠄│
│⠀⠀⠀⠀⢀⣨⣽⣿⣿⣿⣿⣿⣿⣿⣵⣚⣉⣃⣀⣠⣤⣤⣔⣮⣤⣄⣀⣠⣤⣔⣲⣾⡳⠮⠛⠋⢉⡗⠁⠀│
│⠀⠀⠐⠒⡿⠿⠟⡻⠛⠛⠛⠚⠉⠀⠲⣶⣿⣷⣾⣷⣿⣿⣷⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣶⣶⣷⡇⠀⠀│
│⠀⠀⠀⠀⢱⡠⠊⠀⠀⣤⣤⣶⣿⣿⠿⣿⣿⣿⣿⣿⣟⣿⣿⣟⣟⣉⣉⣝⣛⣫⣭⣥⣤⣤⣔⣒⣒⡃⠀⠀│
│⠀⠀⠀⠀⠀⠁⠀⢀⡀⠀⢯⠒⠉⡠⠔⠡⠤⠮⣤⣴⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣿⣶⣶⣶⣶⣿⣿⠥⠤⠄│
│⠀⠀⠀⠀⣀⣠⣄⣤⣭⣿⣿⣿⣿⣿⣿⣿⣿⣯⣯⣯⣭⣽⣿⣿⣿⣿⣿⣯⣭⣍⡙⠛⠝⠉⠀⠀⠀⠀⠀⠀│
│⠀⠀⠠⠴⠿⢯⠭⠿⠯⣭⣷⡾⠿⠿⠿⣿⢿⡿⡻⠿⡟⠛⢛⣛⠯⠭⠒⠒⠉⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⣀⣀⣄⠀⠀⠀⢀⣀⡠⠤⠔⠒⠊⠉⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
0 │⠀⠀⢀⣀⣴⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣻⣷⣄⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⣀⡀│
└────────────────────────────────────────┘
4 8
sepal_length

20
test/fixtures/iris-scatter.txt vendored Normal file
View File

@@ -0,0 +1,20 @@
IRIS-SCATTER
┌────────────────────────────────────────┐
6.9 │⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠄⠃⠀⠀│ sepal_width
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡀⠀⠀⠀⠀⠀⠀⠀⠀⡄⠂⠄⠀⠀⠄⠀⠁│ petal_length
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⠀⠀⡀⡀⠂⠀⡆⠁⠄⠈⠀⠂⠀⠀⠀⠀⠀⠀⠀│ petal_width
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡀⡀⡀⠀⠂⡀⠃⡅⠀⠄⠁⡂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│ species
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠐⠀⠁⠀⠄⢕⠀⠄⡃⠀⠀⠀⡁⠄⠂⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠁⠀⠀⠀⠀⠁⣊⠀⡃⠀⡀⠉⠀⠅⠂⠅⠙⠀⠂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⠀⢅⠀⡄⡮⠀⠅⠇⠀⠒⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠄⠀⠄│
│⠀⠀⠀⠀⠀⠨⠀⠠⠀⡀⣷⠀⠆⠀⠄⠊⠀⠂⠀⠀⠄⠀⠄⡄⠀⠀⠀⡀⠀⠀⠀⠀⠁⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠐⠀⠇⠘⠀⠑⠀⠂⠓⠀⠀⠀⠂⢰⠀⡆⡀⠃⢶⠀⡄⡄⡇⡳⠀⠂⡃⠃⠑⠀⠃⠄⡀⠀⠂⡂⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⡄⠠⠀⠁⠀⠀⡮⠀⠆⡃⠀⠑⠀⠀⡅⠁⠀⠀⡄⠀⠀⠀⠀⠄⠀⠀⠀⠀⠂⠀⠀│
│⠀⠀⠀⠀⠀⠁⠀⢀⠀⠀⢅⠀⠀⠀⠀⠡⠀⠄⡀⠀⠁⠀⠁⡁⡃⠅⠀⠃⠃⠃⠐⠀⠀⠀⡀⠀⠂⠅⠀⠄│
│⠀⠀⠀⠀⡀⢠⠀⢤⠀⡆⣴⠀⡤⠀⠆⡠⠀⠆⠀⠅⢍⠀⠅⠅⠅⢅⠀⡇⡀⠄⡀⠀⠅⠁⠀⠀⠀⠀⠀⠀│
│⠀⠀⠠⠀⠁⢁⠀⠁⠀⡀⡣⠀⠀⠀⠁⡯⠀⡃⡂⠀⡘⠀⠁⠁⠁⠈⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
│⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀│
0 │⠀⠀⢀⠀⡄⣲⠀⣴⠀⡄⣿⠀⣤⠀⡅⣄⠀⡃⡄⡀⣀⠀⡀⡀⡀⣀⠀⡀⡀⡀⣀⠀⡀⡀⡀⠀⡀⡀⠀⡀│
└────────────────────────────────────────┘
4 8
sepal_length

View File

@@ -1,6 +1,8 @@
# frozen_string_literal: true
require 'simplecov'
SimpleCov.start
require 'uplot'
require 'youplot'
require 'test/unit'

View File

@@ -1,4 +0,0 @@
require_relative '../test_helper.rb'
class UplotCommandTest < Test::Unit::TestCase
end

View File

@@ -1,4 +0,0 @@
require_relative '../test_helper.rb'
class UplotPlotTest < Test::Unit::TestCase
end

View File

@@ -1,7 +0,0 @@
require 'test_helper'
class UplotTest < Test::Unit::TestCase
def test_that_it_has_a_version_number
assert_kind_of String, ::Uplot::VERSION
end
end

View File

@@ -0,0 +1,11 @@
# frozen_string_literal: true
require_relative '../../test_helper'
class YouPlotCommandTest < Test::Unit::TestCase
test :count_values do
@m = YouPlot::Backends::Processing
assert_equal([%i[a b c], [3, 2, 1]], @m.count_values(%i[a a a b b c]))
assert_equal([%i[c b a], [3, 2, 1]], @m.count_values(%i[a b b c c c]))
end
end

View File

@@ -0,0 +1,6 @@
# frozen_string_literal: true
require_relative '../../test_helper'
class YouPlotPlotTest < Test::Unit::TestCase
end

View File

@@ -0,0 +1,102 @@
# frozen_string_literal: true
require 'tempfile'
require_relative '../test_helper'
class YouPlotCommandTest < Test::Unit::TestCase
class << self
def startup
@stdin = $stdin.dup
@stderr = $stderr.dup
end
def shutdown
$stdin = @stdin
$stderr = @stderr
end
end
def setup
$stdin = File.open(File.expand_path('../fixtures/iris.csv', __dir__), 'r')
@tmp_file = Tempfile.new
$stderr = @tmp_file
end
def teardown
@tmp_file.close
end
def fixture(fname)
File.read(File.expand_path("../fixtures/#{fname}", __dir__))
end
test :bar do
YouPlot::Command.new(['bar', '-H', '-d,', '-t', 'IRIS-BARPLOT']).run
assert_equal fixture('iris-barplot.txt'), @tmp_file.read
end
test :barplot do
YouPlot::Command.new(['barplot', '-H', '-d,', '-t', 'IRIS-BARPLOT']).run
assert_equal fixture('iris-barplot.txt'), @tmp_file.read
end
test :hist do
YouPlot::Command.new(['hist', '-H', '-d,', '-t', 'IRIS-HISTOGRAM']).run
assert_equal fixture('iris-histogram.txt'), @tmp_file.read
end
test :histogram do
YouPlot::Command.new(['histogram', '-H', '-d,', '-t', 'IRIS-HISTOGRAM']).run
assert_equal fixture('iris-histogram.txt'), @tmp_file.read
end
test :line do
YouPlot::Command.new(['line', '-H', '-d,', '-t', 'IRIS-LINEPLOT']).run
assert_equal fixture('iris-lineplot.txt'), @tmp_file.read
end
test :lineplot do
YouPlot::Command.new(['lineplot', '-H', '-d,', '-t', 'IRIS-LINEPLOT']).run
assert_equal fixture('iris-lineplot.txt'), @tmp_file.read
end
test :lines do
YouPlot::Command.new(['lines', '-H', '-d,', '-t', 'IRIS-LINEPLOTS']).run
assert_equal fixture('iris-lineplots.txt'), @tmp_file.read
end
test :lineplots do
YouPlot::Command.new(['lineplots', '-H', '-d,', '-t', 'IRIS-LINEPLOTS']).run
assert_equal fixture('iris-lineplots.txt'), @tmp_file.read
end
test :s do
YouPlot::Command.new(['s', '-H', '-d,', '-t', 'IRIS-SCATTER']).run
assert_equal fixture('iris-scatter.txt'), @tmp_file.read
end
test :scatter do
YouPlot::Command.new(['scatter', '-H', '-d,', '-t', 'IRIS-SCATTER']).run
assert_equal fixture('iris-scatter.txt'), @tmp_file.read
end
test :d do
YouPlot::Command.new(['d', '-H', '-d,', '-t', 'IRIS-DENSITY']).run
assert_equal fixture('iris-density.txt'), @tmp_file.read
end
test :density do
YouPlot::Command.new(['density', '-H', '-d,', '-t', 'IRIS-DENSITY']).run
assert_equal fixture('iris-density.txt'), @tmp_file.read
end
test :box do
YouPlot::Command.new(['box', '-H', '-d,', '-t', 'IRIS-BOXPLOT']).run
assert_equal fixture('iris-boxplot.txt'), @tmp_file.read
end
test :boxplot do
YouPlot::Command.new(['boxplot', '-H', '-d,', '-t', 'IRIS-BOXPLOT']).run
assert_equal fixture('iris-boxplot.txt'), @tmp_file.read
end
end

View File

@@ -1,8 +1,10 @@
require_relative '../test_helper.rb'
# frozen_string_literal: true
class UplotPreprocessingTest < Test::Unit::TestCase
require_relative '../test_helper'
class YouPlotDSVReaderTest < Test::Unit::TestCase
def setup
@m = Uplot::Preprocessing
@m = YouPlot::DSVReader
end
test :transpose2 do
@@ -122,9 +124,4 @@ class UplotPreprocessingTest < Test::Unit::TestCase
[2, 4],
[3, 5, 6]], false, false))
end
test :count do
assert_equal([%i[a b c], [3, 2, 1]], @m.count(%i[a a a b b c]))
assert_equal([%i[c b a], [3, 2, 1]], @m.count(%i[a b b c c c]))
end
end

9
test/youplot_test.rb Normal file
View File

@@ -0,0 +1,9 @@
# frozen_string_literal: true
require_relative 'test_helper'
class YouPlotTest < Test::Unit::TestCase
def test_that_it_has_a_version_number
assert_kind_of String, ::YouPlot::VERSION
end
end

View File

@@ -1,8 +1,10 @@
require_relative 'lib/uplot/version'
# frozen_string_literal: true
require_relative 'lib/youplot/version'
Gem::Specification.new do |spec|
spec.name = 'u-plot'
spec.version = Uplot::VERSION
spec.name = 'youplot'
spec.version = YouPlot::VERSION
spec.authors = ['kojix2']
spec.email = ['2xijok@gmail.com']
@@ -11,13 +13,13 @@ Gem::Specification.new do |spec|
Create ASCII charts on the terminal with data from standard streams in the
pipeline.
MSG
spec.homepage = 'https://github.com/kojix2/uplot'
spec.homepage = 'https://github.com/kojix2/youplot'
spec.license = 'MIT'
spec.required_ruby_version = Gem::Requirement.new('>= 2.3.0')
spec.required_ruby_version = Gem::Requirement.new('>= 2.4.0')
spec.files = Dir['*.{md,txt}', '{lib,exe}/**/*']
spec.bindir = 'exe'
spec.executables = ['uplot']
spec.executables = ['uplot', 'youplot']
spec.require_paths = ['lib']
spec.add_runtime_dependency 'unicode_plot'