2023-08-15 20:00:54 +03:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2023-08-25 16:23:46 +03:00
|
|
|
require_relative "test_helper"
|
2023-08-15 20:00:54 +03:00
|
|
|
|
|
|
|
return unless defined?(RubyVM::InstructionSequence)
|
|
|
|
|
2023-09-27 19:24:48 +03:00
|
|
|
module Prism
|
2023-08-29 23:18:14 +03:00
|
|
|
class NewlineTest < TestCase
|
|
|
|
base = File.dirname(__dir__)
|
|
|
|
Dir["{lib,test}/**/*.rb", base: base].each do |relative|
|
|
|
|
define_method("test_newline_flags_#{relative}") do
|
|
|
|
assert_newlines(base, relative)
|
|
|
|
end
|
2023-08-15 20:00:54 +03:00
|
|
|
end
|
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
private
|
2023-08-15 20:00:54 +03:00
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
def assert_newlines(base, relative)
|
|
|
|
filepath = File.join(base, relative)
|
|
|
|
source = File.read(filepath, binmode: true, external_encoding: Encoding::UTF_8)
|
|
|
|
expected = rubyvm_lines(source)
|
2023-08-15 20:00:54 +03:00
|
|
|
|
2023-09-27 19:24:48 +03:00
|
|
|
result = Prism.parse_file(filepath)
|
2023-08-29 23:18:14 +03:00
|
|
|
assert_empty result.errors
|
2023-09-27 19:24:48 +03:00
|
|
|
actual = prism_lines(result)
|
2023-08-15 20:00:54 +03:00
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
source.each_line.with_index(1) do |line, line_number|
|
|
|
|
# Lines like `while (foo = bar)` result in two line flags in the
|
|
|
|
# bytecode but only one newline flag in the AST. We need to remove the
|
|
|
|
# extra line flag from the bytecode to make the test pass.
|
|
|
|
if line.match?(/while \(/)
|
|
|
|
index = expected.index(line_number)
|
|
|
|
expected.delete_at(index) if index
|
|
|
|
end
|
2023-08-15 20:00:54 +03:00
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
# Lines like `foo =` where the value is on the next line result in
|
|
|
|
# another line flag in the bytecode but only one newline flag in the
|
|
|
|
# AST.
|
|
|
|
if line.match?(/^\s+\w+ =$/)
|
|
|
|
if source.lines[line_number].match?(/^\s+case/)
|
|
|
|
actual[actual.index(line_number)] += 1
|
|
|
|
else
|
|
|
|
actual.delete_at(actual.index(line_number))
|
|
|
|
end
|
2023-08-15 20:00:54 +03:00
|
|
|
end
|
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
if line.match?(/^\s+\w+ = \[$/)
|
|
|
|
if !expected.include?(line_number) && !expected.include?(line_number + 2)
|
|
|
|
actual[actual.index(line_number)] += 1
|
|
|
|
end
|
2023-08-15 20:00:54 +03:00
|
|
|
end
|
|
|
|
end
|
2023-08-29 23:18:14 +03:00
|
|
|
|
|
|
|
assert_equal expected, actual
|
2023-08-15 20:00:54 +03:00
|
|
|
end
|
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
def ignore_warnings
|
|
|
|
previous_verbosity = $VERBOSE
|
|
|
|
$VERBOSE = nil
|
|
|
|
yield
|
|
|
|
ensure
|
|
|
|
$VERBOSE = previous_verbosity
|
|
|
|
end
|
2023-08-15 20:00:54 +03:00
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
def rubyvm_lines(source)
|
|
|
|
queue = [ignore_warnings { RubyVM::InstructionSequence.compile(source) }]
|
|
|
|
lines = []
|
2023-08-15 20:00:54 +03:00
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
while iseq = queue.shift
|
|
|
|
lines.concat(iseq.trace_points.filter_map { |line, event| line if event == :line })
|
|
|
|
iseq.each_child { |insn| queue << insn unless insn.label.start_with?("ensure in ") }
|
|
|
|
end
|
2023-08-15 20:00:54 +03:00
|
|
|
|
2023-08-29 23:18:14 +03:00
|
|
|
lines.sort
|
2023-08-15 20:00:54 +03:00
|
|
|
end
|
2023-09-26 20:57:20 +03:00
|
|
|
|
2023-09-27 19:24:48 +03:00
|
|
|
def prism_lines(result)
|
2023-09-26 20:57:20 +03:00
|
|
|
result.mark_newlines!
|
|
|
|
|
|
|
|
queue = [result.value]
|
|
|
|
newlines = []
|
|
|
|
|
|
|
|
while node = queue.shift
|
|
|
|
queue.concat(node.compact_child_nodes)
|
|
|
|
newlines << result.source.line(node.location.start_offset) if node&.newline?
|
|
|
|
end
|
|
|
|
|
2023-09-26 21:23:52 +03:00
|
|
|
newlines.sort
|
2023-09-26 20:57:20 +03:00
|
|
|
end
|
2023-08-15 20:00:54 +03:00
|
|
|
end
|
|
|
|
end
|