YouPlot/lib/uplot/command.rb

324 lines
10 KiB
Ruby
Raw Normal View History

require 'optparse'
require 'csv'
2020-08-03 13:57:00 +08:00
require 'unicode_plot'
2020-07-29 16:01:39 +08:00
module Uplot
class Command
2020-08-15 15:46:03 +08:00
attr_accessor :params
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-07-29 16:01:39 +08:00
def initialize(argv)
2020-08-15 15:46:03 +08:00
@params = Params.new
2020-08-03 13:57:00 +08:00
@ptype = nil
@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'
@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
opt.on('-o', '--output', TrueClass) { |v| @output = v }
.on('-d', '--delimiter VAL', String) { |v| @delimiter = v }
.on('-H', '--headers', TrueClass) { |v| @headers = v }
.on('-T', '--transpose', TrueClass) { |v| @transpose = v }
2020-08-15 15:46:03 +08:00
.on('-t', '--title VAL', String) { |v| params.title = v }
.on('-w', '--width VAL', Numeric) { |v| params.width = v }
.on('-h', '--height VAL', Numeric) { |v| params.height = v }
.on('-b', '--border VAL', Numeric) { |v| params.border = v }
.on('-m', '--margin VAL', Numeric) { |v| params.margin = v }
.on('-p', '--padding VAL', Numeric) { |v| params.padding = v }
.on('-c', '--color VAL', String) { |v| params.color = v.to_sym }
.on('-x', '--xlabel VAL', String) { |v| params.xlabel = v }
.on('-y', '--ylabel VAL', String) { |v| params.ylabel = v }
.on('-l', '--labels', TrueClass) { |v| params.labels = v }
2020-08-14 12:47:04 +08:00
.on('--fmt VAL', String) { |v| @fmt = v }
.on('--debug', TrueClass) { |v| @debug = v }
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-03 13:57:00 +08:00
main_parser = create_parser
parsers = Hash.new { |h, k| h[k] = create_parser }
parsers['barplot'] = parsers['bar']
2020-08-15 15:46:03 +08:00
.on('--symbol VAL', String) { |v| params.symbol = v }
.on('--xscale VAL', String) { |v| params.xscale = v }
2020-08-04 20:25:19 +08:00
.on('--count', TrueClass) { |v| @count = v }
2020-08-03 13:57:00 +08:00
parsers['count'] = parsers['c'] # barplot -c
2020-08-15 15:46:03 +08:00
.on('--symbol VAL', String) { |v| params.symbol = v }
2020-07-31 12:41:12 +08:00
parsers['histogram'] = parsers['hist']
2020-08-15 15:46:03 +08:00
.on('-n', '--nbins VAL', Numeric) { |v| params.nbins = v }
.on('--closed VAL', String) { |v| params.closed = v }
.on('--symbol VAL', String) { |v| params.symbol = v }
2020-08-03 13:57:00 +08:00
parsers['lineplot'] = parsers['line']
2020-08-15 15:46:03 +08:00
.on('--canvas VAL', String) { |v| params.canvas = v }
.on('--xlim VAL', String) { |v| params.xlim = get_lim(v) }
.on('--ylim VAL', String) { |v| params.ylim = get_lim(v) }
2020-08-03 13:57:00 +08:00
parsers['lineplots'] = parsers['lines']
2020-08-15 15:46:03 +08:00
.on('--canvas VAL', String) { |v| params.canvas = v }
.on('--xlim VAL', String) { |v| params.xlim = get_lim(v) }
.on('--ylim VAL', String) { |v| params.ylim = get_lim(v) }
2020-08-04 20:25:19 +08:00
parsers['scatter'] = parsers['s']
2020-08-15 15:46:03 +08:00
.on('--canvas VAL', String) { |v| params.canvas = v }
.on('--xlim VAL', String) { |v| params.xlim = get_lim(v) }
.on('--ylim VAL', String) { |v| params.ylim = get_lim(v) }
2020-08-04 20:25:19 +08:00
parsers['density'] = parsers['d']
2020-08-15 15:46:03 +08:00
.on('--grid', TrueClass) { |v| params.grid = v }
.on('--xlim VAL', String) { |v| params.xlim = get_lim(v) }
.on('--ylim VAL', String) { |v| params.ylim = get_lim(v) }
2020-08-03 13:57:00 +08:00
parsers['boxplot'] = parsers['box']
2020-08-15 15:46:03 +08:00
.on('--xlim VAL', String) { |v| params.xlim = get_lim(v) }
2020-08-03 13:57:00 +08:00
parsers.default = nil
2020-07-30 11:44:31 +08:00
main_parser.banner = <<~MSG
2020-08-14 10:45:47 +08:00
Program: uplot (Tools for plotting on the terminal)
2020-07-31 13:48:13 +08:00
Version: #{Uplot::VERSION} (using unicode_plot #{UnicodePlot::VERSION})
Usage: uplot <command> [options]
Command: #{parsers.keys.join(' ')}
2020-08-14 10:45:47 +08:00
Options:
2020-07-30 10:48:27 +08:00
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-07-30 10:48:27 +08:00
@ptype = argv.shift
2020-07-29 16:01:39 +08:00
2020-07-30 11:44:31 +08:00
unless parsers.has_key?(@ptype)
2020-08-14 10:45:47 +08:00
if @ptype.nil?
warn main_parser.help
else
warn "uplot: unrecognized command '#{@ptype}'"
end
2020-07-30 11:44:31 +08:00
exit 1
end
parser = parsers[@ptype]
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)
input.freeze
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-07-30 09:37:20 +08:00
case @ptype
2020-08-03 10:18:27 +08:00
when 'bar', 'barplot'
barplot(data, headers)
when 'count', 'c'
@count = true
barplot(data, headers)
2020-07-30 09:37:20 +08:00
when 'hist', 'histogram'
2020-07-30 14:08:09 +08:00
histogram(data, headers)
2020-07-30 09:37:20 +08:00
when 'line', 'lineplot'
2020-07-30 14:08:09 +08:00
line(data, headers)
2020-08-03 10:18:27 +08:00
when 'lines', 'lineplots'
2020-07-30 14:08:09 +08:00
lines(data, headers)
2020-07-30 14:54:08 +08:00
when 'scatter', 'scatterplot'
scatter(data, headers)
2020-08-02 23:36:13 +08:00
when 'density'
density(data, headers)
2020-08-03 10:18:27 +08:00
when 'box', 'boxplot'
boxplot(data, headers)
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)
data = CSV.parse(input, col_sep: @delimiter)
2020-08-03 09:38:41 +08:00
data.delete([]) # Remove blank lines.
2020-08-03 11:39:27 +08:00
data.delete_if { |i| i.all? nil } # Room for improvement.
p parsed_csv: data if @debug
2020-08-15 14:45:46 +08:00
headers = get_headers(data)
data = get_data(data)
[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)
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)
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)
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