csc360-a1-shell/ruby/shell/builtins.rb
Sami Samhuri 4f4e97475b
[ruby] Modernize Ruby shell parsing and expansion, add C compat test mode (#4)
Replace Ruby's old wordexp-like command splitting with a tokenizer and
parser that understands ; and && while honoring quotes and nesting.

Implement richer expansions for command substitution, arithmetic,
parameter defaults (${var:-...}), brace expansion, and escaped
dollar/backtick behavior via shared quote-state handling.

Expand the test suite with parser/expansion edge cases, escaping
parity checks, builtin usage validation, and job-control refresh tests.

Keep C green by adding a compat test profile for c/Makefile test and
by returning nonzero on builtin failures in -c mode, including clearer
`bg` usage output.
2026-02-07 15:18:41 -08:00

104 lines
2.2 KiB
Ruby

require "shell/job_control"
require "shell/logger"
module Shell
class Builtins
EXPORT_VARIABLE_PATTERN = /\$\w+/
attr_reader :job_control, :logger
def initialize(job_control: nil, logger: nil)
logger ||= Logger.instance
@job_control = job_control || JobControl.new(logger: logger)
@logger = logger
end
def exec(name, args)
send(:"builtin_#{name}", args)
end
def builtin?(name)
respond_to?(:"builtin_#{name}")
end
#################
### Built-ins ###
#################
def builtin_bg(args)
if args.empty?
logger.warn "Usage: bg <command>"
return -1
end
cmd = args.shift
job_control.exec_command(cmd, args, background: true)
end
def builtin_bglist(_args)
jobs = job_control.list
puts unless jobs.empty?
jobs.each do |job|
puts job_control.format_job(job)
end
plural = (jobs.count == 1) ? "" : "s"
puts "#{jobs.count} background job#{plural}"
0
end
def builtin_bgkill(args)
if args.count != 1
logger.warn "Usage: bgkill <job>"
return -1
end
job_id = args.shift.to_i
job_control.kill(job_id)
0
end
def builtin_cd(args)
dir = args.first
oldpwd = Dir.pwd
target = case dir
in nil
Dir.home
in "-"
ENV["OLDPWD"] || oldpwd
else
dir
end
Dir.chdir target
ENV["OLDPWD"] = oldpwd
ENV["PWD"] = Dir.pwd
0
end
def builtin_export(args)
if args.count != 1 || args.first.nil? || !args.first.include?("=")
logger.warn "Usage: export NAME=value"
return -1
end
# only supports one variable and doesn't support quoting
name, *value_parts = args.first.strip.split("=")
if name.nil? || name.empty?
logger.warn "#{red("[ERROR]")} Invalid export command"
return -1
else
ENV[name] = value_parts.join("=").gsub(EXPORT_VARIABLE_PATTERN) { ENV[it[1..]] || "" }
end
0
end
def builtin_pwd(_args)
puts Dir.pwd
0
end
def builtin_clear(_args)
print "\033[2J"
0
end
end
end