require "yaml" require "thor" require "rubyc/core_extensions" module Rubyc class CLI < Thor class_option :require, :aliases => "-r" class_option :sync, :type => :boolean, :default => false, :aliases => "-s", :banner => "Synchronize stdout" def initialize(*args) super libs = options[:require] ? options[:require].strip.split(":") : [] libs.each { |lib| require lib } $stdout.sync = options[:sync] end def help(*args) super(*args) end desc "map BLOCK", "Enumerable#map" long_desc "THIS IS A LONG DESCRIPTION" def map(code) proc = eval("Proc.new{|line,index| l = line; lnum = index + 1;#{code}}") $stdin.each_line.each_with_index do |line, index| puts proc.call(line.chomp, index).to_s end end desc "sum BLOCK", "Active Support Enumerable#sum" def sum(code = nil) code ||= "line" proc = eval("Proc.new{|line| l = line; #{code}}") sum = $stdin.sum do |line| proc.call(line.chomp) end puts sum end desc "select BLOCK", "Enumerable#select" def select(code) proc = eval("Proc.new{|line,index| l = line; lnum = index + 1;#{code}}") $stdin.each_line.each_with_index do |line, index| puts line if proc.call(line.chomp, index) end end desc "reject BLOCK", "Enumerable#reject" def reject(code) proc = eval("Proc.new{|line,index| l = line; lnum = index + 1;#{code}}") $stdin.each_line.each_with_index do |line, index| puts line unless proc.call(line.chomp, index) end end desc "count_by BLOCK", "Count the number of lines that have the same property. The property is defined by the return value of the given the block" def count_by(code = nil) code ||= "line" proc = eval("Proc.new{|line| l = line; #{code}}") counts = $stdin.count_by do |line| proc.call(line.chomp) end puts counts.to_yaml end desc "sort_by BLOCK", "Enumerable#sort_by" def sort_by(code = nil) code ||= "line" proc = eval("Proc.new{|line| l = line; #{code}}") counts = $stdin.sort_by do |line| proc.call(line.chomp) end puts counts end desc "grep BLOCK", "Enumerable#grep" def grep(pattern, code = nil) pattern = eval(pattern) proc = code ? eval("Proc.new{|line| l = line; #{code}}") : nil puts $stdin.grep(pattern, &proc) end desc "scan MATCHER BLOCK", "String#scan" def scan(pattern, code = nil) pattern = eval(pattern) proc = code ? eval("Proc.new{|*match| m = match; #{code}}") : nil str = $stdin.read str.scan(pattern, &proc) end desc "uniq", "Enumerable#uniq" def uniq puts $stdin.to_a.uniq end desc "uniq_by BLOCK", "TODO" def uniq_by(code = nil) code ||= "line" proc = eval("Proc.new{|line| l = line; #{code}}") counts = $stdin.uniq_by do |line| proc.call(line.chomp) end puts counts end desc "compact", "Remove empty lines" def compact $stdin.each { |line| puts line if line.chomp! != "" } end desc "merge NB_LINES [SEPARATOR]", "Merge NB_LINES consecutive lines using SEPARATOR. If SEPARATOR is not given \',\' is used" def merge(nb_lines, sep = ",") $stdin.each_slice(nb_lines.to_i) { |chunk| puts chunk.map { |elem| elem.strip }.join(sep) } end end end