为什么 Ruby on Linux return 对 File.writable?('/tmp/file') 但在尝试写入文件时引发 Errno::EACCES?

Why does Ruby on Linux return true to File.writable?('/tmp/file') but raises Errno::EACCES while trying to write the file?

我正在使用 GNU/Linux 系统。

首先,我有:

现在在 IRB:

┌┄┄[root::archlinux]┈[/tmp]
└──╼⮚ irb
irb(main):001:0> File.writable?('ruby.rb')
=> true
irb(main):002:0> File.stat('ruby.rb')
=> #<File::Stat dev=0x2d, ino=819138, mode=0100644, nlink=1, uid=1000, gid=1000, rdev=0x0, size=0, blksize=4096, blocks=0, atime=2019-07-14 04:44:13 +0530, mtime=2019-07-14 04:44:13 +0530, ctime=2019-07-14 04:44:13 +0530>
irb(main):003:0> File.write('ruby.rb', '#!/usr/bin/ruby -w')
Traceback (most recent call last):
        3: from /root/.irb:351:in `<main>'
        2: from (irb):3
        1: from (irb):3:in `write'
Errno::EACCES (Permission denied @ rb_sysopen - ruby.rb)
irb(main):004:0> 

除此之外,其实我正在尝试做的是写一个日志文件。我正在检查文件是否可写。如果没有,它会向用户发送通知。

我以前遇到过这样的问题,在这种情况下 File#writable?(str) return 为真(它在 /tmp/ 目录上工作)。并且只是使用 begin <...> rescue Errno::EACCES 块来解决问题。但是在目前的项目中,我不想使用救援块。

为什么 File#writable?(str) return 首先是真的?

编辑:首先应该将其移动到 unix.stackexchange。

其次我明白了

编辑 2:我在 imgur 上传了 2 张截图:

可能是以下两种情况之一,文件未打开进行写入,或者用户权限问题。

尝试使用 'w''w+'

打开文件进行写入
File.open('ruby.rb', 'w') { |file| file.write("#!/usr/bin/ruby -w") }

如果这不起作用,请尝试使用 chmod 将 ruby.rb 的文件权限更新为

chmod -R 777 <path_to_file>

好的,这不是 Ruby 特定的问题。也就是说,任何编程语言都会有这样的问题,因为它都是关于 systemd 中实现的文件系统保护。

选项 fs.protected_regular 在 Linux 内核 4.19+ to make data spoofing attacks harder.

中实现

所以,首先你需要检查:

sysctl fs.protected_regular

在你的情况下,它应该打印 1

因此,将值更改为 0:

sysctl fs.protected_regular=0

希望问题能得到解决。或者试试这个:

sysctl fs.protected_regular=0
sysctl fs.protected_fifos=0

请注意,该选项默认处于启用状态。您可能不希望告诉程序的每个用户将以上设置为 0!他们甚至可能不喜欢那样!

所以,在 Ruby:

  1. 您可以删除文件:
File.delete('file') if File.exist?('file')
# Better use the '/tmp/file' or File.join(%w(/ tmp file)) otherwise the file will get delete from the Dir.pwd

每次脚本运行时都会删除该文件。

  1. 您还可以使用 begin ... rescue 块并使用 Warning.warnKernel.warnSTDERR.puts 打印警告消息。