#!/usr/bin/env ruby # frozen_string_literal: true require 'English' require 'optparse' require 'reline' require 'rpl' class RplRepl def initialize( interpreter: Rpl.new ) begin @stty_save = `stty -g`.chomp rescue Error => e puts "Couldn't get terminal characteristics" pp e end @interpreter = interpreter end def run! print_stack unless @interpreter.stack.empty? Reline.completion_proc = proc do |s| ( @interpreter.dictionary.words.keys + @interpreter.dictionary.vars.keys ).grep(/^#{Regexp.escape(s)}/) end Reline.completion_append_character = ' ' prefill = '' loop do unless prefill.empty? Reline.pre_input_hook = lambda do Reline.insert_text( prefill.to_s ) Reline.redisplay Reline.pre_input_hook = nil end end input = Reline.readline( ' ', true ) break if input.nil? || input.strip == 'quit' prefill = '' if input.strip == 'edit' prefill = @interpreter.stack.pop.to_s elsif input.strip == 'history' history = Reline::HISTORY.map { |line| "\"#{line}\"" }.join(' ') @interpreter.run!( "{ #{history} }" ) elsif input.empty? # Remove blank lines from history Reline::HISTORY.pop else begin @interpreter.run!( input ) rescue ArgumentError => e pp e end end print_stack rescue Interrupt puts '^C' `stty #{@stty_save}` if @stty_save exit 0 end end def print_stack stack_size = @interpreter.stack.size @interpreter.stack.each_with_index do |elt, i| puts "#{stack_size - i}: #{elt}" end end end options = { run_REPL: ARGV.empty?, persistence_filename: File.expand_path( '~/.local/state/rpl.rb/machine' ), live_persistence: true, files: [], programs: [] } Version = Rpl::VERSION OptionParser.new do |opts| opts.on('-s', '--state filename', "persist state in filename (default: #{options[:persistence_filename]}) (will be created if needed)") do |filename| options[:persistence_filename] = File.expand_path( filename ) end opts.on('-q', '--no-state', 'Do not load persisted state') do options[:persistence_filename] = nil end opts.on('-d', '--no-persist', 'Do not persist state') do options[:live_persistence] = false end opts.on('-c', '--code "program"', 'run provided "program"') do |program| options[:programs] << program end opts.on('-f', '--file program.rpl', 'load program.rpl') do |filename| options[:files] << filename end opts.on('-i', '--interactive', 'launch interactive REPL') do options[:run_REPL] = true end end.parse! # Instantiate interpreter interpreter = Rpl.new( persistence_filename: options[:persistence_filename], live_persistence: options[:live_persistence] ) # first run provided files if any options[:files].each do |filename| interpreter.run!( "\"#{File.expand_path( filename )}\" feval" ) end # second run provided code if any options[:programs].each do |program| interpreter.run!( program ) end # third launch REPL if (explicitely or implicitely) asked RplRepl.new( interpreter: interpreter ).run! if options[:run_REPL] interpreter.persist_state