YouPlot/lib/youplot/command/parser.rb

278 lines
10 KiB
Ruby
Raw Normal View History

2020-09-18 23:08:09 +08:00
# frozen_string_literal: true
2020-09-17 15:48:03 +08:00
require 'optparse'
require_relative 'params'
2020-11-23 12:09:16 +08:00
module YouPlot
2020-09-17 09:06:31 +08:00
class Command
class Parser
attr_reader :command, :params,
2020-11-06 09:21:36 +08:00
:delimiter, :transpose, :headers, :pass, :output, :fmt,
2020-11-23 22:52:14 +08:00
:color_names, :encoding, :debug
2020-09-17 09:06:31 +08:00
def initialize
2020-11-06 09:21:36 +08:00
@command = nil
@params = Params.new
@delimiter = "\t"
@transpose = false
@headers = nil
@pass = false
@output = $stderr
@fmt = 'xyy'
2020-11-23 22:52:14 +08:00
@encoding = nil
2020-11-06 09:21:36 +08:00
@debug = false
@color_names = false
2020-09-17 09:45:44 +08:00
end
2020-09-17 09:06:31 +08:00
def create_default_parser
OptionParser.new do |opt|
2020-11-23 12:09:16 +08:00
opt.program_name = 'YouPlot'
opt.version = YouPlot::VERSION
2020-10-12 15:14:51 +08:00
opt.summary_width = 24
opt.on_tail('') # Add a blank line at the end
2020-10-12 15:45:28 +08:00
opt.separator('')
2020-11-14 23:29:20 +08:00
opt.on('Common options:')
2020-10-11 07:29:02 +08:00
opt.on('-O', '--pass [VAL]', 'file to output standard input data to [stdout]',
2020-11-23 12:09:16 +08:00
'for inserting YouPlot in the middle of Unix pipes') do |v|
2020-09-29 17:13:03 +08:00
@pass = v || $stdout
end
opt.on('-o', '--output VAL', 'file to output results to [stderr]') do |v|
2020-09-17 09:06:31 +08:00
@output = v
end
2020-10-10 23:11:54 +08:00
opt.on('-d', '--delimiter VAL', String, 'use DELIM instead of TAB for field delimiter') do |v|
2020-09-17 09:06:31 +08:00
@delimiter = v
end
2020-10-10 23:11:54 +08:00
opt.on('-H', '--headers', TrueClass, 'specify that the input has header row') do |v|
2020-09-17 09:06:31 +08:00
@headers = v
end
2020-11-06 08:56:35 +08:00
opt.on('-T', '--transpose', TrueClass, 'transpose the axes of the input data') do |v|
2020-09-17 09:06:31 +08:00
@transpose = v
end
2020-10-10 23:11:54 +08:00
opt.on('-t', '--title VAL', String, 'print string on the top of plot') do |v|
2020-09-17 09:06:31 +08:00
params.title = v
end
2020-10-10 23:11:54 +08:00
opt.on('-x', '--xlabel VAL', String, 'print string on the bottom of the plot') do |v|
2020-09-17 09:06:31 +08:00
params.xlabel = v
end
2020-10-10 23:11:54 +08:00
opt.on('-y', '--ylabel VAL', String, 'print string on the far left of the plot') do |v|
2020-09-17 09:06:31 +08:00
params.ylabel = v
end
2020-10-10 23:11:54 +08:00
opt.on('-w', '--width VAL', Integer, 'number of characters per row') do |v|
2020-09-17 09:06:31 +08:00
params.width = v
end
2020-10-10 23:11:54 +08:00
opt.on('-h', '--height VAL', Numeric, 'number of rows') do |v|
2020-09-17 09:06:31 +08:00
params.height = v
end
2020-10-10 23:11:54 +08:00
opt.on('-b', '--border VAL', String, 'specify the style of the bounding box') do |v|
2020-09-17 09:06:31 +08:00
params.border = v.to_sym
end
2020-10-10 23:11:54 +08:00
opt.on('-m', '--margin VAL', Numeric, 'number of spaces to the left of the plot') do |v|
2020-09-17 09:06:31 +08:00
params.margin = v
end
2020-10-10 23:11:54 +08:00
opt.on('-p', '--padding VAL', Numeric, 'space of the left and right of the plot') do |v|
2020-09-17 09:06:31 +08:00
params.padding = v
end
2020-10-10 23:11:54 +08:00
opt.on('-c', '--color VAL', String, 'color of the drawing') do |v|
2020-09-17 09:45:57 +08:00
params.color = v =~ /\A[0-9]+\z/ ? v.to_i : v.to_sym
2020-09-17 09:06:31 +08:00
end
2020-10-10 23:11:54 +08:00
opt.on('--[no-]labels', TrueClass, 'hide the labels') do |v|
2020-09-17 09:06:31 +08:00
params.labels = v
end
2020-11-23 22:52:14 +08:00
opt.on('--encoding VAL', String, 'Specify the input encoding') do |v|
@encoding = v
end
2020-11-14 22:41:35 +08:00
# 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
2020-11-14 22:09:49 +08:00
opt.on('--debug', TrueClass, 'print preprocessed data') do |v|
2020-09-17 09:06:31 +08:00
@debug = v
end
yield opt if block_given?
end
end
def main_parser
@main_parser ||= create_default_parser do |main_parser|
2020-11-14 23:43:13 +08:00
# Here, help message is stored in the banner.
# Because help of main_parser may be referred by `sub_parser`.
2020-09-17 09:06:31 +08:00
main_parser.banner = \
<<~MSG
2020-10-12 13:52:56 +08:00
2020-11-23 12:09:16 +08:00
Program: YouPlot (Tools for plotting on the terminal)
Version: #{YouPlot::VERSION} (using UnicodePlot #{UnicodePlot::VERSION})
Source: https://github.com/kojix2/youplot
2020-09-17 09:06:31 +08:00
Usage: uplot <command> [options] <in.tsv>
2020-09-17 09:06:31 +08:00
2020-10-12 15:14:51 +08:00
Commands:
2020-11-14 22:09:49 +08:00
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
2020-11-06 08:56:35 +08:00
colors show the list of available colors
2020-11-14 22:09:49 +08:00
count c draw a baplot based on the number of
occurrences (slow)
2020-11-14 23:29:20 +08:00
General options:
--help print command specific help menu
2020-11-23 12:09:16 +08:00
--version print the version of YouPlot
2020-09-17 09:06:31 +08:00
MSG
2020-11-14 23:29:20 +08:00
# 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
2020-09-17 09:06:31 +08:00
end
end
def sub_parser
@sub_parser ||= create_default_parser do |parser|
parser.banner = <<~MSG
2020-10-12 15:14:51 +08:00
2020-11-23 12:09:16 +08:00
Usage: YouPlot #{command} [options] <in.tsv>
2020-10-12 15:45:28 +08:00
Options for #{command}:
MSG
case command
2020-11-14 23:43:00 +08:00
2020-11-14 23:29:20 +08:00
# If you type only `uplot` in the terminal.
when nil
2020-11-14 23:29:20 +08:00
warn main_parser.banner
warn "\n"
exit 1
when :barplot, :bar
2020-10-12 15:45:28 +08:00
parser.on_head('--symbol VAL', String, 'character to be used to plot the bars') do |v|
params.symbol = v
end
2020-10-12 15:45:28 +08:00
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
2020-10-12 15:45:28 +08:00
parser.on_head('--symbol VAL', String, 'character to be used to plot the bars') do |v|
params.symbol = v
end
when :histogram, :hist
2020-10-12 15:45:28 +08:00
parser.on_head('-n', '--nbins VAL', Numeric, 'approximate number of bins') do |v|
params.nbins = v
end
2020-10-12 15:45:28 +08:00
parser.on_head('--closed VAL', String) do |v|
params.closed = v
end
2020-10-12 15:45:28 +08:00
parser.on_head('--symbol VAL', String, 'character to be used to plot the bars') do |v|
params.symbol = v
end
when :lineplot, :line
2020-10-12 15:45:28 +08:00
parser.on_head('--canvas VAL', String, 'type of canvas') do |v|
params.canvas = v
end
2020-10-12 15:45:28 +08:00
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
2020-10-12 15:45:28 +08:00
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
2020-10-12 15:45:28 +08:00
parser.on_head('--canvas VAL', String) do |v|
params.canvas = v
end
2020-10-12 15:45:28 +08:00
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
2020-10-12 15:45:28 +08:00
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
2020-10-12 15:45:28 +08:00
parser.on_head('--canvas VAL', String) do |v|
params.canvas = v
end
2020-10-12 15:45:28 +08:00
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
2020-10-12 15:45:28 +08:00
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
2020-10-12 15:45:28 +08:00
parser.on_head('--grid', TrueClass) do |v|
params.grid = v
end
2020-10-12 15:45:28 +08:00
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
2020-10-12 15:45:28 +08:00
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
2020-10-12 15:45:28 +08:00
parser.on_head('--xlim VAL', Array, 'plotting range for the x coordinate') do |v|
params.xlim = v.take(2)
end
when :colors
2020-11-06 09:21:36 +08:00
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
2020-09-17 09:06:31 +08:00
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)
2020-09-17 09:06:31 +08:00
rescue OptionParser::ParseError => e
warn "uplot: #{e.message}"
exit 1
end
end
end
end
2020-10-10 22:07:57 +08:00
end