cave configs: auto-action, tab-completion, history
* --auto-action for automatic actions for files, defined by /etc/paludis/configs_auto_action. * Readline for tab-completion and history support.
This commit is contained in:
parent
522f100ce2
commit
193c69b5e7
|
@ -1,9 +1,13 @@
|
|||
#!/usr/bin/env ruby
|
||||
|
||||
Main = self
|
||||
public
|
||||
|
||||
require 'Paludis'
|
||||
require 'getoptlong'
|
||||
require 'pathname'
|
||||
require 'shellwords'
|
||||
require 'readline'
|
||||
begin
|
||||
require 'irb-pager'
|
||||
rescue LoadError
|
||||
|
@ -42,8 +46,8 @@ class UsageError <Exception
|
|||
end
|
||||
|
||||
class NoSelection <UsageError
|
||||
def new x = nil
|
||||
super x || 'Select a file first.'
|
||||
def initialize x = nil
|
||||
super( x || 'Select a file first.')
|
||||
end
|
||||
end
|
||||
|
||||
|
@ -57,15 +61,6 @@ else
|
|||
def diff( f1, f2) system 'diff', f1, f2 end
|
||||
end
|
||||
|
||||
def list_cfgs selected, list
|
||||
keys = {}
|
||||
list.keys.sort.map do |i|
|
||||
file = list[i]
|
||||
keys[["\\#{i}"]] = (selected == file ? '* ' : '')+file.to_s
|
||||
end
|
||||
list_keys keys
|
||||
end
|
||||
|
||||
Log.instance.with do |nst|
|
||||
nst.log_level = LogLevel::Warning
|
||||
nst.program_name = $0
|
||||
|
@ -112,11 +107,128 @@ def list_keys cmds
|
|||
end
|
||||
end
|
||||
|
||||
selected = nil
|
||||
loop do
|
||||
begin
|
||||
found = Hash.new {|h,k| h[k] = [] }
|
||||
def list_cfgs selected, list
|
||||
keys = {}
|
||||
list.keys.sort.map do |i|
|
||||
file = list[i]
|
||||
keys[["\\#{i}"]] = (selected == file ? '* ' : '')+file.to_s
|
||||
end
|
||||
list_keys keys
|
||||
end
|
||||
|
||||
class Commands < Hash
|
||||
class Builder
|
||||
def initialize cmds
|
||||
@cmds = cmds
|
||||
end
|
||||
|
||||
def self.build cmds, &run
|
||||
r = new cmds
|
||||
r.instance_eval &run
|
||||
r
|
||||
end
|
||||
|
||||
def help text = nil
|
||||
@help = text
|
||||
end
|
||||
|
||||
def on *names, &run
|
||||
options = names.last.is_a?( Hash) ? names.pop.dup : {}
|
||||
names = names.flatten.compact
|
||||
if names.empty?
|
||||
@fallback_cmd = Cmd.new nil, @help, run
|
||||
else
|
||||
cmd = Cmd.new names, @help, &run
|
||||
names.each do |name|
|
||||
name = name.to_s
|
||||
if /\\(.)/ =~ name
|
||||
@cmds[$1.to_sym] = cmd
|
||||
name = name.gsub /\\/, ''
|
||||
end
|
||||
@cmds[name.to_sym] = cmd
|
||||
end
|
||||
end
|
||||
ensure
|
||||
@help = nil
|
||||
end
|
||||
end
|
||||
|
||||
class Cmd <Proc
|
||||
attr_reader :names, :help
|
||||
|
||||
def initialize names, help, &exe
|
||||
@names, @help = names, help
|
||||
super &exe
|
||||
end
|
||||
end
|
||||
|
||||
class CommandError < Exception
|
||||
end
|
||||
class ExpectingCommandError < CommandError
|
||||
def initialize *args
|
||||
args = ['Command expected'] if args.empty?
|
||||
super *args
|
||||
end
|
||||
end
|
||||
class UnknownCommandError < CommandError
|
||||
def initialize *args
|
||||
args = ['This Command i do not know'] if args.empty?
|
||||
super *args
|
||||
end
|
||||
end
|
||||
|
||||
attr_accessor :fallback_cmd
|
||||
|
||||
def self.new &builder
|
||||
r = super()
|
||||
Builder.build r, &builder
|
||||
r
|
||||
end
|
||||
|
||||
def === x
|
||||
include?( x.to_sym) or super
|
||||
end
|
||||
|
||||
def list_keys
|
||||
h = {}
|
||||
self.values.map {|c| h[c.names] = c.help }
|
||||
Main::list_keys( h)
|
||||
end
|
||||
end
|
||||
|
||||
class FileList
|
||||
attr_reader :found
|
||||
def initialize found
|
||||
@found = found
|
||||
end
|
||||
|
||||
def list
|
||||
return @list if @list
|
||||
list = {}
|
||||
found.each_with_index do |(file, cfgs), i|
|
||||
i += 1
|
||||
list[i] = file
|
||||
cfgs.sort!
|
||||
end
|
||||
@list = list
|
||||
end
|
||||
end
|
||||
|
||||
class UI
|
||||
attr_accessor :need_to_print
|
||||
attr_reader :selected, :cmds, :list, :found, :dirs
|
||||
def selected= x
|
||||
@selected, @need_to_print = x, !!x
|
||||
end
|
||||
|
||||
def initialize cmds, dirs
|
||||
@cmds, @dirs = cmds, dirs
|
||||
end
|
||||
|
||||
def prepare
|
||||
@need_to_print = false
|
||||
@found = Hash.new {|h,k| h[k] = [] }
|
||||
@list = {}
|
||||
find_cfgs( dirs).each do |cfg|
|
||||
found[cfg.dirname + cfg.basename.to_s.sub( /^\._cfg...._/, '')] += [cfg]
|
||||
end
|
||||
|
@ -124,75 +236,153 @@ loop do
|
|||
puts "No files to update."
|
||||
exit 0
|
||||
end
|
||||
found.each_with_index do |(file,cfgs),i|
|
||||
found.each_with_index do |(file, cfgs), i|
|
||||
i += 1
|
||||
list[i] = file
|
||||
cfgs.sort!
|
||||
end
|
||||
selected = nil unless found[selected]
|
||||
self.selected = nil unless found.has_key? selected
|
||||
end
|
||||
|
||||
def run
|
||||
prepare
|
||||
prompt.chomp.split( /\s+/).each &method( :execute)
|
||||
instance_eval &cmds[:diff] if need_to_print
|
||||
rescue Interrupt
|
||||
rescue UsageError
|
||||
puts $!.message
|
||||
end
|
||||
|
||||
def prompt
|
||||
prompt =
|
||||
if selected
|
||||
print "(cfg:\e[1m#{selected}\e[0m)# "
|
||||
"(cfg:\e[1m#{selected}\e[0m)# "
|
||||
else
|
||||
puts
|
||||
list_cfgs selected, list
|
||||
puts
|
||||
print "(select?)# "
|
||||
"(select?)# "
|
||||
end
|
||||
line = STDIN.gets or exit(0)
|
||||
line.chomp.split( /\s+/).each do |x|
|
||||
Readline.completion_proc = lambda do |str|
|
||||
reg = /^#{Regexp.quote str}/
|
||||
list.keys.select {|i| reg =~ i.to_s } +
|
||||
found.keys.grep( reg) +
|
||||
cmds.keys.grep( reg) +
|
||||
found.keys.select {|f| reg =~ f.basename.to_s }
|
||||
end
|
||||
line = Readline.readline( prompt, true) or exit
|
||||
Readline::HISTORY.pop if /^\s*$/ =~ line or (Readline::HISTORY.length >= 2 && Readline::HISTORY[-2] == line)
|
||||
line
|
||||
end
|
||||
|
||||
def execute x
|
||||
case x
|
||||
when /^\d+$/
|
||||
m = list[x.to_i]
|
||||
raise UsageError, "Unknown index #{x}" unless m
|
||||
selected = m
|
||||
when *%w[h help ?]
|
||||
self.selected = m
|
||||
when cmds
|
||||
instance_eval &cmds[x.to_sym]
|
||||
else
|
||||
m = Pathname.new x
|
||||
unless (f = found[m]).empty?
|
||||
self.selected = m
|
||||
else
|
||||
raise UsageError, "Unknown command #{x}"
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
auto_action_file = Pathname.new '/etc/paludis/configs_auto_action'
|
||||
|
||||
cmds = Commands.new do
|
||||
help 'Help'
|
||||
on %w[\help \?] do
|
||||
puts
|
||||
list_keys( {
|
||||
%w[\help \?] => 'Help',
|
||||
%w[\1 \2 ...] => 'Select file via index',
|
||||
%w[FILENAME] => 'Select file via filename',
|
||||
%w[\list] => 'List all files need updates',
|
||||
%w[\diff] => 'Displays difference of selected file',
|
||||
%w[\accept \yes] => 'Accepts selected file',
|
||||
%w[\reject \no] => 'Rejects selected file',
|
||||
%w[\quit \exit] => 'Quit',
|
||||
})
|
||||
cmds.list_keys
|
||||
puts
|
||||
when *%w[d diff]
|
||||
end
|
||||
|
||||
help 'Displays difference of selected file'
|
||||
on '\diff' do
|
||||
raise NoSelection unless selected
|
||||
puts
|
||||
IRB::Pager::pager( pager: 'less -R') { diff selected, found[selected][0] }
|
||||
puts
|
||||
when *%w[l list]
|
||||
self.need_to_print = false
|
||||
end
|
||||
|
||||
help 'List all files need updates'
|
||||
on '\list' do
|
||||
list_cfgs selected, list if selected
|
||||
when *%w[a accept y yes]
|
||||
end
|
||||
|
||||
help 'Accepts selected file'
|
||||
on %w[\accept \yes] do
|
||||
raise NoSelection unless selected
|
||||
accept_cfg selected, found[selected][0]
|
||||
selected = nil
|
||||
when *%w[r reject n no]
|
||||
end
|
||||
|
||||
help 'Rejects selected file'
|
||||
on %w[\reject \no] do
|
||||
raise NoSelection unless selected
|
||||
reject_cfg selected, found[selected][0]
|
||||
selected = nil
|
||||
when *%w[q quit e exit] then exit(0)
|
||||
when 'auto-action'
|
||||
File.readlines '/etc/paludis/config_auto_action' do |line|
|
||||
end
|
||||
|
||||
help 'Quit'
|
||||
on %w[\quit \exit] do
|
||||
exit 0
|
||||
end
|
||||
|
||||
on 'auto-action' do
|
||||
unless auto_action_file.exist?
|
||||
raise UsageError, "No config for auto-action found: #{auto_action_file}"
|
||||
end
|
||||
auto_action_file.each_line do |line|
|
||||
next if /^\s*#/ =~ line
|
||||
action, file = line.chomp!.split( "\t", 2)
|
||||
file = Pathname.new file
|
||||
if found.has_key? file
|
||||
case action
|
||||
when *%w[a accept]
|
||||
accept_cfg nil, nil if file.exist?
|
||||
end
|
||||
end
|
||||
else
|
||||
m = Pathname.new x
|
||||
if f = found[m]
|
||||
selected = m
|
||||
else
|
||||
raise UsageError, "Uknown command #{x}"
|
||||
accept_cfg file, found[file][-1]
|
||||
found[file][0...-1].each {|f| reject_cfg file, f }
|
||||
when *%w[r reject]
|
||||
found[file].each {|f| reject_cfg file, f }
|
||||
end
|
||||
end
|
||||
end
|
||||
rescue UsageError
|
||||
puts $!
|
||||
end
|
||||
end
|
||||
|
||||
ui = UI.new cmds, dirs
|
||||
|
||||
opts = GetoptLong.new(
|
||||
[ '-h', '--help', GetoptLong::NO_ARGUMENT ],
|
||||
[ '-A', '--auto-action', GetoptLong::OPTIONAL_ARGUMENT ]
|
||||
)
|
||||
opts.ordering = GetoptLong::REQUIRE_ORDER
|
||||
opts.each do |opt, arg|
|
||||
case opt
|
||||
when '-h'
|
||||
puts <<EOF
|
||||
Usage: #{ENV['cave']||'cave'} #{File.basename $0} [-h] [-A]
|
||||
|
||||
Without options, interactive user interface will be started
|
||||
Options:
|
||||
-A, --auto-action[=file] runs automatical actions from file or #{auto_action_file}
|
||||
|
||||
EOF
|
||||
exit
|
||||
when '-A'
|
||||
auto_action_file = Pathname.new arg if arg and not arg.empty?
|
||||
ui.prepare
|
||||
ui.execute 'auto-action'
|
||||
exit
|
||||
end
|
||||
end
|
||||
|
||||
loop { ui.run }
|
||||
|
|
Loading…
Reference in a new issue