#!/usr/bin/env ruby # frozen_string_literal: true # encoding=utf-8 require 'English' require 'clipboard' require 'open3' require 'optparse' require 'tty-prompt' require 'yaml' require_relative 'colorize' require_relative 'env' require_relative 'shared' require_relative 'tap' require_relative 'markdown_exec/version' include Tap # rubocop:disable Style/MixinUsage $stderr.sync = true $stdout.sync = true BLOCK_SIZE = 1024 # hash with keys sorted by name # class Hash def sort_by_key keys.sort.to_h { |key| [key, self[key]] } end end # is the value a non-empty string or a binary? # # :reek:ManualDispatch ### temp class Object def present? case self.class.to_s when 'FalseClass', 'TrueClass' true else self && (!respond_to?(:blank?) || !blank?) end end end # is value empty? # class String BLANK_RE = /\A[[:space:]]*\z/.freeze def blank? empty? || BLANK_RE.match?(self) end end public # display_level values DISPLAY_LEVEL_BASE = 0 # required output DISPLAY_LEVEL_ADMIN = 1 DISPLAY_LEVEL_DEBUG = 2 DISPLAY_LEVEL_DEFAULT = DISPLAY_LEVEL_ADMIN DISPLAY_LEVEL_MAX = DISPLAY_LEVEL_DEBUG # @execute_files[ind] = @execute_files[ind] + [block] EF_STDOUT = 0 EF_STDERR = 1 EF_STDIN = 2 # execute markdown documents # module MarkdownExec # :reek:IrresponsibleModule class Error < StandardError; end ## an imported markdown document # class MDoc def initialize(table) @table = table end def code(block) all = [block[:name]] + recursively_required(block[:reqs]) all.reverse.map do |req| get_block_by_name(req).fetch(:body, '') end .flatten(1) .tap_inspect end def get_block_by_name(name, default = {}) @table.select { |block| block[:name] == name }.fetch(0, default) end def list_recursively_required_blocks(name) name_block = get_block_by_name(name) raise "Named code block `#{name}` not found." if name_block.nil? || name_block.keys.empty? all = [name_block[:name]] + recursively_required(name_block[:reqs]) # in order of appearance in document @table.select { |block| all.include? block[:name] } .map { |block| block.fetch(:body, '') } .flatten(1) .tap_inspect end def option_exclude_blocks(opts) block_name_excluded_match = Regexp.new opts[:block_name_excluded_match] if opts[:hide_blocks_by_name] @table.reject { |block| block[:name].match(block_name_excluded_match) } else @table end end def recursively_required(reqs) all = [] rem = reqs while rem.count.positive? rem = rem.map do |req| next if all.include? req all += [req] get_block_by_name(req).fetch(:reqs, []) end .compact .flatten(1) .tap_inspect(name: 'rem') end all.tap_inspect end end # format option defaults and values # # :reek:TooManyInstanceVariables class BlockLabel def initialize(filename:, headings:, menu_blocks_with_docname:, menu_blocks_with_headings:, title:) @filename = filename @headings = headings @menu_blocks_with_docname = menu_blocks_with_docname @menu_blocks_with_headings = menu_blocks_with_headings @title = title end def make ([@title] + (if @menu_blocks_with_headings [@headings.compact.join(' # ')] else [] end) + ( if @menu_blocks_with_docname [@filename] else [] end )).join(' ') end end FNR11 = '/' FNR12 = ',~' # format option defaults and values # class SavedAsset def initialize(filename:, prefix:, time:, blockname:) @filename = filename @prefix = prefix @time = time @blockname = blockname end def script_name fne = @filename.gsub(FNR11, FNR12) "#{[@prefix, @time.strftime('%F-%H-%M-%S'), fne, ',', @blockname].join('_')}.sh".tap_inspect end def stdout_name "#{[@prefix, @time.strftime('%F-%H-%M-%S'), @filename, @blockname].join('_')}.out.txt".tap_inspect end end # format option defaults and values # class OptionValue def initialize(value) @value = value end # as default value in env_str() # def for_hash(default = nil) return default if @value.nil? case @value.class.to_s when 'String', 'Integer' @value when 'FalseClass', 'TrueClass' @value ? true : false when @value.empty? default else @value.to_s end end # for output as default value in list_default_yaml() # def for_yaml(default = nil) return default if @value.nil? case @value.class.to_s when 'String' "'#{@value}'" when 'Integer' @value when 'FalseClass', 'TrueClass' @value ? true : false when @value.empty? default else @value.to_s end end end # a generated list of saved files # class Sfiles def initialize(folder, glob) @folder = folder @glob = glob end def list_all Dir.glob(File.join(@folder, @glob)).tap_inspect end def most_recent(arr = list_all) return unless arr return if arr.count < 1 arr.max.tap_inspect end def most_recent_list(arr = list_all) return unless arr return if (ac = arr.count) < 1 arr.sort[-[ac, options[:list_count]].min..].reverse.tap_inspect end end ## # # :reek:DuplicateMethodCall { allow_calls: ['block', 'item', 'lm', 'opts', 'option', '@options', 'required_blocks'] } # :reek:MissingSafeMethod { exclude: [ read_configuration_file! ] } # :reek:TooManyInstanceVariables ### temp # :reek:TooManyMethods ### temp class MarkParse attr_reader :options def initialize(options = {}) @options = options @prompt = TTY::Prompt.new(interrupt: :exit) @execute_aborted_at = nil @execute_completed_at = nil @execute_error = nil @execute_error_message = nil @execute_files = nil @execute_options = nil @execute_script_filespec = nil @execute_started_at = nil @option_parser = nil end ## # options necessary to start, parse input, defaults for cli options def base_options menu_iter do |item| # noisy item.tap_inspect name: :item, format: :yaml next unless item[:opt_name].present? item_default = item[:default] # noisy item_default.tap_inspect name: :item_default value = if item_default.nil? item_default else env_str(item[:env_var], default: OptionValue.new(item_default).for_hash) end [item[:opt_name], item[:proc1] ? item[:proc1].call(value) : value] end.compact.to_h.merge( { menu_exit_at_top: true, menu_with_exit: true } ).tap_inspect format: :yaml end def default_options { bash: true, # bash block parsing in get_block_summary() exclude_expect_blocks: true, hide_blocks_by_name: true, output_saved_script_filename: false, prompt_approve_block: 'Process?', prompt_select_block: 'Choose a block:', prompt_select_md: 'Choose a file:', prompt_select_output: 'Choose a file:', saved_script_filename: nil, # calculated struct: true # allow get_block_summary() } end def approve_block(opts, mdoc) required_blocks = mdoc.list_recursively_required_blocks(opts[:block_name]) display_command(opts, required_blocks) if opts[:output_script] || opts[:user_must_approve] allow = true if opts[:user_must_approve] loop do (sel = @prompt.select(opts[:prompt_approve_block], filter: true) do |menu| menu.default 1 # menu.enum '.' # menu.filter true menu.choice 'Yes', 1 menu.choice 'No', 2 menu.choice 'Copy script to clipboard', 3 menu.choice 'Save script', 4 end).tap_inspect name: :sel allow = (sel == 1) if sel == 3 text = required_blocks.flatten.join($INPUT_RECORD_SEPARATOR) Clipboard.copy(text) fout "Clipboard updated: #{required_blocks.count} blocks," / " #{required_blocks.flatten.count} lines," / " #{text.length} characters" end if sel == 4 write_command_file(opts.merge(save_executed_script: true), required_blocks) fout "File saved: #{@options[:saved_filespec]}" end break if [1, 2].include? sel end end (opts[:ir_approve] = allow).tap_inspect name: :allow selected = mdoc.get_block_by_name opts[:block_name] if opts[:ir_approve] write_command_file opts, required_blocks command_execute opts, required_blocks.flatten.join("\n") save_execution_output output_execution_summary output_execution_result end selected[:name] end # :reek:DuplicateMethodCall # :reek:UncommunicativeVariableName { exclude: [ e ] } # :reek:LongYieldList def command_execute(opts, command) @execute_files = Hash.new([]) @execute_options = opts @execute_started_at = Time.now.utc Open3.popen3(@options[:shell], '-c', command) do |stdin, stdout, stderr, exec_thr| # pid = exec_thr.pid # pid of the started process Thread.new do until (line = stdout.gets).nil? @execute_files[EF_STDOUT] = @execute_files[EF_STDOUT] + [line] print line if opts[:output_stdout] yield nil, line, nil, exec_thr if block_given? end rescue IOError # thread killed, do nothing end Thread.new do until (line = stderr.gets).nil? @execute_files[EF_STDERR] = @execute_files[EF_STDERR] + [line] print line if opts[:output_stdout] yield nil, nil, line, exec_thr if block_given? end rescue IOError # thread killed, do nothing end in_thr = Thread.new do while exec_thr.alive? # reading input until the child process ends stdin.puts(line = $stdin.gets) @execute_files[EF_STDIN] = @execute_files[EF_STDIN] + [line] yield line, nil, nil, exec_thr if block_given? end end exec_thr.join in_thr.kill # @return_code = exec_thr.value end @execute_completed_at = Time.now.utc rescue Errno::ENOENT => e # error triggered by missing command in script @execute_aborted_at = Time.now.utc @execute_error_message = e.message @execute_error = e @execute_files[EF_STDERR] += [@execute_error_message] fout "Error ENOENT: #{e.inspect}" rescue SignalException => e # SIGTERM triggered by user or system @execute_aborted_at = Time.now.utc @execute_error_message = 'SIGTERM' @execute_error = e @execute_files[EF_STDERR] += [@execute_error_message] fout "Error ENOENT: #{e.inspect}" end def count_blocks_in_filename fenced_start_and_end_match = Regexp.new @options[:fenced_start_and_end_match] cnt = 0 File.readlines(@options[:filename]).each do |line| cnt += 1 if line.match(fenced_start_and_end_match) end cnt / 2 end # :reek:DuplicateMethodCall def display_command(_opts, required_blocks) frame = ' #=#=#'.yellow fout frame required_blocks.each { |cb| fout cb } fout frame end # :reek:DuplicateMethodCall def exec_block(options, _block_name = '') options = default_options.merge options update_options options, over: false # document and block reports # files = list_files_per_options(options) simple_commands = { doc_glob: -> { fout options[:md_filename_glob] }, list_blocks: lambda do fout_list (files.map do |file| make_block_labels(filename: file, struct: true) end).flatten(1) end, list_default_yaml: -> { fout_list list_default_yaml }, list_docs: -> { fout_list files }, list_default_env: -> { fout_list list_default_env }, list_recent_output: -> { fout_list list_recent_output }, list_recent_scripts: -> { fout_list list_recent_scripts }, pwd: -> { fout File.expand_path('..', __dir__) }, run_last_script: -> { run_last_script }, select_recent_output: -> { select_recent_output }, select_recent_script: -> { select_recent_script }, tab_completions: -> { fout tab_completions }, menu_export: -> { fout menu_export } } simple_commands.each_key do |key| if @options[key] simple_commands[key].call return # rubocop:disable Lint/NonLocalExitFromIterator end end # process # @options[:filename] = select_md_file(files) select_and_approve_block( bash: true, struct: true ) fout "saved_filespec: #{@execute_script_filespec}" if @options[:output_saved_script_filename] end # standard output; not for debug # def fout(str) puts str end def fout_list(str) puts str end def fout_section(name, data) puts "# #{name}" puts data.to_yaml end # :reek:LongParameterList def get_block_summary(opts, headings:, block_title:, current:) return [current] unless opts[:struct] return [summarize_block(headings, block_title).merge({ body: current })] unless opts[:bash] bm = block_title.match(Regexp.new(opts[:block_name_match])) reqs = block_title.scan(Regexp.new(opts[:block_required_scan])) .map { |scanned| scanned[1..] } if bm && bm[1] [summarize_block(headings, bm[:title]).merge({ body: current, reqs: reqs })] else [summarize_block(headings, block_title).merge({ body: current, reqs: reqs })] end end def approved_fout?(level) level <= @options[:display_level] end # display output at level or lower than filter (DISPLAY_LEVEL_DEFAULT) # def lout(str, level: DISPLAY_LEVEL_BASE) return unless approved_fout? level # fout level == DISPLAY_LEVEL_BASE ? str : DISPLAY_LEVEL_XBASE_PREFIX + str fout level == DISPLAY_LEVEL_BASE ? str : @options[:display_level_xbase_prefix] + str end # :reek:DuplicateMethodCall def list_blocks_in_file(call_options = {}, &options_block) opts = optsmerge call_options, options_block unless opts[:filename]&.present? fout 'No blocks found.' exit 1 end unless File.exist? opts[:filename] fout 'Document is missing.' exit 1 end fenced_start_and_end_match = Regexp.new opts[:fenced_start_and_end_match] fenced_start_ex = Regexp.new opts[:fenced_start_ex_match] block_title = '' blocks = [] current = nil headings = [] in_block = false File.readlines(opts[:filename]).each do |line| continue unless line if opts[:menu_blocks_with_headings] if (lm = line.match(Regexp.new(opts[:heading3_match]))) headings = [headings[0], headings[1], lm[:name]] elsif (lm = line.match(Regexp.new(opts[:heading2_match]))) headings = [headings[0], lm[:name]] elsif (lm = line.match(Regexp.new(opts[:heading1_match]))) headings = [lm[:name]] end end if line.match(fenced_start_and_end_match) if in_block if current block_title = current.join(' ').gsub(/ +/, ' ')[0..64] if block_title.nil? || block_title.empty? blocks += get_block_summary opts, headings: headings, block_title: block_title, current: current current = nil end in_block = false block_title = '' else # new block # lm = line.match(fenced_start_ex) block_allow = false if opts[:bash_only] block_allow = true if lm && (lm[:shell] == 'bash') else block_allow = true block_allow = !(lm && (lm[:shell] == 'expect')) if opts[:exclude_expect_blocks] end in_block = true if block_allow && (!opts[:title_match] || (lm && lm[:name] && lm[:name].match(opts[:title_match]))) current = [] block_title = (lm && lm[:name]) end end elsif current current += [line.chomp] end end blocks.tap_inspect end def list_default_env menu_iter do |item| next unless item[:env_var].present? [ "#{item[:env_var]}=#{value_for_cli item[:default]}", item[:description].present? ? item[:description] : nil ].compact.join(' # ') end.compact.sort end def list_default_yaml menu_iter do |item| next unless item[:opt_name].present? && item[:default].present? [ "#{item[:opt_name]}: #{OptionValue.new(item[:default]).for_yaml}", item[:description].present? ? item[:description] : nil ].compact.join(' # ') end.compact.sort end def list_files_per_options(options) list_files_specified( specified_filename: options[:filename]&.present? ? options[:filename] : nil, specified_folder: options[:path], default_filename: 'README.md', default_folder: '.' ).tap_inspect end # :reek:LongParameterList def list_files_specified(specified_filename: nil, specified_folder: nil, default_filename: nil, default_folder: nil, filetree: nil) fn = File.join(if specified_filename&.present? if specified_folder&.present? [specified_folder, specified_filename] elsif specified_filename.start_with? '/' [specified_filename] else [default_folder, specified_filename] end elsif specified_folder&.present? if filetree [specified_folder, @options[:md_filename_match]] else [specified_folder, @options[:md_filename_glob]] end else [default_folder, default_filename] end) if filetree filetree.select { |filename| filename == fn || filename.match(/^#{fn}$/) || filename.match(%r{^#{fn}/.+$}) } else Dir.glob(fn) end.tap_inspect end def list_markdown_files_in_path Dir.glob(File.join(@options[:path], @options[:md_filename_glob])).tap_inspect end def list_named_blocks_in_file(call_options = {}, &options_block) opts = optsmerge call_options, options_block block_name_excluded_match = Regexp.new opts[:block_name_excluded_match] list_blocks_in_file(opts).map do |block| next if opts[:hide_blocks_by_name] && block[:name].match(block_name_excluded_match) block end.compact.tap_inspect end def list_recent_output Sfiles.new(@options[:saved_stdout_folder], @options[:saved_stdout_glob]).most_recent_list end def list_recent_scripts Sfiles.new(@options[:saved_script_folder], @options[:saved_script_glob]).most_recent_list end def make_block_labels(call_options = {}) opts = options.merge(call_options) list_blocks_in_file(opts).map do |block| # next if opts[:hide_blocks_by_name] && block[:name].match(%r{^:\(.+\)$}) BlockLabel.new(filename: opts[:filename], headings: block.fetch(:headings, []), menu_blocks_with_docname: opts[:menu_blocks_with_docname], menu_blocks_with_headings: opts[:menu_blocks_with_headings], title: block[:title]).make end.compact.tap_inspect end # :reek:DuplicateMethodCall # :reek:UncommunicativeMethodName ### temp def menu_data1 val_as_bool = ->(value) { value.class.to_s == 'String' ? (value.chomp != '0') : value } val_as_int = ->(value) { value.to_i } val_as_str = ->(value) { value.to_s } # val_true = ->(_value) { true } # for commands, sets option to true menu_options = [ { arg_name: 'PATH', default: '.', description: 'Read configuration file', long_name: 'config', proc1: lambda { |value| read_configuration_file! options, value } }, { arg_name: 'BOOL', default: false, description: 'Debug output', env_var: 'MDE_DEBUG', long_name: 'debug', short_name: 'd', proc1: lambda { |value| tap_config value.to_i != 0 } }, { arg_name: "INT.#{DISPLAY_LEVEL_BASE}-#{DISPLAY_LEVEL_MAX}", default: DISPLAY_LEVEL_DEFAULT, description: "Output display level (#{DISPLAY_LEVEL_BASE} to #{DISPLAY_LEVEL_MAX})", env_var: 'MDE_DISPLAY_LEVEL', long_name: 'display-level', opt_name: :display_level, proc1: val_as_int }, { arg_name: 'NAME', compreply: false, description: 'Name of block', env_var: 'MDE_BLOCK_NAME', long_name: 'block-name', opt_name: :block_name, short_name: 'f', proc1: val_as_str }, { arg_name: 'RELATIVE_PATH', compreply: '.', description: 'Name of document', env_var: 'MDE_FILENAME', long_name: 'filename', opt_name: :filename, short_name: 'f', proc1: val_as_str }, { description: 'List blocks', long_name: 'list-blocks', opt_name: :list_blocks, proc1: val_as_bool }, { arg_name: 'INT.1-', default: 32, description: 'Max. items to return in list', env_var: 'MDE_LIST_COUNT', long_name: 'list-count', opt_name: :list_count, proc1: val_as_int }, { description: 'List default configuration as environment variables', long_name: 'list-default-env', opt_name: :list_default_env }, { description: 'List default configuration as YAML', long_name: 'list-default-yaml', opt_name: :list_default_yaml }, { description: 'List docs in current folder', long_name: 'list-docs', opt_name: :list_docs, proc1: val_as_bool }, { description: 'List recent saved output', long_name: 'list-recent-output', opt_name: :list_recent_output, proc1: val_as_bool }, { description: 'List recent saved scripts', long_name: 'list-recent-scripts', opt_name: :list_recent_scripts, proc1: val_as_bool }, { arg_name: 'PREFIX', default: MarkdownExec::BIN_NAME, description: 'Name prefix for stdout files', env_var: 'MDE_LOGGED_STDOUT_FILENAME_PREFIX', long_name: 'logged-stdout-filename-prefix', opt_name: :logged_stdout_filename_prefix, proc1: val_as_str }, { arg_name: 'BOOL', default: false, description: 'Display document name in block selection menu', env_var: 'MDE_MENU_BLOCKS_WITH_DOCNAME', long_name: 'menu-blocks-with-docname', opt_name: :menu_blocks_with_docname, proc1: val_as_bool }, { arg_name: 'BOOL', default: false, description: 'Display headings (levels 1,2,3) in block selection menu', env_var: 'MDE_MENU_BLOCKS_WITH_HEADINGS', long_name: 'menu-blocks-with-headings', opt_name: :menu_blocks_with_headings, proc1: val_as_bool }, { arg_name: 'BOOL', default: false, description: 'Display summary for execution', env_var: 'MDE_OUTPUT_EXECUTION_SUMMARY', long_name: 'output-execution-summary', opt_name: :output_execution_summary, proc1: val_as_bool }, { arg_name: 'BOOL', default: false, description: 'Display script prior to execution', env_var: 'MDE_OUTPUT_SCRIPT', long_name: 'output-script', opt_name: :output_script, proc1: val_as_bool }, { arg_name: 'BOOL', default: true, description: 'Display standard output from execution', env_var: 'MDE_OUTPUT_STDOUT', long_name: 'output-stdout', opt_name: :output_stdout, proc1: val_as_bool }, { arg_name: 'RELATIVE_PATH', default: '.', description: 'Path to documents', env_var: 'MDE_PATH', long_name: 'path', opt_name: :path, short_name: 'p', proc1: val_as_str }, { description: 'Gem home folder', long_name: 'pwd', opt_name: :pwd, proc1: val_as_bool }, { description: 'Run most recently saved script', long_name: 'run-last-script', opt_name: :run_last_script, proc1: val_as_bool }, { arg_name: 'BOOL', default: false, description: 'Save executed script', env_var: 'MDE_SAVE_EXECUTED_SCRIPT', long_name: 'save-executed-script', opt_name: :save_executed_script, proc1: val_as_bool }, { arg_name: 'BOOL', default: false, description: 'Save standard output of the executed script', env_var: 'MDE_SAVE_EXECUTION_OUTPUT', long_name: 'save-execution-output', opt_name: :save_execution_output, proc1: val_as_bool }, { arg_name: 'INT', default: 0o755, description: 'chmod for saved scripts', env_var: 'MDE_SAVED_SCRIPT_CHMOD', long_name: 'saved-script-chmod', opt_name: :saved_script_chmod, proc1: val_as_int }, { arg_name: 'PREFIX', default: MarkdownExec::BIN_NAME, description: 'Name prefix for saved scripts', env_var: 'MDE_SAVED_SCRIPT_FILENAME_PREFIX', long_name: 'saved-script-filename-prefix', opt_name: :saved_script_filename_prefix, proc1: val_as_str }, { arg_name: 'RELATIVE_PATH', default: 'logs', description: 'Saved script folder', env_var: 'MDE_SAVED_SCRIPT_FOLDER', long_name: 'saved-script-folder', opt_name: :saved_script_folder, proc1: val_as_str }, { arg_name: 'GLOB', default: 'mde_*.sh', description: 'Glob matching saved scripts', env_var: 'MDE_SAVED_SCRIPT_GLOB', long_name: 'saved-script-glob', opt_name: :saved_script_glob, proc1: val_as_str }, { arg_name: 'RELATIVE_PATH', default: 'logs', description: 'Saved stdout folder', env_var: 'MDE_SAVED_STDOUT_FOLDER', long_name: 'saved-stdout-folder', opt_name: :saved_stdout_folder, proc1: val_as_str }, { arg_name: 'GLOB', default: 'mde_*.out.txt', description: 'Glob matching saved outputs', env_var: 'MDE_SAVED_STDOUT_GLOB', long_name: 'saved-stdout-glob', opt_name: :saved_stdout_glob, proc1: val_as_str }, { description: 'Select and execute a recently saved output', long_name: 'select-recent-output', opt_name: :select_recent_output, proc1: val_as_bool }, { description: 'Select and execute a recently saved script', long_name: 'select-recent-script', opt_name: :select_recent_script, proc1: val_as_bool }, { description: 'YAML export of menu', long_name: 'menu-export', opt_name: :menu_export, proc1: val_as_bool }, { description: 'List tab completions', long_name: 'tab-completions', opt_name: :tab_completions, proc1: val_as_bool }, { arg_name: 'BOOL', default: true, description: 'Pause for user to approve script', env_var: 'MDE_USER_MUST_APPROVE', long_name: 'user-must-approve', opt_name: :user_must_approve, proc1: val_as_bool }, { description: 'Show current configuration values', short_name: '0', proc1: lambda { |_| options_finalize options fout options.sort_by_key.to_yaml } }, { description: 'App help', long_name: 'help', short_name: 'h', proc1: lambda { |_| fout menu_help exit } }, { description: "Print the gem's version", long_name: 'version', short_name: 'v', proc1: lambda { |_| fout MarkdownExec::VERSION exit } }, { description: 'Exit app', long_name: 'exit', short_name: 'x', proc1: ->(_) { exit } }, { default: '^\(.*\)$', description: 'Pattern for blocks to hide from user-selection', env_var: 'MDE_BLOCK_NAME_EXCLUDED_MATCH', opt_name: :block_name_excluded_match, proc1: val_as_str }, { default: ':(?\S+)( |$)', env_var: 'MDE_BLOCK_NAME_MATCH', opt_name: :block_name_match, proc1: val_as_str }, { default: '\+\S+', env_var: 'MDE_BLOCK_REQUIRED_SCAN', opt_name: :block_required_scan, proc1: val_as_str }, { default: '> ', env_var: 'MDE_DISPLAY_LEVEL_XBASE_PREFIX', opt_name: :display_level_xbase_prefix, proc1: val_as_str }, { default: '^`{3,}', env_var: 'MDE_FENCED_START_AND_END_MATCH', opt_name: :fenced_start_and_end_match, proc1: val_as_str }, { default: '^`{3,}(?<shell>[^`\s]*) *(?<name>.*)$', env_var: 'MDE_FENCED_START_EX_MATCH', opt_name: :fenced_start_ex_match, proc1: val_as_str }, { default: '^# *(?<name>[^#]*?) *$', env_var: 'MDE_HEADING1_MATCH', opt_name: :heading1_match, proc1: val_as_str }, { default: '^## *(?<name>[^#]*?) *$', env_var: 'MDE_HEADING2_MATCH', opt_name: :heading2_match, proc1: val_as_str }, { default: '^### *(?<name>.+?) *$', env_var: 'MDE_HEADING3_MATCH', opt_name: :heading3_match, proc1: val_as_str }, { default: '*.[Mm][Dd]', env_var: 'MDE_MD_FILENAME_GLOB', opt_name: :md_filename_glob, proc1: val_as_str }, { default: '.+\\.md', env_var: 'MDE_MD_FILENAME_MATCH', opt_name: :md_filename_match, proc1: val_as_str }, { description: 'Options for viewing saved output file', env_var: 'MDE_OUTPUT_VIEWER_OPTIONS', opt_name: :output_viewer_options, proc1: val_as_str }, { default: 24, description: 'Maximum # of rows in select list', env_var: 'MDE_SELECT_PAGE_HEIGHT', opt_name: :select_page_height, proc1: val_as_int }, { default: '#!/usr/bin/env', description: 'Shebang for saved scripts', env_var: 'MDE_SHEBANG', opt_name: :shebang, proc1: val_as_str }, { default: 'bash', description: 'Shell for launched scripts', env_var: 'MDE_SHELL', opt_name: :shell, proc1: val_as_str } ] # commands first, options second (menu_options.reject { |option| option[:arg_name] }) + (menu_options.select { |option| option[:arg_name] }) end def menu_iter(data = menu_data1, &block) data.map(&block) end def menu_help @option_parser.help end ## post-parse options configuration # def options_finalize(rest) ## position 0: file or folder (optional) # if (pos = rest.fetch(0, nil))&.present? if Dir.exist?(pos) @options[:path] = pos elsif File.exist?(pos) @options[:filename] = pos else raise "Invalid parameter: #{pos}" end end ## position 1: block name (optional) # block_name = rest.fetch(1, nil) @options[:block_name] = block_name if block_name.present? end # :reek:ControlParameter def optsmerge(call_options = {}, options_block = nil) class_call_options = @options.merge(call_options || {}) if options_block options_block.call class_call_options else class_call_options end.tap_inspect end def output_execution_result oq = [['Block', @options[:block_name], DISPLAY_LEVEL_ADMIN], ['Command', [MarkdownExec::BIN_NAME, @options[:filename], @options[:block_name]].join(' '), DISPLAY_LEVEL_ADMIN]] [['Script', :saved_filespec], ['StdOut', :logged_stdout_filespec]].each do |label, name| oq << [label, @options[name], DISPLAY_LEVEL_ADMIN] if @options[name] end oq.map do |label, value, level| lout ["#{label}:".yellow, value.to_s].join(' '), level: level end end def output_execution_summary return unless @options[:output_execution_summary] fout_section 'summary', { execute_aborted_at: @execute_aborted_at, execute_completed_at: @execute_completed_at, execute_error: @execute_error, execute_error_message: @execute_error_message, execute_files: @execute_files, execute_options: @execute_options, execute_started_at: @execute_started_at, execute_script_filespec: @execute_script_filespec } end def prompt_with_quit(prompt_text, items, opts = {}) exit_option = '* Exit' all_items = if @options[:menu_exit_at_top] (@options[:menu_with_exit] ? [exit_option] : []) + items else items + (@options[:menu_with_exit] ? [exit_option] : []) end sel = @prompt.select(prompt_text, all_items, opts.merge(filter: true)) sel == exit_option ? nil : sel end # :reek:UtilityFunction ### temp def read_configuration_file!(options, configuration_path) return unless File.exist?(configuration_path) # rubocop:disable Security/YAMLLoad options.merge!((YAML.load(File.open(configuration_path)) || {}) .transform_keys(&:to_sym)) # rubocop:enable Security/YAMLLoad end # :reek:NestedIterators def run ## default configuration # @options = base_options ## read local configuration file # read_configuration_file! @options, ".#{MarkdownExec::APP_NAME.downcase}.yml" @option_parser = option_parser = OptionParser.new do |opts| executable_name = File.basename($PROGRAM_NAME) opts.banner = [ "#{MarkdownExec::APP_NAME}" \ " - #{MarkdownExec::APP_DESC} (#{MarkdownExec::VERSION})", "Usage: #{executable_name} [(path | filename [block_name])] [options]" ].join("\n") menu_iter do |item| next unless item[:long_name].present? || item[:short_name].present? opts.on(*[if item[:long_name].present? "--#{item[:long_name]}#{item[:arg_name].present? ? " #{item[:arg_name]}" : ''}" end, item[:short_name].present? ? "-#{item[:short_name]}" : nil, [item[:description], item[:default].present? ? "[#{value_for_cli item[:default]}]" : nil].compact.join(' '), lambda { |value| # ret = item[:proc1].call(value) ret = item[:proc1] ? item[:proc1].call(value) : value options[item[:opt_name]] = ret if item[:opt_name] ret }].compact) end end option_parser.load # filename defaults to basename of the program without suffix in a directory ~/.options option_parser.environment # env defaults to the basename of the program. rest = option_parser.parse! # (into: options) options_finalize rest exec_block options, options[:block_name] end def saved_name_split(name) mf = name.match(/#{@options[:saved_script_filename_prefix]}_(?<time>[0-9\-]+)_(?<file>.+)_,_(?<block>.+)\.sh/) return unless mf @options[:block_name] = mf[:block].tap_inspect name: :options_block_name @options[:filename] = mf[:file].gsub(FNR12, FNR11).tap_inspect name: :options_filename end def run_last_script filename = Sfiles.new(@options[:saved_script_folder], @options[:saved_script_glob]).most_recent return unless filename saved_name_split filename @options[:save_executed_script] = false select_and_approve_block end def save_execution_output @options.tap_inspect name: :options return unless @options[:save_execution_output] @options[:logged_stdout_filename] = SavedAsset.new(blockname: @options[:block_name], filename: File.basename(@options[:filename], '.*'), prefix: @options[:logged_stdout_filename_prefix], time: Time.now.utc).stdout_name @options[:logged_stdout_filespec] = File.join @options[:saved_stdout_folder], @options[:logged_stdout_filename] @logged_stdout_filespec = @options[:logged_stdout_filespec] (dirname = File.dirname(@options[:logged_stdout_filespec])).tap_inspect name: :dirname Dir.mkdir dirname unless File.exist?(dirname) ol = ["-STDOUT-\n"] ol += @execute_files&.fetch(EF_STDOUT, []) ol += ["\n-STDERR-\n"] ol += @execute_files&.fetch(EF_STDERR, []) ol += ["\n-STDIN-\n"] ol += @execute_files&.fetch(EF_STDIN, []) ol += ["\n"] File.write(@options[:logged_stdout_filespec], ol.join) end def select_and_approve_block(call_options = {}, &options_block) opts = optsmerge call_options, options_block blocks_in_file = list_blocks_in_file(opts.merge(struct: true)) mdoc = MDoc.new(blocks_in_file) repeat_menu = true && !opts[:block_name].present? loop do unless opts[:block_name].present? pt = (opts[:prompt_select_block]).to_s blocks_in_file.each do |block| block.merge! label: BlockLabel.new(filename: opts[:filename], headings: block.fetch(:headings, []), menu_blocks_with_docname: opts[:menu_blocks_with_docname], menu_blocks_with_headings: opts[:menu_blocks_with_headings], title: block[:title]).make end block_labels = mdoc.option_exclude_blocks(opts).map { |block| block[:label] } return nil if block_labels.count.zero? sel = prompt_with_quit pt, block_labels, per_page: opts[:select_page_height] return nil if sel.nil? # if sel.nil? # repeat_menu = false # break # end label_block = blocks_in_file.select { |block| block[:label] == sel }.fetch(0, nil) opts[:block_name] = @options[:block_name] = label_block[:name] end # if repeat_menu approve_block opts, mdoc # end break unless repeat_menu opts[:block_name] = '' end end def select_md_file(files = list_markdown_files_in_path) opts = options if (count = files.count) == 1 files[0] elsif count >= 2 prompt_with_quit opts[:prompt_select_md].to_s, files, per_page: opts[:select_page_height] end end def select_recent_output filename = prompt_with_quit @options[:prompt_select_output].to_s, list_recent_output, per_page: @options[:select_page_height] return unless filename.present? `open #{filename} #{options[:output_viewer_options]}` end def select_recent_script filename = prompt_with_quit @options[:prompt_select_md].to_s, list_recent_scripts, per_page: @options[:select_page_height] return if filename.nil? saved_name_split filename select_and_approve_block( bash: true, save_executed_script: false, struct: true ) end def summarize_block(headings, title) { headings: headings, name: title, title: title } end def menu_export(data = menu_data1) data.map do |item| item.delete(:proc1) item end.to_yaml end def tab_completions(data = menu_data1) data.map do |item| "--#{item[:long_name]}" if item[:long_name] end.compact end # :reek:BooleanParameter # :reek:ControlParameter def update_options(opts = {}, over: true) if over @options = @options.merge opts else @options.merge! opts end @options.tap_inspect format: :yaml end def write_command_file(call_options, required_blocks) return unless call_options[:save_executed_script] time_now = Time.now.utc opts = optsmerge call_options opts[:saved_script_filename] = SavedAsset.new(blockname: opts[:block_name], filename: opts[:filename], prefix: opts[:saved_script_filename_prefix], time: time_now).script_name @execute_script_filespec = @options[:saved_filespec] = File.join opts[:saved_script_folder], opts[:saved_script_filename] dirname = File.dirname(@options[:saved_filespec]) Dir.mkdir dirname unless File.exist?(dirname) (shebang = if @options[:shebang]&.present? "#{@options[:shebang]} #{@options[:shell]}\n" else '' end ).tap_inspect name: :shebang File.write(@options[:saved_filespec], shebang + "# file_name: #{opts[:filename]}\n" \ "# block_name: #{opts[:block_name]}\n" \ "# time: #{time_now}\n" \ "#{required_blocks.flatten.join("\n")}\n") return if @options[:saved_script_chmod].zero? File.chmod @options[:saved_script_chmod], @options[:saved_filespec] end end end