Skip to content

Commit ff81a29

Browse files
committed
Add Prism::Source#line_to_byte_offset and replace direct accesses to offsets
1 parent 71fcb89 commit ff81a29

File tree

4 files changed

+58
-2
lines changed

4 files changed

+58
-2
lines changed

lib/prism/parse_result.rb

Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,15 @@ def slice(byte_offset, length)
7676
source.byteslice(byte_offset, length) or raise
7777
end
7878

79+
# Converts the line number to a byte offset corresponding to the start of that line
80+
def line_to_byte_offset(line)
81+
l = line - @start_line
82+
if l < 0 || l >= offsets.size
83+
raise ArgumentError, "line #{line} is out of range"
84+
end
85+
offsets[l]
86+
end
87+
7988
# Binary search through the offsets to find the line number for the given
8089
# byte offset.
8190
def line(byte_offset)

sig/prism/parse_result.rbs

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ module Prism
1414
def encoding: () -> Encoding
1515
def lines: () -> Array[String]
1616
def slice: (Integer byte_offset, Integer length) -> String
17+
def line_to_byte_offset: (Integer line) -> Integer
1718
def line: (Integer byte_offset) -> Integer
1819
def line_start: (Integer byte_offset) -> Integer
1920
def line_end: (Integer byte_offset) -> Integer

templates/lib/prism/node.rb.erb

Lines changed: 1 addition & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -184,8 +184,7 @@ module Prism
184184
queue = [self] #: Array[Prism::node]
185185
result = [] #: Array[Prism::node]
186186

187-
line_offset = source.offsets[line - 1] or raise
188-
search_offset = line_offset + column
187+
search_offset = source.line_to_byte_offset(line) + column
189188

190189
while (node = queue.shift)
191190
result << node

test/prism/ruby/source_test.rb

Lines changed: 47 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
# frozen_string_literal: true
2+
3+
require_relative "../test_helper"
4+
5+
module Prism
6+
class SourceTest < TestCase
7+
def test_line_to_byte_offset
8+
parse_result = Prism.parse(<<~SRC)
9+
abcd
10+
efgh
11+
ijkl
12+
SRC
13+
source = parse_result.source
14+
15+
assert_equal 0, source.line_to_byte_offset(1)
16+
assert_equal 5, source.line_to_byte_offset(2)
17+
assert_equal 10, source.line_to_byte_offset(3)
18+
assert_equal 15, source.line_to_byte_offset(4)
19+
e = assert_raise(ArgumentError) { source.line_to_byte_offset(5) }
20+
assert_equal "line 5 is out of range", e.message
21+
e = assert_raise(ArgumentError) { source.line_to_byte_offset(0) }
22+
assert_equal "line 0 is out of range", e.message
23+
e = assert_raise(ArgumentError) { source.line_to_byte_offset(-1) }
24+
assert_equal "line -1 is out of range", e.message
25+
end
26+
27+
def test_line_to_byte_offset_with_start_line
28+
parse_result = Prism.parse(<<~SRC, line: 11)
29+
abcd
30+
efgh
31+
ijkl
32+
SRC
33+
source = parse_result.source
34+
35+
assert_equal 0, source.line_to_byte_offset(11)
36+
assert_equal 5, source.line_to_byte_offset(12)
37+
assert_equal 10, source.line_to_byte_offset(13)
38+
assert_equal 15, source.line_to_byte_offset(14)
39+
e = assert_raise(ArgumentError) { source.line_to_byte_offset(15) }
40+
assert_equal "line 15 is out of range", e.message
41+
e = assert_raise(ArgumentError) { source.line_to_byte_offset(10) }
42+
assert_equal "line 10 is out of range", e.message
43+
e = assert_raise(ArgumentError) { source.line_to_byte_offset(9) }
44+
assert_equal "line 9 is out of range", e.message
45+
end
46+
end
47+
end

0 commit comments

Comments
 (0)