25行Ruby代码编写Shell(译)

如果使用Linux或Mac,每次你打开一个终端就在使用shell应用程序。shell是一个接口,帮助你在你的系统执行命令。

此外,外壳也提供环境变量和有用的功能,如一个命令历史和自动完成。

如果你是喜欢深入学习的人,这篇文章将十分适合你!

一个Shell是如何工作的呢?

为了构建自己的shell应用程序,让我们思考一个shell是什么:第一,有一个提示,通常与一些额外的信息,比如你的当前用户和当前目录,然后输入一个命令,当你按下enter结果都显示在你的屏幕上。

是的,这听起来非常基础,但是这没有提醒你什么吗?

如果你想到 pry那么你是对的! shell基本上就是适用于您的操作系统的REPL(Read-Eval-Print-Loop)。

了解这些后,我们可以写您的shell的第一个版本:

prompt = "> "
 
print prompt
 
while (input = gets.chomp)
  break if input == "exit"
 
  system(input)
  print prompt
end

这将给我们一个最小,但能工作的shell。我们可以通过使用许多其他类似应用程序使用的库来改善。那个库被称为 Readline.

使用Readline库

Readline是Ruby标准库的一部分,所以不用安装,你只需要 require它。

使用Readline的优势之一是,它可以自动保持命令历史。它还可以照顾打印命令提示符和许多其他的事情。

这里是我们shell的v2,这次使用了Readline:

require 'readline'
 
while input = Readline.readline("> ", true)
  break if input == "exit"
 
  system(input)
end

太好了,我们摆脱了 puts提示,而且现在我们有一些Readline的强大的功能。例如,我们可以使用键盘快捷键来删除一个字(CTRL + W),甚至搜索历史(CTRL + R)!

让我们添加一个新的命令打印完整的历史:

require 'readline'
 
while input = Readline.readline("> ", true)
  break                       if input == "exit"
  puts Readline::HISTORY.to_a if input == "hist"
 
  # Remove blank lines from history
  Readline::HISTORY.pop if input == ""
 
  system(input)
end

有趣的事实:如果你用pry尝试这段代码,你会看到pry的命令历史记录!原因是pry也使用 Readline, 而Readline::HISTORY用来共享状态。

现在你可以输入 hist 获取命令历史记录:)

添加自动完成

谢谢这个你喜欢的外壳的自动完成功能帮你少打许多字。Readline很容易地将此功能集成到您的shell。

让我们从我们的命令历史来实现自动完成命令。

例子:

comp = proc { |s| LIST.grep(/^#{Regexp.escape(s)}/) }
 
Readline.completion_append_character = " "
Readline.completion_proc = comp
 
## rest of the code goes here ##

这个代码中,你应该能够用键来自动完成输入命令。现在让我们再进一步,添加目录自动完成。

例子:

comp = proc do |s|
  directory_list = Dir.glob("#{s}*")
 
  if directory_list.size > 0
    directory_list
  else
    Readline::HISTORY.grep(/^#{Regexp.escape(s)}/)
  end
end

completion_proc返回列表可能的候选人,在这种情况下,我们只需要用Dir.glob检查输入的字符串是否使用目录名称的一部分。Readline将处理其余的事情!

系统实现方法

现在你已经有一个不太差的外壳,带有命令历史&自动完成,只有25行代码:)

但是总有一些事情我想深入挖掘,所以你可以看到实际执行一个命令的时候,在幕后发生了什么。

这是用的system方法,这种方法在C语言只是发送你的命令到/bin/sh,这是一个shell应用程序。让我们来看看你可以在Ruby中实现/bin/sh所做的。

注意:这个只会工作在Linux / Mac:)

system方法

def system(command)
  fork {
    exec(command)
  }
end

这里发生的是, fork为当前进程创建一个新的副本,然后通过 exec方法,这个进程被我们要运行的命令替换。这在Linux编程是一个很常见的模式。

如果你不fork,那么当前进程将被取代,这意味着当您正在运行的命令(ls, cd或其他)完成,你的Ruby程序将终止。

你可以看到这里发生的:

def system(command)
  exec(command)
end
 
system('ls')
 
# This code will never run!
puts "after system"

结论

在本帖中,你知道一个shell是一个添加的与系统进行交互的接口(想想 irb / pry)。您还了解了如何通过使用强大的Readline库构建自己的壳,它提供了许多内置功能,如历史&自动完成(但你必须定义如何工作)。

然后您学习了 fork+ exec模式,通常用于Linux编程项目。

如果你喜欢这篇文章你能帮我一个忙分享与你所有的喜欢Ruby的朋友吗?它将帮助博客成长和更多的人可以学习:)

原英文链接


广告时间:

Ruby是知识海洋的贝壳,一起来捡吧。

高性价比培训,欢迎了解和交流。简学互动Ruby、Rails基础培训

你可能感兴趣的:(25行Ruby代码编写Shell(译))