Feature #6012
closedProc#source_location also return the column
Added by rogerdpack (Roger Pack) over 13 years ago. Updated 10 months ago.
Description
As originally suggested in http://blade.nagaokaut.ac.jp/cgi-bin/scat.rb/ruby/ruby-core/42418
Suggestion/feature request:
have #source_location also return the beginning column where it was defined.
["test.rb", 8, 33]
Thanks!
-roger-
        
           Updated by rogerdpack (Roger Pack) over 13 years ago
          
          
        
        
          
            Actions
          
          #1
            [ruby-core:42580]
          Updated by rogerdpack (Roger Pack) over 13 years ago
          
          
        
        
          
            Actions
          
          #1
            [ruby-core:42580]
        
      
      oops make that a feature request, but I'm unable to edit them myself.
Cheers!
-r
        
           Updated by nahi (Hiroshi Nakamura) over 13 years ago
          
          
        
        
          
            Actions
          
          #2
          Updated by nahi (Hiroshi Nakamura) over 13 years ago
          
          
        
        
          
            Actions
          
          #2
        
      
      - Tracker changed from Bug to Feature
        
           Updated by ko1 (Koichi Sasada) over 13 years ago
          
          
        
        
          
            Actions
          
          #3
            [ruby-core:42928]
          Updated by ko1 (Koichi Sasada) over 13 years ago
          
          
        
        
          
            Actions
          
          #3
            [ruby-core:42928]
        
      
      - Category set to core
- Assignee set to nobu (Nobuyoshi Nakada)
- Target version set to 2.0.0
        
           Updated by trans (Thomas Sawyer) over 13 years ago
          
          
        
        
          
            Actions
          
          #4
            [ruby-core:42951]
          Updated by trans (Thomas Sawyer) over 13 years ago
          
          
        
        
          
            Actions
          
          #4
            [ruby-core:42951]
        
      
      Would this effect Method#source_location too?
I'm not sure I am really digging this idea. First of all it means I have to go back and fix some code. Secondly it means I have to always worry about the additional piece of data even though most of the time it doesn't matter. And if the return can vary between 2 or 3 elements that's another thing to worry with.
On the other hand I can understand that it could be useful information in some cases.
In times like this that I think "Embrace the Object".
proc.source_location  #=>  #<SourceLocation @file="foo.rb" @line=12 @column=14>
And then a few different methods could provide that information in various useful forms.
proc.source_location.to_a  #=>  ["foo.rb", 12, 14]
proc.source_location.to_s  #=>  "foo.rb:12"
proc.source_location.values_at(:file, :line)  #=>  ["foo.rb", 12]
Or what have you.
        
           Updated by trans (Thomas Sawyer) over 13 years ago
          
          
        
        
          
            Actions
          
          #5
            [ruby-core:42952]
          Updated by trans (Thomas Sawyer) over 13 years ago
          
          
        
        
          
            Actions
          
          #5
            [ruby-core:42952]
        
      
      BTW & OT: When is any one going to explain how we format code examples as monospace text on this site?
        
           Updated by drbrain (Eric Hodel) over 13 years ago
          
          
        
        
          
            Actions
          
          #6
            [ruby-core:42961]
          Updated by drbrain (Eric Hodel) over 13 years ago
          
          
        
        
          
            Actions
          
          #6
            [ruby-core:42961]
        
      
      On Feb 26, 2012, at 6:33 AM, Thomas Sawyer wrote:
BTW & OT: When is any one going to explain how we format code examples as monospace text on this site?
Click the RD button and use RD formatting (two spaces).
Here's a bash alias to help, which works for rdoc too.
alias rdindent='pr -l1 -o2'
        
           Updated by trans (Thomas Sawyer) over 13 years ago
          
          
        
        
          
            Actions
          
          #7
            [ruby-core:42964]
          Updated by trans (Thomas Sawyer) over 13 years ago
          
          
        
        
          
            Actions
          
          #7
            [ruby-core:42964]
        
      
      Thanks Eric! I ((never)) noticed that ((%RD%)) "button" before (hardly looks like a button).
Why did it put:
=begin
=end
In the textarea when I clicked on it? ... maybe I'll find out by submitting this...
=begin
What's with the =begin =end?
Testing 1 2 3...
Try ((em)) (({code})) ((|ls|)) ((%var%)).
=end
Sorry for the noise.
        
           Updated by trans (Thomas Sawyer) over 13 years ago
          
          
        
        
          
            Actions
          
          #8
            [ruby-core:42965]
          Updated by trans (Thomas Sawyer) over 13 years ago
          
          
        
        
          
            Actions
          
          #8
            [ruby-core:42965]
        
      
      Well, that failed miserably. LOL :-)
        
           Updated by shyouhei (Shyouhei Urabe) over 13 years ago
          
          
        
        
          
            Actions
          
          #9
          Updated by shyouhei (Shyouhei Urabe) over 13 years ago
          
          
        
        
          
            Actions
          
          #9
        
      
      - Status changed from Open to Assigned
        
           Updated by yhara (Yutaka HARA) almost 13 years ago
          
          
        
        
          
            Actions
          
          #10
            [ruby-core:48281]
          Updated by yhara (Yutaka HARA) almost 13 years ago
          
          
        
        
          
            Actions
          
          #10
            [ruby-core:48281]
        
      
      - Target version changed from 2.0.0 to 2.6
        
           Updated by naruse (Yui NARUSE) almost 8 years ago
          
          
        
        
          
            Actions
          
          #11
          Updated by naruse (Yui NARUSE) almost 8 years ago
          
          
        
        
          
            Actions
          
          #11
        
      
      - Target version deleted (2.6)
        
           Updated by mame (Yusuke Endoh) almost 7 years ago
          
          
        
        
          
            Actions
          
          #12
            [ruby-core:90769]
          Updated by mame (Yusuke Endoh) almost 7 years ago
          
          
        
        
          
            Actions
          
          #12
            [ruby-core:90769]
        
      
      Now the abstract syntax tree has column information, so we can implement this issue. We even add the last point of method.
