class SyntaxSuggest::CaptureCodeContext

将“无效代码块”转换为有用的上下文

该算法分为三个主要阶段

  1. 清理/格式化输入源代码

  2. 搜索无效代码块

  3. 将无效代码块格式化为有意义的内容

此类处理第三部分。

该算法在第 2 步中非常擅长捕获单个块中的所有语法错误,但是结果可能包含歧义。 人类擅长模式匹配和过滤,并且可以在头脑中删除无关的数据,但是他们无法添加不存在的额外数据。

对于已知的歧义情况,此类会向歧义添加上下文,以便程序员拥有完整的信息。

除了处理这些歧义之外,它还会捕获周围的代码上下文信息

puts block.to_s # => "def bark"

context = CaptureCodeContext.new(
  blocks: block,
  code_lines: code_lines
)

lines = context.call.map(&:original)
puts lines.join
# =>
  class Dog
    def bark
  end

属性

code_lines[R]

公共类方法

new(blocks:, code_lines:) 点击以切换源代码
# File syntax_suggest/capture_code_context.rb, line 51
def initialize(blocks:, code_lines:)
  @blocks = Array(blocks)
  @code_lines = code_lines
  @visible_lines = @blocks.map(&:visible_lines).flatten
  @lines_to_output = @visible_lines.dup
end

公共实例方法

call() 点击以切换源代码
# File syntax_suggest/capture_code_context.rb, line 58
def call
  @blocks.each do |block|
    capture_first_kw_end_same_indent(block)
    capture_last_end_same_indent(block)
    capture_before_after_kws(block)
    capture_falling_indent(block)
  end

  sorted_lines
end
capture_before_after_kws(block) 点击以切换源代码

显示周围的 kw/end 对

显示这些额外对的原因是当仅匹配一个可见行时出现歧义的情况。

例如

1  class Dog
2    def bark
4    def eat
5    end
6  end

在这种情况下,第 2 行可能缺少一个 'end',或者第 4 行是错误添加的额外行(这种情况会发生)。

当我们检测到上述问题时,它会将问题显示为仅在第 2 行

2    def bark

显示“邻居”关键字对会提供额外的上下文

2    def bark
4    def eat
5    end
# File syntax_suggest/capture_code_context.rb, line 127
def capture_before_after_kws(block)
  return unless block.visible_lines.count == 1

  around_lines = Capture::BeforeAfterKeywordEnds.new(
    code_lines: @code_lines,
    block: block
  ).call

  around_lines -= block.lines

  @lines_to_output.concat(around_lines)
end
capture_falling_indent(block) 点击以切换源代码

显示“下降”缩进提供的代码周围的上下文

转换为

it "foo" do

变成

class OH
  def hello
    it "foo" do
  end
end
# File syntax_suggest/capture_code_context.rb, line 91
def capture_falling_indent(block)
  Capture::FallingIndentLines.new(
    block: block,
    code_lines: @code_lines
  ).call do |line|
    @lines_to_output << line
  end
end
capture_first_kw_end_same_indent(block) 点击以切换源代码

‘capture_last_end_same_indent` 的逻辑反向

当一个无效的代码块在另一个 `end` 之后缺少关键字 `end` 时,不清楚哪个 end 缺少关键字。

举个例子

class Dog       # 1
    puts "woof" # 2
  end           # 3
end             # 4

问题行将被识别为

> end            # 4

发生这种情况是因为第 1、2 和 3 行在技术上是有效的代码,并且首先展开,被认为是有效的并且被隐藏。我们需要取消隐藏第 1 行上的匹配关键字。 也反向操作,如果有不匹配的 end,也显示它

# File syntax_suggest/capture_code_context.rb, line 221
def capture_first_kw_end_same_indent(block)
  return if block.visible_lines.length != 1
  return unless block.visible_lines.first.is_end?

  visible_line = block.visible_lines.first
  lines = @code_lines[block.lines.first.index..visible_line.index]
  matching_kw = lines.reverse.detect { |line| line.indent == block.current_indent && line.is_kw? }
  return unless matching_kw

  @lines_to_output << matching_kw

  kw_count = 0
  end_count = 0
  orphan_end = @code_lines[matching_kw.index..visible_line.index].detect do |line|
    kw_count += 1 if line.is_kw?
    end_count += 1 if line.is_end?

    end_count >= kw_count
  end

  return unless orphan_end
  @lines_to_output << orphan_end
end
capture_last_end_same_indent(block) 点击以切换源代码

当一个无效的代码块在另一个 end 之前缺少一个 end 的关键字时,不清楚哪个关键字缺少 end

举个例子

class Dog       # 1
  def bark      # 2
    puts "woof" # 3
end             # 4

但是由于 github.com/ruby/syntax_suggest/issues/32,问题行将被识别为

> class Dog       # 1

因为第 2、3 和 4 行在技术上是有效的代码,并且首先展开,被认为是有效的并且被隐藏。 我们需要取消隐藏匹配的 end 行 4。也反向操作,如果有不匹配的关键字,也显示它

# File syntax_suggest/capture_code_context.rb, line 161
def capture_last_end_same_indent(block)
  return if block.visible_lines.length != 1
  return unless block.visible_lines.first.is_kw?

  visible_line = block.visible_lines.first
  lines = @code_lines[visible_line.index..block.lines.last.index]

  # Find first end with same indent
  # (this would return line 4)
  #
  #   end             # 4
  matching_end = lines.detect { |line| line.indent == block.current_indent && line.is_end? }
  return unless matching_end

  @lines_to_output << matching_end

  # Work backwards from the end to
  # see if there are mis-matched
  # keyword/end pairs
  #
  # Return the first mis-matched keyword
  # this would find line 2
  #
  #     def bark      # 2
  #       puts "woof" # 3
  #   end             # 4
  end_count = 0
  kw_count = 0
  kw_line = @code_lines[visible_line.index..matching_end.index].reverse.detect do |line|
    end_count += 1 if line.is_end?
    kw_count += 1 if line.is_kw?

    !kw_count.zero? && kw_count >= end_count
  end
  return unless kw_line
  @lines_to_output << kw_line
end
sorted_lines() 点击以切换源代码
# File syntax_suggest/capture_code_context.rb, line 69
def sorted_lines
  @lines_to_output.select!(&:not_empty?)
  @lines_to_output.uniq!
  @lines_to_output.sort!

  @lines_to_output
end