#!/usr/bin/env ruby # frozen_string_literal: true # encoding=utf-8 require 'English' require 'clipboard' require 'fileutils' require 'open3' require 'optparse' require 'shellwords' require 'tmpdir' require 'tty-prompt' require 'yaml' require_relative 'block_label' require_relative 'cached_nested_file_reader' require_relative 'cli' require_relative 'colorize' require_relative 'env' require_relative 'fcb' require_relative 'filter' require_relative 'markdown_exec/version' require_relative 'mdoc' require_relative 'option_value' require_relative 'saved_assets' require_relative 'saved_files_matcher' require_relative 'shared' require_relative 'tap' include CLI include Tap tap_config envvar: MarkdownExec::TAP_DEBUG $stderr.sync = true $stdout.sync = true MDE_HISTORY_ENV_NAME = 'MDE_MENU_HISTORY' # custom error: file specified is missing # class FileMissingError < StandardError; end # hash with keys sorted by name # add Hash.sym_keys # class Hash unless defined?(sort_by_key) def sort_by_key keys.sort.to_h { |key| [key, self[key]] } end end unless defined?(sym_keys) def sym_keys transform_keys(&:to_sym) end end end class LoadFile Load = true Reuse = false end class MenuState BACK = :back CONTINUE = :continue EXIT = :exit end # integer value for comparison # def options_fetch_display_level(options) options.fetch(:display_level, 1) end # integer value for comparison # def options_fetch_display_level_xbase_prefix(options) options.fetch(:level_xbase_prefix, '') end # stdout manager # module FOUT # 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 def approved_fout?(level) level <= options_fetch_display_level(@options) 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 : options_fetch_display_level_xbase_prefix(@options) + str end end def dp(str) lout " => #{str}", level: DISPLAY_LEVEL_DEBUG end def rpry require 'pry-nav' require 'pry-stack_explorer' end public # :reek:UtilityFunction def list_recent_output(saved_stdout_folder, saved_stdout_glob, list_count) SavedFilesMatcher.most_recent_list(saved_stdout_folder, saved_stdout_glob, list_count) end # :reek:UtilityFunction def list_recent_scripts(saved_script_folder, saved_script_glob, list_count) SavedFilesMatcher.most_recent_list(saved_script_folder, saved_script_glob, list_count) end # convert regex match groups to a hash with symbol keys # # :reek:UtilityFunction def extract_named_captures_from_option(str, option) str.match(Regexp.new(option))&.named_captures&.sym_keys end module ArrayUtil def self.partition_by_predicate(arr) true_list = [] false_list = [] arr.each do |element| if yield(element) true_list << element else false_list << element end end [true_list, false_list] end end module StringUtil # Splits the given string on the first occurrence of the specified character. # Returns an array containing the portion of the string before the character and the rest of the string. # # @param input_str [String] The string to be split. # @param split_char [String] The character on which to split the string. # @return [Array] An array containing two elements: the part of the string before split_char, and the rest of the string. def self.partition_at_first(input_str, split_char) split_index = input_str.index(split_char) if split_index.nil? [input_str, ''] else [input_str[0...split_index], input_str[(split_index + 1)..-1]] end end end # execute markdown documents # module MarkdownExec # :reek:IrresponsibleModule FNR11 = '/' FNR12 = ',~' SHELL_COLOR_OPTIONS = { BlockType::BASH => :menu_bash_color, BlockType::LINK => :menu_link_color, BlockType::OPTS => :menu_opts_color, BlockType::VARS => :menu_vars_color }.freeze ## # # rubocop:disable Layout/LineLength # :reek:DuplicateMethodCall { allow_calls: ['block', 'item', 'lm', 'opts', 'option', '@options', 'required_blocks'] } # rubocop:enable Layout/LineLength # :reek:MissingSafeMethod { exclude: [ read_configuration_file! ] } # :reek:TooManyInstanceVariables ### temp # :reek:TooManyMethods ### temp class MarkParse attr_reader :options include ArrayUtil include StringUtil include FOUT def initialize(options = {}) @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 @options = options @prompt = tty_prompt_without_disabled_symbol end # Adds Back and Exit options to the CLI menu # # @param blocks_in_file [Array] The current blocks in the menu def add_menu_chrome_blocks!(blocks_in_file) return unless @options[:menu_link_format].present? if @options[:menu_with_back] && history_state_exist? append_chrome_block(blocks_in_file, MenuState::BACK) end if @options[:menu_with_exit] append_chrome_block(blocks_in_file, MenuState::EXIT) end append_divider(blocks_in_file, @options, :initial) append_divider(blocks_in_file, @options, :final) end ## # Appends a summary of a block (FCB) to the blocks array. # def append_block_summary(blocks, fcb, opts) ## enhance fcb with block summary # blocks.push get_block_summary(opts, fcb) end # Appends a chrome block, which is a menu option for Back or Exit # # @param blocks_in_file [Array] The current blocks in the menu # @param type [Symbol] The type of chrome block to add (:back or :exit) def append_chrome_block(blocks_in_file, type) case type when MenuState::BACK state = history_state_partition(@options) @hs_curr = state[:unit] @hs_rest = state[:rest] option_name = @options[:menu_option_back_name] insert_at_top = @options[:menu_back_at_top] when MenuState::EXIT option_name = @options[:menu_option_exit_name] insert_at_top = @options[:menu_exit_at_top] end formatted_name = format(@options[:menu_link_format], safeval(option_name)) chrome_block = FCB.new( chrome: true, dname: formatted_name.send(@options[:menu_link_color].to_sym), oname: formatted_name ) if insert_at_top blocks_in_file.unshift(chrome_block) else blocks_in_file.push(chrome_block) end end # Appends a divider to the blocks array. # @param blocks [Array] The array of block elements. # @param opts [Hash] Options containing divider configuration. # @param position [Symbol] :initial or :final divider position. def append_divider(blocks, opts, position) divider_key = position == :initial ? :menu_initial_divider : :menu_final_divider unless opts[:menu_divider_format].present? && opts[divider_key].present? return end oname = format(opts[:menu_divider_format], safeval(opts[divider_key])) divider = FCB.new( chrome: true, disabled: '', dname: oname.send(opts[:menu_divider_color].to_sym), oname: oname ) position == :initial ? blocks.unshift(divider) : blocks.push(divider) end # Execute a code block after approval and provide user interaction options. # # This method displays required code blocks, asks for user approval, and # executes the code block if approved. It also allows users to copy the # code to the clipboard or save it to a file. # # @param opts [Hash] Options hash containing configuration settings. # @param mdoc [YourMDocClass] An instance of the MDoc class. # def approve_and_execute_block(selected, opts, mdoc) if selected.fetch(:shell, '') == BlockType::LINK handle_shell_link(opts, selected.fetch(:body, ''), mdoc) elsif opts.fetch(:s_back, false) handle_back_link(opts) elsif selected[:shell] == BlockType::OPTS handle_shell_opts(opts, selected) else handle_remainder_blocks(mdoc, opts, selected) end end # return arguments before `--` # def arguments_for_mde(argv = ARGV) case ind = argv.find_index('--') when nil argv when 0 [] else argv[0..ind - 1] end end ## # options necessary to start, parse input, defaults for cli options # def base_options menu_iter do |item| next unless item[:opt_name].present? item_default = item[:default] value = if item_default.nil? item_default else env_str(item[:env_var], default: OptionValue.for_hash(item_default)) end [item[:opt_name], item[:proccode] ? item[:proccode].call(value) : value] end.compact.to_h end # Finds the first hash-like element within an enumerable collection where the specified key # matches the given value. Returns a default value if no match is found. # # @param blocks [Enumerable] An enumerable collection of hash-like objects. # @param key [Object] The key to look up in each hash-like object. # @param value [Object] The value to compare against the value associated with the key. # @param default [Object] The default value to return if no match is found (optional). # @return [Object, nil] The found hash-like object, or the default value if no match is found. def block_find(blocks, key, value, default = nil) blocks.find { |item| item[key] == value } || default end def blocks_per_opts(blocks, opts) return blocks if opts[:struct] blocks.map do |block| block.fetch(:text, nil) || block.oname end.compact.reject(&:empty?) end def calculated_options { bash: true, # bash block parsing in get_block_summary() saved_script_filename: nil, # calculated struct: true # allow get_block_summary() } end # Check whether the document exists and is readable def check_file_existence(filename) unless filename&.present? fout 'No blocks found.' return false end unless File.exist? filename fout 'Document is missing.' return false end true end def clear_required_file ENV['MDE_LINK_REQUIRED_FILE'] = '' end # Collect required code blocks based on the provided options. # # @param opts [Hash] Options hash containing configuration settings. # @param mdoc [YourMDocClass] An instance of the MDoc class. # @return [Array] Required code blocks as an array of lines. def collect_required_code_lines(mdoc, selected, opts: {}) # Apply hash in opts block to environment variables if selected[:shell] == BlockType::VARS data = YAML.load(selected[:body].join("\n")) data.each_key do |key| ENV[key] = value = data[key].to_s next unless opts[:menu_vars_set_format].present? print format( opts[:menu_vars_set_format], { key: key, value: value } ).send(opts[:menu_vars_set_color].to_sym) end end required = mdoc.collect_recursively_required_code(opts[:block_name], opts: opts) read_required_blocks_from_temp_file + required[:code] end def cfile @cfile ||= CachedNestedFileReader.new( import_pattern: @options.fetch(:import_pattern) ) end EF_STDOUT = :stdout EF_STDERR = :stderr EF_STDIN = :stdin # Existing command_execute method def command_execute(opts, command, args: []) @execute_files = Hash.new([]) @execute_options = opts @execute_started_at = Time.now.utc Open3.popen3(opts[:shell], '-c', command, opts[:filename], *args) do |stdin, stdout, stderr, exec_thr| handle_stream(opts, stdout, EF_STDOUT) do |line| yield nil, line, nil, exec_thr if block_given? end handle_stream(opts, stderr, EF_STDERR) do |line| yield nil, nil, line, exec_thr if block_given? end in_thr = handle_stream(opts, $stdin, EF_STDIN) do |line| stdin.puts(line) yield line, nil, nil, exec_thr if block_given? end exec_thr.join sleep 0.1 in_thr.kill if in_thr&.alive? end @execute_completed_at = Time.now.utc rescue Errno::ENOENT => err #d 'command error ENOENT triggered by missing command in script' @execute_aborted_at = Time.now.utc @execute_error_message = err.message @execute_error = err @execute_files[EF_STDERR] += [@execute_error_message] fout "Error ENOENT: #{err.inspect}" rescue SignalException => err #d 'command SIGTERM triggered by user or system' @execute_aborted_at = Time.now.utc @execute_error_message = 'SIGTERM' @execute_error = err @execute_files[EF_STDERR] += [@execute_error_message] fout "Error ENOENT: #{err.inspect}" end def command_or_user_selected_block(blocks_in_file, blocks_menu, default, opts) if opts[:block_name].present? block = blocks_in_file.find do |item| item[:oname] == opts[:block_name] end else block, state = wait_for_user_selected_block(blocks_in_file, blocks_menu, default, opts) end [block, state] end def copy_to_clipboard(required_lines) text = required_lines.flatten.join($INPUT_RECORD_SEPARATOR) Clipboard.copy(text) fout "Clipboard updated: #{required_lines.count} blocks," \ " #{required_lines.flatten.count} lines," \ " #{text.length} characters" end def count_blocks_in_filename fenced_start_and_end_regex = Regexp.new @options[:fenced_start_and_end_regex] cnt = 0 cfile.readlines(@options[:filename]).each do |line| cnt += 1 if line.match(fenced_start_and_end_regex) end cnt / 2 end ## # Creates and adds a formatted block to the blocks array based on the provided match and format options. # @param blocks [Array] The array of blocks to add the new block to. # @param fcb [FCB] The file control block containing the line to match against. # @param match_data [MatchData] The match data containing named captures for formatting. # @param format_option [String] The format string to be used for the new block. # @param color_method [Symbol] The color method to apply to the block's display name. def create_and_add_chrome_block(blocks, _fcb, match_data, format_option, color_method) oname = format(format_option, match_data.named_captures.transform_keys(&:to_sym)) blocks.push FCB.new( chrome: true, disabled: '', dname: oname.send(color_method), oname: oname ) end ## # Processes lines within the file and converts them into blocks if they match certain criteria. # @param blocks [Array] The array to append new blocks to. # @param fcb [FCB] The file control block being processed. # @param opts [Hash] Options containing configuration for line processing. # @param use_chrome [Boolean] Indicates if the chrome styling should be applied. def create_and_add_chrome_blocks(blocks, fcb, opts, use_chrome) return unless use_chrome if opts[:menu_note_match].present? && (mbody = fcb.body[0].match opts[:menu_note_match]) create_and_add_chrome_block(blocks, fcb, mbody, opts[:menu_note_format], opts[:menu_note_color].to_sym) elsif opts[:menu_divider_match].present? && (mbody = fcb.body[0].match opts[:menu_divider_match]) create_and_add_chrome_block(blocks, fcb, mbody, opts[:menu_divider_format], opts[:menu_divider_color].to_sym) elsif opts[:menu_task_match].present? && (mbody = fcb.body[0].match opts[:menu_task_match]) create_and_add_chrome_block(blocks, fcb, mbody, opts[:menu_task_format], opts[:menu_task_color].to_sym) end end def create_and_write_file_with_permissions(file_path, content, chmod_value) dirname = File.dirname(file_path) FileUtils.mkdir_p dirname File.write(file_path, content) return if chmod_value.zero? File.chmod chmod_value, file_path end # Deletes a required temporary file specified by an environment variable. # The function checks if the file exists before attempting to delete it. # Clears the environment variable after deletion. # def delete_required_temp_file temp_blocks_file_path = ENV.fetch('MDE_LINK_REQUIRED_FILE', nil) if temp_blocks_file_path.nil? || temp_blocks_file_path.empty? return end FileUtils.rm_f(temp_blocks_file_path) clear_required_file end # Derives a title from the body of an FCB object. # @param fcb [Object] The FCB object whose title is to be derived. # @return [String] The derived title. def derive_title_from_body(fcb) body_content = fcb&.body return '' unless body_content if body_content.count == 1 body_content.first else format_multiline_body_as_title(body_content) end end ## Determines the correct filename to use for searching files # def determine_filename(specified_filename: nil, specified_folder: nil, default_filename: nil, default_folder: nil, filetree: nil) if specified_filename&.present? return specified_filename if specified_filename.start_with?('/') File.join(specified_folder || default_folder, specified_filename) elsif specified_folder&.present? File.join(specified_folder, filetree ? @options[:md_filename_match] : @options[:md_filename_glob]) else File.join(default_folder, default_filename) end end # :reek:DuplicateMethodCall def display_required_code(opts, required_lines) frame = opts[:output_divider].send(opts[:output_divider_color].to_sym) fout frame required_lines.each { |cb| fout cb } fout frame end def execute_approved_block(opts, required_lines) write_command_file(opts, required_lines) command_execute( opts, required_lines.flatten.join("\n"), args: opts.fetch(:s_pass_args, []) ) initialize_and_save_execution_output output_execution_summary output_execution_result end # Reports and executes block logic def execute_block_logic(files) @options[:filename] = select_document_if_multiple(files) select_approve_and_execute_block({ bash: true, struct: true }) end ## Executes the block specified in the options # def execute_block_with_error_handling(rest) finalize_cli_argument_processing(rest) @options[:s_cli_rest] = rest execute_code_block_based_on_options(@options) rescue FileMissingError => err puts "File missing: #{err}" rescue StandardError => err warn(error = "ERROR ** MarkParse.execute_block_with_error_handling(); #{err.inspect}") binding.pry if $tap_enable raise ArgumentError, error end # Main method to execute a block based on options and block_name def execute_code_block_based_on_options(options) options = calculated_options.merge(options) update_options(options, over: false) simple_commands = { doc_glob: -> { fout options[:md_filename_glob] }, list_blocks: lambda do fout_list (files.map do |file| menu_with_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: lambda { fout_list list_recent_output( @options[:saved_stdout_folder], @options[:saved_stdout_glob], @options[:list_count] ) }, list_recent_scripts: lambda { fout_list list_recent_scripts( options[:saved_script_folder], options[:saved_script_glob], options[:list_count] ) }, 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 } } return if execute_simple_commands(simple_commands) files = prepare_file_list(options) execute_block_logic(files) return unless @options[:output_saved_script_filename] fout "saved_filespec: #{@execute_script_filespec}" rescue StandardError => err warn(error = "ERROR ** MarkParse.execute_code_block_based_on_options(); #{err.inspect}") binding.pry if $tap_enable raise ArgumentError, error end # Executes command based on the provided option keys def execute_simple_commands(simple_commands) simple_commands.each_key do |key| if @options[key] simple_commands[key].call return true end end false end ## post-parse options configuration # def finalize_cli_argument_processing(rest) ## position 0: file or folder (optional) # if (pos = rest.shift)&.present? if Dir.exist?(pos) @options[:path] = pos elsif File.exist?(pos) @options[:filename] = pos else raise FileMissingError, pos, caller end end ## position 1: block name (optional) # block_name = rest.shift @options[:block_name] = block_name if block_name.present? end # Formats multiline body content as a title string. # @param body_lines [Array] The lines of body content. # @return [String] Formatted title. def format_multiline_body_as_title(body_lines) body_lines.map.with_index do |line, index| index.zero? ? line : " #{line}" end.join("\n") << "\n" end ## summarize blocks # def get_block_summary(call_options, fcb) opts = optsmerge call_options # return fcb.body unless opts[:struct] return fcb unless opts[:bash] fcb.call = fcb.title.match(Regexp.new(opts[:block_calls_scan]))&.fetch(1, nil) titlexcall = if fcb.call fcb.title.sub("%#{fcb.call}", '') else fcb.title end bm = extract_named_captures_from_option(titlexcall, opts[:block_name_match]) fcb.stdin = extract_named_captures_from_option(titlexcall, opts[:block_stdin_scan]) fcb.stdout = extract_named_captures_from_option(titlexcall, opts[:block_stdout_scan]) shell_color_option = SHELL_COLOR_OPTIONS[fcb[:shell]] fcb.title = fcb.oname = bm && bm[1] ? bm[:title] : titlexcall fcb.dname = if shell_color_option && opts[shell_color_option].present? fcb.oname.send(opts[shell_color_option].to_sym) else fcb.oname end fcb end # Handles the link-back operation. # # @param opts [Hash] Configuration options hash. # @return [Array] A tuple containing a LoadFile flag and an empty string. def handle_back_link(opts) history_state_pop(opts) [LoadFile::Load, ''] end # Handles the execution and display of remainder blocks from a selected menu item. # # @param mdoc [Object] Document object containing code blocks. # @param opts [Hash] Configuration options hash. # @param selected [Hash] Selected item from the menu. # @return [Array] A tuple containing a LoadFile flag and an empty string. # @note The function can prompt the user for approval before executing code if opts[:user_must_approve] is true. def handle_remainder_blocks(mdoc, opts, selected) required_lines = collect_required_code_lines(mdoc, selected, opts: opts) if opts[:output_script] || opts[:user_must_approve] display_required_code(opts, required_lines) end allow = if opts[:user_must_approve] prompt_for_user_approval(opts, required_lines) else true end opts[:s_ir_approve] = allow if opts[:s_ir_approve] execute_approved_block(opts, required_lines) end [LoadFile::Reuse, ''] end # Handles the link-shell operation. # # @param opts [Hash] Configuration options hash. # @param body [Array] The body content. # @param mdoc [Object] Document object containing code blocks. # @return [Array] A tuple containing a LoadFile flag and a block name. def handle_shell_link(opts, body, mdoc) data = body.present? ? YAML.load(body.join("\n")) : {} data_file = data.fetch('file', nil) return [LoadFile::Reuse, ''] unless data_file history_state_push(mdoc, data_file, opts) data.fetch('vars', []).each do |var| ENV[var[0]] = var[1].to_s end [LoadFile::Load, data.fetch('block', '')] end # Handles options for the shell. # # @param opts [Hash] Configuration options hash. # @param selected [Hash] Selected item from the menu. # @return [Array] A tuple containing a LoadFile::Reuse flag and an empty string. def handle_shell_opts(opts, selected, tgt2 = nil) data = YAML.load(selected[:body].join("\n")) data.each_key do |key| opts[key.to_sym] = value = data[key] tgt2[key.to_sym] = value if tgt2 next unless opts[:menu_opts_set_format].present? print format( opts[:menu_opts_set_format], { key: key, value: value } ).send(opts[:menu_opts_set_color].to_sym) end [LoadFile::Reuse, ''] end # Handles reading and processing lines from a given IO stream # # @param stream [IO] The IO stream to read from (e.g., stdout, stderr, stdin). # @param file_type [Symbol] The type of file to which the stream corresponds. def handle_stream(opts, stream, file_type, swap: false) Thread.new do until (line = stream.gets).nil? @execute_files[file_type] = @execute_files[file_type] + [line.strip] print line if opts[:output_stdout] yield line if block_given? end rescue IOError #d 'stdout IOError, thread killed, do nothing' end end def history_state_exist? history = ENV.fetch(MDE_HISTORY_ENV_NAME, '') history.present? ? history : nil end def history_state_partition(opts) unit, rest = StringUtil.partition_at_first( ENV.fetch(MDE_HISTORY_ENV_NAME, ''), opts[:history_document_separator] ) { unit: unit, rest: rest }.tap_inspect end def history_state_pop(opts) state = history_state_partition(opts) opts[:filename] = state[:unit] ENV[MDE_HISTORY_ENV_NAME] = state[:rest] delete_required_temp_file end def history_state_push(mdoc, data_file, opts) [data_file, opts[:block_name]].tap_inspect 'filename, blockname' new_history = opts[:filename] + opts[:history_document_separator] + ENV.fetch(MDE_HISTORY_ENV_NAME, '') opts[:filename] = data_file write_required_blocks_to_temp_file(mdoc, opts[:block_name], opts) ENV[MDE_HISTORY_ENV_NAME] = new_history end ## Sets up the options and returns the parsed arguments # def initialize_and_parse_cli_options @options = base_options read_configuration_file!(@options, ".#{MarkdownExec::APP_NAME.downcase}.yml") @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| menu_option_append opts, @options, item end end @option_parser.load @option_parser.environment rest = @option_parser.parse!(arguments_for_mde) @options[:s_pass_args] = ARGV[rest.count + 1..] rest end def initialize_and_save_execution_output return unless @options[:save_execution_output] @options[:logged_stdout_filename] = SavedAsset.stdout_name(blockname: @options[:block_name], filename: File.basename(@options[:filename], '.*'), prefix: @options[:logged_stdout_filename_prefix], time: Time.now.utc) @logged_stdout_filespec = @options[:logged_stdout_filespec] = File.join @options[:saved_stdout_folder], @options[:logged_stdout_filename] @logged_stdout_filespec = @options[:logged_stdout_filespec] write_execution_output_to_file end # Initializes variables for regex and other states def initialize_state(opts) { fenced_start_and_end_regex: Regexp.new(opts[:fenced_start_and_end_regex]), fenced_start_extended_regex: Regexp.new(opts[:fenced_start_extended_regex]), fcb: FCB.new, in_fenced_block: false, headings: [] } end # Main function to iterate through blocks in file def iter_blocks_in_file(opts = {}, &block) return unless check_file_existence(opts[:filename]) state = initialize_state(opts) selected_messages = yield :filter cfile.readlines(opts[:filename]).each do |line| next unless line update_line_and_block_state(line, state, opts, selected_messages, &block) end end ## # Returns a lambda expression based on the given procname. # @param procname [String] The name of the process to generate a lambda for. # @param options [Hash] The options hash, necessary for some lambdas to access. # @return [Lambda] The corresponding lambda expression. def lambda_for_procname(procname, options) case procname when 'debug' lambda { |value| tap_config value: value } when 'exit' ->(_) { exit } when 'help' lambda { |_| fout menu_help exit } when 'path' ->(value) { read_configuration_file!(options, value) } when 'show_config' lambda { |_| finalize_cli_argument_processing(options) fout options.sort_by_key.to_yaml } when 'val_as_bool' lambda { |value| value.instance_of?(::String) ? (value.chomp != '0') : value } when 'val_as_int' ->(value) { value.to_i } when 'val_as_str' ->(value) { value.to_s } when 'version' lambda { |_| fout MarkdownExec::VERSION exit } else procname end 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.for_yaml(item[:default])}", item[:description].present? ? item[:description] : nil ].compact.join(' # ') end.compact.sort end def list_files_per_options(options) list_files_specified( determine_filename( specified_filename: options[:filename]&.present? ? options[:filename] : nil, specified_folder: options[:path], default_filename: 'README.md', default_folder: '.' ) ) end ## Searches for files based on the specified or default filenames and folders # def list_files_specified(fn, filetree = nil) return Dir.glob(fn) unless filetree filetree.select do |filename| filename == fn || filename.match(/^#{fn}$/) || filename.match(%r{^#{fn}/.+$}) end end def list_markdown_files_in_path Dir.glob(File.join(@options[:path], @options[:md_filename_glob])) end ## output type (body string or full object) per option struct and bash # def list_named_blocks_in_file(call_options = {}, &options_block) opts = optsmerge call_options, options_block blocks_per_opts( menu_from_file(opts.merge(struct: true)).select do |fcb| Filter.fcb_select?(opts.merge(no_chrome: true), fcb) end, opts ) end # return true if values were modified # execute block once per filename # def load_auto_blocks(opts, blocks_in_file) return unless opts[:document_load_opts_block_name].present? return if opts[:s_most_recent_filename] == opts[:filename] block = block_find(blocks_in_file, :oname, opts[:document_load_opts_block_name]) return unless block handle_shell_opts(opts, block, @options) opts[:s_most_recent_filename] = opts[:filename] true end def mdoc_and_menu_from_file(opts) menu_blocks = menu_from_file(opts.merge(struct: true)) mdoc = MDoc.new(menu_blocks) do |nopts| opts.merge!(nopts) end [menu_blocks, mdoc] end ## Handles the file loading and returns the blocks in the file and MDoc instance # def mdoc_menu_and_selected_from_file(opts) blocks_in_file, mdoc = mdoc_and_menu_from_file(opts) if load_auto_blocks(opts, blocks_in_file) # recreate menu with new options # blocks_in_file, mdoc = mdoc_and_menu_from_file(opts) end blocks_menu = mdoc.fcbs_per_options(opts.merge(struct: true)) add_menu_chrome_blocks!(blocks_menu) [blocks_in_file, blocks_menu, mdoc] end def menu_chrome_colored_option(opts, option_symbol = :menu_option_back_name) if opts[:menu_chrome_color] menu_chrome_formatted_option(opts, option_symbol).send(opts[:menu_chrome_color].to_sym) else menu_chrome_formatted_option(opts, option_symbol) end end def menu_chrome_formatted_option(opts, option_symbol = :menu_option_back_name) val1 = safeval(opts.fetch(option_symbol, '')) val1 unless opts[:menu_chrome_format] format(opts[:menu_chrome_format], val1) end def menu_export(data = menu_for_optparse) data.map do |item| item.delete(:procname) item end.to_yaml end def menu_for_blocks(menu_options) options = calculated_options.merge menu_options menu = [] iter_blocks_in_file(options) do |btype, fcb| case btype when :filter %i[blocks line] when :line if options[:menu_divider_match] && (mbody = fcb.body[0].match(options[:menu_divider_match])) menu.push FCB.new({ dname: mbody[:name], oname: mbody[:name], disabled: '' }) end if options[:menu_note_match] && (mbody = fcb.body[0].match(options[:menu_note_match])) menu.push FCB.new({ dname: mbody[:name], oname: mbody[:name], disabled: '' }) end when :blocks menu += [fcb.oname] end end menu end ## # Generates a menu suitable for OptionParser from the menu items defined in YAML format. # @return [Array] The array of option hashes for OptionParser. def menu_for_optparse menu_from_yaml.map do |menu_item| menu_item.merge( opt_name: menu_item[:opt_name]&.to_sym, proccode: lambda_for_procname(menu_item[:procname], options) ) end end ## # Returns a list of blocks in a given file, including dividers, tasks, and other types of blocks. # The list can be customized via call_options and options_block. # # @param call_options [Hash] Options passed as an argument. # @param options_block [Proc] Block for dynamic option manipulation. # @return [Array] An array of FCB objects representing the blocks. # def menu_from_file(call_options = {}, &options_block) opts = optsmerge(call_options, options_block) use_chrome = !opts[:no_chrome] blocks = [] iter_blocks_in_file(opts) do |btype, fcb| case btype when :blocks append_block_summary(blocks, fcb, opts) when :filter # what btypes are responded to? %i[blocks line] when :line create_and_add_chrome_blocks(blocks, fcb, opts, use_chrome) end end blocks rescue StandardError => err warn(error = "ERROR ** MarkParse.menu_from_file(); #{err.inspect}") warn(caller[0..4]) raise StandardError, error end def menu_help @option_parser.help end def menu_iter(data = menu_for_optparse, &block) data.map(&block) end def menu_option_append(opts, options, item) unless item[:long_name].present? || item[:short_name].present? return end opts.on(*[ # - long name if item[:long_name].present? "--#{item[:long_name]}#{item[:arg_name].present? ? " #{item[:arg_name]}" : ''}" end, # - short name item[:short_name].present? ? "-#{item[:short_name]}" : nil, # - description and default [item[:description], (if item[:default].present? "[#{value_for_cli item[:default]}]" end)].compact.join(' '), # apply proccode, if present, to value # save value to options hash if option is named # lambda { |value| (item[:proccode] ? item[:proccode].call(value) : value).tap do |converted| options[item[:opt_name]] = converted if item[:opt_name] end } ].compact) end def menu_with_block_labels(call_options = {}) opts = options.merge(call_options) menu_from_file(opts).map do |fcb| BlockLabel.make( filename: opts[:filename], headings: fcb.fetch(:headings, []), menu_blocks_with_docname: opts[:menu_blocks_with_docname], menu_blocks_with_headings: opts[:menu_blocks_with_headings], title: fcb[:title], text: fcb[:text], body: fcb[:body] ) end.compact end def next_block_name_from_command_line_arguments(opts) if opts[:s_cli_rest].present? opts[:block_name] = opts[:s_cli_rest].pop false # repeat_menu else true # repeat_menu end 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 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| if @options[name] oq << [label, @options[name], DISPLAY_LEVEL_ADMIN] end 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 # Prepare the blocks menu by adding labels and other necessary details. # # @param blocks_in_file [Array] The list of blocks from the file. # @param opts [Hash] The options hash. # @return [Array] The updated blocks menu. def prepare_blocks_menu(blocks_in_file, opts) # next if fcb.fetch(:disabled, false) # next unless fcb.fetch(:name, '').present? replace_consecutive_blanks(blocks_in_file).map do |fcb| next if Filter.prepared_not_in_menu?(opts, fcb) fcb.merge!( name: fcb.dname, label: BlockLabel.make( body: fcb[:body], filename: opts[:filename], headings: fcb.fetch(:headings, []), menu_blocks_with_docname: opts[:menu_blocks_with_docname], menu_blocks_with_headings: opts[:menu_blocks_with_headings], text: fcb[:text], title: fcb[:title] ) ) fcb.to_h end.compact end # Prepares and fetches file listings def prepare_file_list(options) list_files_per_options(options) end def process_fenced_block(fcb, opts, selected_messages, &block) fcb.oname = fcb.dname = fcb.title || '' return unless fcb.body update_title_from_body(fcb) if block && selected_messages.include?(:blocks) && Filter.fcb_select?(opts, fcb) block.call :blocks, fcb end end def process_line(line, _opts, selected_messages, &block) return unless block && selected_messages.include?(:line) # dp 'text outside of fcb' fcb = FCB.new fcb.body = [line] block.call(:line, fcb) end class MenuOptions YES = 1 NO = 2 SCRIPT_TO_CLIPBOARD = 3 SAVE_SCRIPT = 4 end ## # Presents a menu to the user for approving an action and performs additional tasks based on the selection. # The function provides options for approval, rejection, copying data to clipboard, or saving data to a file. # # @param opts [Hash] A hash containing various options for the menu. # @param required_lines [Array] Lines of text or code that are subject to user approval. # # @option opts [String] :prompt_approve_block Prompt text for the approval menu. # @option opts [String] :prompt_yes Text for the 'Yes' choice in the menu. # @option opts [String] :prompt_no Text for the 'No' choice in the menu. # @option opts [String] :prompt_script_to_clipboard Text for the 'Copy to Clipboard' choice in the menu. # @option opts [String] :prompt_save_script Text for the 'Save to File' choice in the menu. # # @return [Boolean] Returns true if the user approves (selects 'Yes'), false otherwise. ## def prompt_for_user_approval(opts, required_lines) # Present a selection menu for user approval. sel = @prompt.select(opts[:prompt_approve_block], filter: true) do |menu| menu.default MenuOptions::YES menu.choice opts[:prompt_yes], MenuOptions::YES menu.choice opts[:prompt_no], MenuOptions::NO menu.choice opts[:prompt_script_to_clipboard], MenuOptions::SCRIPT_TO_CLIPBOARD menu.choice opts[:prompt_save_script], MenuOptions::SAVE_SCRIPT end if sel == MenuOptions::SCRIPT_TO_CLIPBOARD copy_to_clipboard(required_lines) elsif sel == MenuOptions::SAVE_SCRIPT save_to_file(opts, required_lines) end sel == MenuOptions::YES rescue TTY::Reader::InputInterrupt exit 1 end def prompt_select_continue(opts) sel = @prompt.select( opts[:prompt_after_bash_exec], filter: true, quiet: true ) do |menu| menu.choice opts[:prompt_yes] menu.choice opts[:prompt_exit] end sel == opts[:prompt_exit] ? MenuState::EXIT : MenuState::CONTINUE rescue TTY::Reader::InputInterrupt exit 1 end # :reek:UtilityFunction ### temp def read_configuration_file!(options, configuration_path) return unless File.exist?(configuration_path) options.merge!((YAML.load(File.open(configuration_path)) || {}) .transform_keys(&:to_sym)) end # Reads required code blocks from a temporary file specified by an environment variable. # # @return [Array] An array containing the lines read from the temporary file. # @note Relies on the 'MDE_LINK_REQUIRED_FILE' environment variable to locate the file. def read_required_blocks_from_temp_file temp_blocks = [] temp_blocks_file_path = ENV.fetch('MDE_LINK_REQUIRED_FILE', nil) if temp_blocks_file_path.nil? || temp_blocks_file_path.empty? return temp_blocks end if File.exist?(temp_blocks_file_path) temp_blocks = File.readlines(temp_blocks_file_path, chomp: true) end temp_blocks end # Replace duplicate blanks (where :oname is not present) with a single blank line. # # @param [Array] lines Array of hashes to process. # @return [Array] Cleaned array with consecutive blanks collapsed into one. def replace_consecutive_blanks(lines) lines.chunk_while do |i, j| i[:oname].to_s.empty? && j[:oname].to_s.empty? end.map do |chunk| if chunk.any? do |line| line[:oname].to_s.strip.empty? end chunk.first else chunk end end.flatten end def run clear_required_file execute_block_with_error_handling(initialize_and_parse_cli_options) delete_required_temp_file rescue StandardError => err warn(error = "ERROR ** MarkParse.run(); #{err.inspect}") binding.pry if $tap_enable raise ArgumentError, error end def run_last_script filename = SavedFilesMatcher.most_recent(@options[:saved_script_folder], @options[:saved_script_glob]) return unless filename saved_name_split filename @options[:save_executed_script] = false select_approve_and_execute_block end def safeval(str) eval(str) rescue StandardError warn $! binding.pry if $tap_enable raise StandardError, $! end def save_to_file(opts, required_lines) write_command_file(opts.merge(save_executed_script: true), required_lines) fout "File saved: #{@options[:saved_filespec]}" end def saved_name_split(name) # rubocop:disable Layout/LineLength mf = /#{@options[:saved_script_filename_prefix]}_(?