# test.rb
◆def foo # ◆: line 2, column 0
end★     # ★: line 3, column 3
p method(:foo).source_location #=> ["test.rb", 2, 0, 3, 3]
        
           Updated by ioquatix (Samuel Williams) almost 7 years ago
          
          
        
        
          
            Actions
          
          #13
            [ruby-core:91197]
          Updated by ioquatix (Samuel Williams) almost 7 years ago
          
          
        
        
          
            Actions
          
          #13
            [ruby-core:91197]
        
      
      If changing this API is too complicated due to backwards compatibility, why not introduce new more general API:
Method#source -> Source.new(path, line_number, line_count, code, ...)
Usage:
method.source.code
method.source.path
method.source.location -> [2, 0, 3, 3]
Maybe including byte offset and length would also be useful (for seek).
        
           Updated by ioquatix (Samuel Williams) over 6 years ago
          
          
        
        
          
            Actions
          
          #14
            [ruby-core:91512]
          Updated by ioquatix (Samuel Williams) over 6 years ago
          
          
        
        
          
            Actions
          
          #14
            [ruby-core:91512]
        
      
      I also wish there was some meaningful implementation of proc.source.hash that was reasonably consistent across invocations of Ruby. Even if it was just best effort.
        
           Updated by ioquatix (Samuel Williams) over 6 years ago
          
          
        
        
          
            Actions
          
          #15
            [ruby-core:91514]
          Updated by ioquatix (Samuel Williams) over 6 years ago
          
          
        
        
          
            Actions
          
          #15
            [ruby-core:91514]
        
      
      I was playing around with this idea trying to make an implementation of class Source.
Is the source file cached in Ruby? Or should we use File.read to load it into memory?
It seems inefficient for large files, to find line/column offset. It would be nice to have absolute offset to seek to.
Maybe it's possible for source_location to append one more thing - the actual source code - if possible. This would be useful for situations like eval, where you might define something for a path that doesn't actually exist, but the source code is still available.
        
           Updated by duerst (Martin Dürst) over 6 years ago
          
          
        
        
          
            Actions
          
          #16
            [ruby-core:91518]
          Updated by duerst (Martin Dürst) over 6 years ago
          
          
        
        
          
            Actions
          
          #16
            [ruby-core:91518]
        
      
      ioquatix (Samuel Williams) wrote:
I also wish there was some meaningful implementation of
proc.source.hashthat was reasonably consistent across invocations of Ruby. Even if it was just best effort.
Please make that a separate feature if you are serious about it.
        
           Updated by Eregon (Benoit Daloze) about 1 year ago
          
          
        
        
          
            Actions
          
          #17
          Updated by Eregon (Benoit Daloze) about 1 year ago
          
          
        
        
          
            Actions
          
          #17
        
      
      - Related to Feature #17930: Add column information into error backtrace added
        
           Updated by Eregon (Benoit Daloze) about 1 year ago
          
          
        
        
          
            Actions
          
          #18
            [ruby-core:119320]
          Updated by Eregon (Benoit Daloze) about 1 year ago
          
          
        
        
          
            Actions
          
          #18
            [ruby-core:119320]
        
      
      It seems good to revisit this, the workarounds are pretty messy and CRuby-specific, e.g. https://github.com/rails/rails/pull/53055/files
