What does __FILE__ mean in Ruby?
Asked Answered
U

4

230

I see this all the time in Ruby:

require File.dirname(__FILE__) + "/../../config/environment"  

What does __FILE__ mean?

Unsettled answered 22/10, 2008 at 3:28 Comment(1)
To help tie together references, also see ruby-forum.com/topic/143383 and #4326259Fivefold
N
157

It is a reference to the current file name. In the file foo.rb, __FILE__ would be interpreted as "foo.rb".

Edit: Ruby 1.9.2 and 1.9.3 appear to behave a little differently from what Luke Bayes said in his comment. With these files:

# test.rb
puts __FILE__
require './dir2/test.rb'
# dir2/test.rb
puts __FILE__

Running ruby test.rb will output

test.rb
/full/path/to/dir2/test.rb
Nita answered 22/10, 2008 at 3:30 Comment(8)
This answer is not accurate. FILE is the "relative" path to the file from the current execution directory - not absolute. You must use File.expand_path(FILE) to get the absolute pathEnshrine
Double underscores were automatically removed within the comment above.Enshrine
is this still true in ruby 1.9.2?Cricket
@Cricket It works kinda different in 1.9.2. The file initially loaded has a relative __FILE__, but other included/required files have absolute paths.Nita
@LukeBayes You can preserve underscores and other special characters in Stack Overflow comments by surrounding the literal text with backticks. Write `__FILE__` to get __FILE__.Aristotelianism
@RoryO'Kane Before someone misreads this and tries to enter this into a ruby script, do not write `__FILE__` in ruby, unless you want the world to maybe explode.Selfrespect
I've found references to the docs, but they're not too specific.Sasnett
__FILE__ is relative only for the first (main) file. Because in this case load_file() is called directly and passed $0. When requiring files require_internal() uses search_required() to find the file and passes to load_file() the resulting absolute path. The behavior is handy to make $0 == __FILE__ kind of checks (self-executing files).Sasnett
E
64

The value of __FILE__ is a relative path that is created and stored (but never updated) when your file is loaded. This means that if you have any calls to Dir.chdir anywhere else in your application, this path will expand incorrectly.

puts __FILE__
Dir.chdir '../../'
puts __FILE__

One workaround to this problem is to store the expanded value of __FILE__ outside of any application code. As long as your require statements are at the top of your definitions (or at least before any calls to Dir.chdir), this value will continue to be useful after changing directories.

$MY_FILE_PATH = File.expand_path(File.dirname(__FILE__))

# open class and do some stuff that changes directory

puts $MY_FILE_PATH
Enshrine answered 21/5, 2009 at 17:17 Comment(1)
This means that if you have any calls to Dir.chdir anywhere else in your application, this path will expand incorrectly. The path expanded correctly under my test. My ruby version is 2.3.7, maybe the newer ruby version fixed this issue.Spate
C
29

__FILE__ is the filename with extension of the file containing the code being executed.

In foo.rb, __FILE__ would be "foo.rb".

If foo.rb were in the dir /home/josh then File.dirname(__FILE__) would return /home/josh.

Coquito answered 2/12, 2008 at 19:47 Comment(2)
Exactly looking for this. +1 :=) @CoquitoVamp
A link to the documentation hereBi
A
18

In Ruby, the Windows version anyways, I just checked and __FILE__ does not contain the full path to the file. Instead it contains the path to the file relative to where it's being executed from.

In PHP __FILE__ is the full path (which in my opinion is preferable). This is why, in order to make your paths portable in Ruby, you really need to use this:

File.expand_path(File.dirname(__FILE__) + "relative/path/to/file")

I should note that in Ruby 1.9.1 __FILE__ contains the full path to the file, the above description was for when I used Ruby 1.8.7.

In order to be compatible with both Ruby 1.8.7 and 1.9.1 (not sure about 1.9) you should require files by using the construct I showed above.

Angele answered 24/4, 2009 at 6:59 Comment(4)
You need to prepend a slash to your relative path: File.expand_path(File.dirname(__FILE__) + "/relative/path/to/file")Heterography
You should never hard-code the directory separator. Use File.join instead: File.expand_path( File.join( File.dirname(__FILE__), "relative", "path", "to", "file") )Bricabrac
Andrew, can you helpe-me please? I am writing to a file using the following: path = YAML::load_file(File.open(File.join(File.dirname(FILE) + "/massa/users.yml"))) path[:cad_user][:user] = input_email_create.value File.open(File.join(File.dirname(FILE) + '/massa/users.yml'), 'w') {|f| f.write path.to_yaml } but I get the message: No such file or directory @ rb_sysopenInverson
Not sure how it behaved back then, but these days __FILE__ == $0 for the first (main) file, and an absolute path for the rest. This is handy for __FILE__ == $0 kind of checks (self-executing files). More about it in my other comment.Sasnett

© 2022 - 2024 — McMap. All rights reserved.