2020-07-30 12:42:51 +08:00
|
|
|
require 'optparse'
|
|
|
|
require 'csv'
|
2020-08-03 13:57:00 +08:00
|
|
|
require 'unicode_plot'
|
2020-07-30 12:42:51 +08:00
|
|
|
|
2020-07-29 16:01:39 +08:00
|
|
|
module Uplot
|
|
|
|
class Command
|
2020-08-15 15:46:03 +08:00
|
|
|
Params = Struct.new(
|
|
|
|
:title,
|
|
|
|
:width,
|
|
|
|
:height,
|
|
|
|
:border,
|
|
|
|
:margin,
|
|
|
|
:padding,
|
|
|
|
:color,
|
|
|
|
:xlabel,
|
|
|
|
:ylabel,
|
|
|
|
:labels,
|
|
|
|
:symbol,
|
|
|
|
:xscale,
|
|
|
|
:nbins,
|
|
|
|
:closed,
|
|
|
|
:canvas,
|
|
|
|
:xlim,
|
|
|
|
:ylim,
|
|
|
|
:grid
|
|
|
|
) do
|
|
|
|
def to_hc
|
|
|
|
to_h.compact
|
|
|
|
end
|
|
|
|
end
|
2020-08-15 16:10:41 +08:00
|
|
|
|
|
|
|
attr_accessor :params, :plot_type
|
2020-08-15 18:41:53 +08:00
|
|
|
attr_reader :raw_inputs
|
2020-08-15 16:10:41 +08:00
|
|
|
|
2020-07-29 16:01:39 +08:00
|
|
|
def initialize(argv)
|
2020-08-15 15:46:03 +08:00
|
|
|
@params = Params.new
|
|
|
|
|
2020-08-15 16:10:41 +08:00
|
|
|
@plot_type = nil
|
2020-08-03 13:57:00 +08:00
|
|
|
@headers = nil
|
2020-07-30 14:08:09 +08:00
|
|
|
@delimiter = "\t"
|
2020-07-31 16:35:21 +08:00
|
|
|
@transpose = false
|
2020-08-03 13:57:00 +08:00
|
|
|
@output = false
|
|
|
|
@count = false
|
|
|
|
@fmt = 'xyy'
|
2020-08-15 18:41:53 +08:00
|
|
|
|
|
|
|
@raw_inputs = []
|
|
|
|
@debug = false
|
|
|
|
|
2020-07-29 16:01:39 +08:00
|
|
|
parse_options(argv)
|
|
|
|
end
|
|
|
|
|
2020-07-30 15:33:08 +08:00
|
|
|
def create_parser
|
2020-08-14 12:47:04 +08:00
|
|
|
OptionParser.new do |opt|
|
|
|
|
opt.program_name = 'uplot'
|
|
|
|
opt.version = Uplot::VERSION
|
2020-08-15 16:48:25 +08:00
|
|
|
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.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
|
2020-08-14 12:47:04 +08:00
|
|
|
end
|
2020-07-30 11:44:31 +08:00
|
|
|
end
|
2020-07-29 16:01:39 +08:00
|
|
|
|
2020-07-30 11:44:31 +08:00
|
|
|
def parse_options(argv)
|
2020-08-15 16:48:25 +08:00
|
|
|
main_parser = create_parser
|
|
|
|
parsers = Hash.new { |h, k| h[k] = create_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
|
|
|
|
|
|
|
|
# Preventing the generation of new sub-commands
|
|
|
|
parsers.default = nil
|
|
|
|
|
|
|
|
# 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: #{parsers.keys.join(' ')}
|
|
|
|
|
|
|
|
Options:
|
|
|
|
MSG
|
2020-08-14 10:45:47 +08:00
|
|
|
|
|
|
|
begin
|
|
|
|
main_parser.order!(argv)
|
|
|
|
rescue OptionParser::ParseError => e
|
|
|
|
warn "uplot: #{e.message}"
|
|
|
|
exit 1
|
|
|
|
end
|
|
|
|
|
2020-08-15 16:10:41 +08:00
|
|
|
@plot_type = argv.shift&.to_sym
|
2020-07-29 16:01:39 +08:00
|
|
|
|
2020-08-15 16:10:41 +08:00
|
|
|
unless parsers.has_key?(plot_type)
|
|
|
|
if plot_type.nil?
|
2020-08-14 10:45:47 +08:00
|
|
|
warn main_parser.help
|
|
|
|
else
|
2020-08-15 16:10:41 +08:00
|
|
|
warn "uplot: unrecognized command '#{plot_type}'"
|
2020-08-14 10:45:47 +08:00
|
|
|
end
|
2020-07-30 11:44:31 +08:00
|
|
|
exit 1
|
|
|
|
end
|
2020-08-15 16:10:41 +08:00
|
|
|
parser = parsers[plot_type]
|
2020-08-14 10:45:47 +08:00
|
|
|
|
|
|
|
begin
|
|
|
|
parser.parse!(argv) unless argv.empty?
|
|
|
|
rescue OptionParser::ParseError => e
|
|
|
|
warn "uplot: #{e.message}"
|
|
|
|
exit 1
|
|
|
|
end
|
2020-07-30 10:48:27 +08:00
|
|
|
end
|
|
|
|
|
2020-07-29 16:01:39 +08:00
|
|
|
def run
|
2020-07-30 09:37:20 +08:00
|
|
|
# Sometimes the input file does not end with a newline code.
|
|
|
|
while input = Kernel.gets(nil)
|
2020-07-30 12:42:51 +08:00
|
|
|
input.freeze
|
2020-08-15 18:41:53 +08:00
|
|
|
@raw_inputs << input
|
2020-07-30 14:08:09 +08:00
|
|
|
data, headers = preprocess(input)
|
2020-08-03 09:36:36 +08:00
|
|
|
pp input: input, data: data, headers: headers if @debug
|
2020-08-15 16:10:41 +08:00
|
|
|
case plot_type
|
|
|
|
when :bar, :barplot
|
2020-08-03 10:18:27 +08:00
|
|
|
barplot(data, headers)
|
2020-08-15 16:10:41 +08:00
|
|
|
when :count, :c
|
2020-08-03 10:18:27 +08:00
|
|
|
@count = true
|
|
|
|
barplot(data, headers)
|
2020-08-15 16:10:41 +08:00
|
|
|
when :hist, :histogram
|
2020-07-30 14:08:09 +08:00
|
|
|
histogram(data, headers)
|
2020-08-15 16:10:41 +08:00
|
|
|
when :line, :lineplot
|
2020-07-30 14:08:09 +08:00
|
|
|
line(data, headers)
|
2020-08-15 16:10:41 +08:00
|
|
|
when :lines, :lineplots
|
2020-07-30 14:08:09 +08:00
|
|
|
lines(data, headers)
|
2020-08-15 16:10:41 +08:00
|
|
|
when :scatter, :scatterplot
|
2020-07-30 14:54:08 +08:00
|
|
|
scatter(data, headers)
|
2020-08-15 16:10:41 +08:00
|
|
|
when :density
|
2020-08-02 23:36:13 +08:00
|
|
|
density(data, headers)
|
2020-08-15 16:10:41 +08:00
|
|
|
when :box, :boxplot
|
2020-08-03 10:18:27 +08:00
|
|
|
boxplot(data, headers)
|
2020-08-15 16:10:41 +08:00
|
|
|
else
|
|
|
|
raise "unrecognized plot_type: #{plot_type}"
|
2020-07-30 09:37:20 +08:00
|
|
|
end.render($stderr)
|
2020-07-29 16:01:39 +08:00
|
|
|
|
2020-07-30 11:47:02 +08:00
|
|
|
print input if @output
|
2020-07-30 09:37:20 +08:00
|
|
|
end
|
2020-07-29 16:01:39 +08:00
|
|
|
end
|
|
|
|
|
2020-08-15 14:45:46 +08:00
|
|
|
# Transpose different sized ruby arrays
|
2020-08-03 14:10:59 +08:00
|
|
|
# https://stackoverflow.com/q/26016632
|
2020-08-15 14:45:46 +08:00
|
|
|
def transpose2(arr)
|
2020-08-01 21:57:49 +08:00
|
|
|
Array.new(arr.map(&:length).max) { |i| arr.map { |e| e[i] } }
|
2020-07-31 17:54:40 +08:00
|
|
|
end
|
|
|
|
|
2020-07-30 14:08:09 +08:00
|
|
|
def preprocess(input)
|
2020-08-15 18:42:18 +08:00
|
|
|
arr = CSV.parse(input, col_sep: @delimiter)
|
|
|
|
# Remove blank lines.
|
|
|
|
arr.delete([])
|
|
|
|
# Remove rows where all elements are nil
|
|
|
|
arr.delete_if { |i| i.all? nil }
|
|
|
|
p parsed_csv: arr if @debug
|
|
|
|
headers = get_headers(arr)
|
|
|
|
data = get_data(arr)
|
2020-08-15 14:45:46 +08:00
|
|
|
[data, headers]
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_headers(data)
|
|
|
|
if @headers
|
|
|
|
if @transpose
|
|
|
|
data.map(&:first)
|
|
|
|
else
|
|
|
|
data[0]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_data(data)
|
2020-07-31 17:54:40 +08:00
|
|
|
if @transpose
|
|
|
|
if @headers
|
2020-08-15 14:45:46 +08:00
|
|
|
data.map { |row| row[1..-1] }
|
|
|
|
else
|
|
|
|
data
|
2020-07-31 17:54:40 +08:00
|
|
|
end
|
2020-07-30 14:08:09 +08:00
|
|
|
else
|
2020-08-15 14:45:46 +08:00
|
|
|
if @headers
|
|
|
|
transpose2(data[1..-1])
|
|
|
|
else
|
2020-08-15 15:01:43 +08:00
|
|
|
transpose2(data)
|
2020-08-15 14:45:46 +08:00
|
|
|
end
|
2020-07-30 14:08:09 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-08-03 09:24:40 +08:00
|
|
|
def preprocess_count(data)
|
2020-08-15 15:02:31 +08:00
|
|
|
# tally was added in Ruby 2.7
|
2020-08-03 14:10:59 +08:00
|
|
|
if Enumerable.method_defined? :tally
|
|
|
|
data[0].tally
|
2020-08-15 15:02:31 +08:00
|
|
|
else
|
|
|
|
# https://github.com/marcandre/backports
|
2020-08-03 14:10:59 +08:00
|
|
|
data[0].each_with_object(Hash.new(0)) { |item, res| res[item] += 1 }
|
|
|
|
.tap { |h| h.default = nil }
|
2020-08-15 15:02:31 +08:00
|
|
|
end
|
|
|
|
.sort { |a, b| a[1] <=> b[1] }
|
|
|
|
.reverse
|
|
|
|
.transpose
|
2020-08-03 09:24:40 +08:00
|
|
|
end
|
|
|
|
|
2020-07-30 14:54:08 +08:00
|
|
|
def barplot(data, headers)
|
2020-08-03 09:24:40 +08:00
|
|
|
data = preprocess_count(data) if @count
|
2020-08-15 15:46:03 +08:00
|
|
|
params.title ||= headers[1] if headers
|
|
|
|
UnicodePlot.barplot(data[0], data[1].map(&:to_f), **params.to_hc)
|
2020-07-30 14:54:08 +08:00
|
|
|
end
|
|
|
|
|
2020-07-30 14:08:09 +08:00
|
|
|
def histogram(data, headers)
|
2020-08-15 15:46:03 +08:00
|
|
|
params.title ||= headers[0] if headers # labels?
|
2020-07-30 14:08:09 +08:00
|
|
|
series = data[0].map(&:to_f)
|
2020-08-15 15:46:03 +08:00
|
|
|
UnicodePlot.histogram(series, **params.to_hc)
|
2020-07-29 16:01:39 +08:00
|
|
|
end
|
2020-07-29 17:24:08 +08:00
|
|
|
|
2020-07-31 12:41:12 +08:00
|
|
|
def get_lim(str)
|
|
|
|
str.split(/-|:|\.\./)[0..1].map(&:to_f)
|
|
|
|
end
|
|
|
|
|
2020-07-30 14:08:09 +08:00
|
|
|
def line(data, headers)
|
2020-07-30 12:42:51 +08:00
|
|
|
if data.size == 1
|
2020-08-15 15:46:03 +08:00
|
|
|
params.ylabel ||= headers[0] if headers
|
2020-08-14 09:36:27 +08:00
|
|
|
y = data[0].map(&:to_f)
|
2020-08-15 15:46:03 +08:00
|
|
|
UnicodePlot.lineplot(y, **params.to_hc)
|
2020-07-30 12:42:51 +08:00
|
|
|
else
|
2020-08-15 15:46:03 +08:00
|
|
|
params.xlabel ||= headers[0] if headers
|
|
|
|
params.ylabel ||= headers[1] if headers
|
2020-08-14 09:36:27 +08:00
|
|
|
x = data[0].map(&:to_f)
|
|
|
|
y = data[1].map(&:to_f)
|
2020-08-15 15:46:03 +08:00
|
|
|
UnicodePlot.lineplot(x, y, **params.to_hc)
|
2020-07-29 17:24:08 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-08-04 19:13:34 +08:00
|
|
|
def get_method2(method1)
|
|
|
|
(method1.to_s + '!').to_sym
|
2020-08-04 14:18:40 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def xyy_plot(data, headers, method1) # improve method name
|
|
|
|
method2 = get_method2(method1)
|
2020-07-30 14:54:08 +08:00
|
|
|
data.map! { |series| series.map(&:to_f) }
|
2020-08-15 15:46:03 +08:00
|
|
|
params.name ||= headers[1] if headers
|
|
|
|
params.xlabel ||= headers[0] if headers
|
|
|
|
params.ylim ||= data[1..-1].flatten.minmax # need?
|
|
|
|
plot = UnicodePlot.public_send(method1, data[0], data[1], **params.to_hc)
|
2020-07-30 12:42:51 +08:00
|
|
|
2.upto(data.size - 1) do |i|
|
2020-08-04 14:18:40 +08:00
|
|
|
UnicodePlot.public_send(method2, plot, data[0], data[i], name: headers[i])
|
2020-07-29 17:24:08 +08:00
|
|
|
end
|
|
|
|
plot
|
|
|
|
end
|
2020-07-30 14:54:08 +08:00
|
|
|
|
2020-08-04 14:18:40 +08:00
|
|
|
def xyxy_plot(data, headers, method1) # improve method name
|
|
|
|
method2 = get_method2(method1)
|
2020-08-03 11:39:27 +08:00
|
|
|
data.map! { |series| series.map(&:to_f) }
|
|
|
|
data = data.each_slice(2).to_a
|
2020-08-15 15:46:03 +08:00
|
|
|
params.name ||= headers[0] if headers
|
|
|
|
params.xlim = data.map(&:first).flatten.minmax
|
|
|
|
params.ylim = data.map(&:last).flatten.minmax
|
2020-08-03 11:39:27 +08:00
|
|
|
x1, y1 = data.shift
|
2020-08-15 15:46:03 +08:00
|
|
|
plot = UnicodePlot.public_send(method1, x1, y1, **params.to_hc)
|
2020-08-03 11:39:27 +08:00
|
|
|
data.each_with_index do |(xi, yi), i|
|
2020-08-04 14:18:40 +08:00
|
|
|
UnicodePlot.public_send(method2, plot, xi, yi, name: headers[(i + 1) * 2])
|
2020-08-03 11:39:27 +08:00
|
|
|
end
|
|
|
|
plot
|
|
|
|
end
|
|
|
|
|
2020-08-03 10:33:14 +08:00
|
|
|
def lines(data, headers)
|
2020-08-03 11:39:27 +08:00
|
|
|
case @fmt
|
|
|
|
when 'xyy'
|
2020-08-04 14:18:40 +08:00
|
|
|
xyy_plot(data, headers, :lineplot)
|
2020-08-03 11:39:27 +08:00
|
|
|
when 'xyxy'
|
2020-08-04 14:18:40 +08:00
|
|
|
xyxy_plot(data, headers, :lineplot)
|
2020-08-03 10:33:14 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-07-30 14:54:08 +08:00
|
|
|
def scatter(data, headers)
|
2020-08-03 11:39:27 +08:00
|
|
|
case @fmt
|
|
|
|
when 'xyy'
|
2020-08-04 14:18:40 +08:00
|
|
|
xyy_plot(data, headers, :scatterplot)
|
2020-08-03 11:39:27 +08:00
|
|
|
when 'xyxy'
|
2020-08-04 14:18:40 +08:00
|
|
|
xyxy_plot(data, headers, :scatterplot)
|
2020-07-30 14:54:08 +08:00
|
|
|
end
|
|
|
|
end
|
2020-07-30 15:33:08 +08:00
|
|
|
|
2020-08-02 23:36:13 +08:00
|
|
|
def density(data, headers)
|
2020-08-03 11:39:27 +08:00
|
|
|
case @fmt
|
|
|
|
when 'xyy'
|
2020-08-04 14:18:40 +08:00
|
|
|
xyy_plot(data, headers, :densityplot)
|
2020-08-03 11:39:27 +08:00
|
|
|
when 'xyxy'
|
2020-08-04 14:18:40 +08:00
|
|
|
xyxy_plot(data, headers, :densityplot)
|
2020-08-02 23:36:13 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-07-30 15:33:08 +08:00
|
|
|
def boxplot(data, headers)
|
2020-08-04 19:39:42 +08:00
|
|
|
headers ||= (1..data.size).map(&:to_s)
|
2020-07-30 15:33:08 +08:00
|
|
|
data.map! { |series| series.map(&:to_f) }
|
2020-08-15 15:46:03 +08:00
|
|
|
UnicodePlot.boxplot(headers, data, params.to_hc)
|
2020-07-30 15:33:08 +08:00
|
|
|
end
|
2020-07-29 16:01:39 +08:00
|
|
|
end
|
|
|
|
end
|