I think @mame (Yusuke Endoh) 's suggestion in https://bugs.ruby-lang.org/issues/6012#note-12 is fine although I think it would be convenient to also expose the byte offsets (start and end):
.source_location #=> [path, start_line, start_column, start_offset, end_line, end_column, end_offset]
Then it would be really easy to "get the source code of a Proc".
And of course we should do the same for Method/UnboundMethod (#8751).
        
           Updated by Eregon (Benoit Daloze) about 1 year ago
          
          · Edited
        
        
          
            Actions
          
          #19
            [ruby-core:119322]
          Updated by Eregon (Benoit Daloze) about 1 year ago
          
          · Edited
        
        
          
            Actions
          
          #19
            [ruby-core:119322]
        
      
      I also really like @ioquatix (Samuel Williams) 's suggestion in https://bugs.ruby-lang.org/issues/6012#note-13 and it is a lot more flexible and more efficient too (since computing e.g. column information if unused is not cheap).
For instance method.source.code is great because it completely hides the details how to get the source code and slice it.
TruffleRuby currently keeps the source code in memory and so could provide this automatically without needing to reread the file from disk.
CRuby keeps it but only if RubyVM.keep_script_lines = true, so then could use that if available and automatically fallback to read the file from disk (great, because we should avoid users/gems referring to RubyVM in their code).
One question is where would we place/how would we name this class?
We could reuse Thread::Backtrace::Location as it's quite similar and already has path, lineno.
But it's not really related to a backtrace here.
Still it seems quite a good fit, and I don't have much idea where to place it otherwise (top-level Source seems way too prone for conflicts).
I think in term of the interface we should have:
- start_line
- start_column
- start_offset
- end_line
- end_column
- end_offset
- code: gets the source of the Proc/Method/UnboundMethod
Related and similar ideas in #18231.
        
           Updated by Eregon (Benoit Daloze) about 1 year ago
          
          
        
        
          
            Actions
          
          #20
          Updated by Eregon (Benoit Daloze) about 1 year ago
          
          
        
        
          
            Actions
          
          #20
        
      
      - Related to Feature #18231: `RubyVM.keep_script_lines` added
        
           Updated by Eregon (Benoit Daloze) 10 months ago
          
          
        
        
          
            Actions
          
          #21
            [ruby-core:120485]
          Updated by Eregon (Benoit Daloze) 10 months ago
          
          
        
        
          
            Actions
          
          #21
            [ruby-core:120485]
        
      
      One idea for this new SourceLocation/CodeLocation class would be under the new Ruby module, see https://bugs.ruby-lang.org/issues/20884#note-3
        
           Updated by bkuhlmann (Brooke Kuhlmann) 10 months ago
          
          
        
        
          
            Actions
          
          #22
            [ruby-core:120489]
          Updated by bkuhlmann (Brooke Kuhlmann) 10 months ago
          
          
        
        
          
            Actions
          
          #22
            [ruby-core:120489]
        
      
      💡 Please see Feature 21005 as an evolution of this discussion based on feedback I've had with Kevin Newton and Benoit Daloze. Thanks.
        
           Updated by Eregon (Benoit Daloze) 10 months ago
          
          
        
        
          
            Actions
          
          #23
          Updated by Eregon (Benoit Daloze) 10 months ago
          
          
        
        
          
            Actions
          
          #23
        
      
      - Related to Feature #21005: Update the source location method to include line start/stop and column start/stop details added
        
           Updated by nobu (Nobuyoshi Nakada) 10 months ago
          
          
        
        
          
            Actions
          
          #24
            [ruby-core:120566]
          Updated by nobu (Nobuyoshi Nakada) 10 months ago
          
          
        
        
          
            Actions
          
          #24
            [ruby-core:120566]
        
      
      I'm not sure about attributes and Binding#source_location.
        
           Updated by mame (Yusuke Endoh) 10 months ago
          
          
        
        
          
            Actions
          
          #25
            [ruby-core:120572]
          Updated by mame (Yusuke Endoh) 10 months ago
          
          
        
        
          
            Actions
          
          #25
            [ruby-core:120572]
        
      
      This was discussed at the October 2024 dev meeting and @matz (Yukihiro Matsumoto) approved source_location to return [path, start_line, start_column, end_line, end_column].
Matz was negative neither on adding a new class like Ruby::CodeLocation for this purpose, or adding offset information.
We briefly re-asked matz about this at today's dev meeting, and he had not changed his mind.
        
           Updated by nobu (Nobuyoshi Nakada) 10 months ago
          
          
        
        
          
            Actions
          
          #26
          Updated by nobu (Nobuyoshi Nakada) 10 months ago
          
          
        
        
          
            Actions
          
          #26
        
      
      - Status changed from Assigned to Closed
Applied in changeset git|073c4e1cc712064e626914fa4a5a8061f903a637.
[Feature #6012] Extend source_location for end position and columns
        
           Updated by Eregon (Benoit Daloze) 10 months ago
          
          
        
        
          
            Actions
          
          #27
            [ruby-core:120576]
          Updated by Eregon (Benoit Daloze) 10 months ago
          
          
        
        
          
            Actions
          
          #27
            [ruby-core:120576]
        
      
      Great to see this was accepted and even already implemented! (thanks @nobu (Nobuyoshi Nakada)!)
Byte offsets would be mostly for performance ("character" columns in multibyte files can be expensive to compute) and convenience (e.g. read the code at the given source_location from the file).
They are not strictly necessary.
        
           Updated by Dan0042 (Daniel DeLorme) 10 months ago
          
          
        
        
          
            Actions
          
          #28
            [ruby-core:120581]
          Updated by Dan0042 (Daniel DeLorme) 10 months ago
          
          
        
        
          
            Actions
          
          #28
            [ruby-core:120581]
        
      
      mame (Yusuke Endoh) wrote in #note-25:
Matz was negative neither on adding a new class like Ruby::CodeLocation for this purpose
Any idea why? Because it seems like the "obviously correct" design so I'm curious why it was decided against.