2018-03-04 10:09:32 -05:00
|
|
|
require_relative '../../spec_helper'
|
2017-05-07 08:04:49 -04:00
|
|
|
|
|
|
|
describe "File.utime" do
|
2018-12-28 19:22:52 -05:00
|
|
|
|
|
|
|
before :all do
|
2020-04-02 20:47:19 -04:00
|
|
|
@time_is_float = platform_is :windows
|
2018-12-28 19:22:52 -05:00
|
|
|
end
|
|
|
|
|
2017-05-07 08:04:49 -04:00
|
|
|
before :each do
|
|
|
|
@atime = Time.now
|
|
|
|
@mtime = Time.now
|
|
|
|
@file1 = tmp("specs_file_utime1")
|
|
|
|
@file2 = tmp("specs_file_utime2")
|
|
|
|
touch @file1
|
|
|
|
touch @file2
|
|
|
|
end
|
|
|
|
|
|
|
|
after :each do
|
|
|
|
rm_r @file1, @file2
|
|
|
|
end
|
|
|
|
|
|
|
|
it "sets the access and modification time of each file" do
|
|
|
|
File.utime(@atime, @mtime, @file1, @file2)
|
2018-12-28 19:22:52 -05:00
|
|
|
if @time_is_float
|
|
|
|
File.atime(@file1).should be_close(@atime, 0.0001)
|
|
|
|
File.mtime(@file1).should be_close(@mtime, 0.0001)
|
|
|
|
File.atime(@file2).should be_close(@atime, 0.0001)
|
|
|
|
File.mtime(@file2).should be_close(@mtime, 0.0001)
|
|
|
|
else
|
2019-04-27 12:53:23 -04:00
|
|
|
File.atime(@file1).to_i.should be_close(@atime.to_i, TIME_TOLERANCE)
|
|
|
|
File.mtime(@file1).to_i.should be_close(@mtime.to_i, TIME_TOLERANCE)
|
|
|
|
File.atime(@file2).to_i.should be_close(@atime.to_i, TIME_TOLERANCE)
|
|
|
|
File.mtime(@file2).to_i.should be_close(@mtime.to_i, TIME_TOLERANCE)
|
2018-12-28 19:22:52 -05:00
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "uses the current times if two nil values are passed" do
|
2018-12-28 19:22:52 -05:00
|
|
|
tn = Time.now
|
2017-05-07 08:04:49 -04:00
|
|
|
File.utime(nil, nil, @file1, @file2)
|
2018-12-28 19:22:52 -05:00
|
|
|
if @time_is_float
|
|
|
|
File.atime(@file1).should be_close(tn, 0.050)
|
|
|
|
File.mtime(@file1).should be_close(tn, 0.050)
|
|
|
|
File.atime(@file2).should be_close(tn, 0.050)
|
|
|
|
File.mtime(@file2).should be_close(tn, 0.050)
|
|
|
|
else
|
2019-04-27 12:53:23 -04:00
|
|
|
File.atime(@file1).to_i.should be_close(Time.now.to_i, TIME_TOLERANCE)
|
|
|
|
File.mtime(@file1).to_i.should be_close(Time.now.to_i, TIME_TOLERANCE)
|
|
|
|
File.atime(@file2).to_i.should be_close(Time.now.to_i, TIME_TOLERANCE)
|
|
|
|
File.mtime(@file2).to_i.should be_close(Time.now.to_i, TIME_TOLERANCE)
|
2018-12-28 19:22:52 -05:00
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "accepts an object that has a #to_path method" do
|
|
|
|
File.utime(@atime, @mtime, mock_to_path(@file1), mock_to_path(@file2))
|
|
|
|
end
|
2017-12-01 10:41:50 -05:00
|
|
|
|
2017-12-27 11:12:47 -05:00
|
|
|
it "accepts numeric atime and mtime arguments" do
|
2018-12-28 19:22:52 -05:00
|
|
|
if @time_is_float
|
|
|
|
File.utime(@atime.to_f, @mtime.to_f, @file1, @file2)
|
|
|
|
File.atime(@file1).should be_close(@atime, 0.0001)
|
|
|
|
File.mtime(@file1).should be_close(@mtime, 0.0001)
|
|
|
|
File.atime(@file2).should be_close(@atime, 0.0001)
|
|
|
|
File.mtime(@file2).should be_close(@mtime, 0.0001)
|
|
|
|
else
|
|
|
|
File.utime(@atime.to_i, @mtime.to_i, @file1, @file2)
|
2019-04-27 12:53:23 -04:00
|
|
|
File.atime(@file1).to_i.should be_close(@atime.to_i, TIME_TOLERANCE)
|
|
|
|
File.mtime(@file1).to_i.should be_close(@mtime.to_i, TIME_TOLERANCE)
|
|
|
|
File.atime(@file2).to_i.should be_close(@atime.to_i, TIME_TOLERANCE)
|
|
|
|
File.mtime(@file2).to_i.should be_close(@mtime.to_i, TIME_TOLERANCE)
|
2018-12-28 19:22:52 -05:00
|
|
|
end
|
2017-12-27 11:12:47 -05:00
|
|
|
end
|
|
|
|
|
2021-10-20 15:57:05 -04:00
|
|
|
it "may set nanosecond precision" do
|
|
|
|
t = Time.utc(2007, 11, 1, 15, 25, 0, 123456.789r)
|
|
|
|
File.utime(t, t, @file1)
|
|
|
|
File.atime(@file1).nsec.should.between?(0, 123500000)
|
|
|
|
File.mtime(@file1).nsec.should.between?(0, 123500000)
|
2021-10-05 13:41:44 -04:00
|
|
|
end
|
|
|
|
|
2022-03-28 11:47:04 -04:00
|
|
|
it "returns the number of filenames in the arguments" do
|
|
|
|
File.utime(@atime.to_f, @mtime.to_f, @file1, @file2).should == 2
|
|
|
|
end
|
|
|
|
|
2017-12-01 10:41:50 -05:00
|
|
|
platform_is :linux do
|
|
|
|
platform_is wordsize: 64 do
|
2020-11-12 09:36:47 -05:00
|
|
|
it "allows Time instances in the far future to set mtime and atime (but some filesystems limit it up to 2446-05-10 or 2038-01-19)" do
|
2020-02-08 21:04:53 -05:00
|
|
|
# https://ext4.wiki.kernel.org/index.php/Ext4_Disk_Layout#Inode_Timestamps
|
|
|
|
# "Therefore, timestamps should not overflow until May 2446."
|
2020-11-12 09:36:47 -05:00
|
|
|
# https://lwn.net/Articles/804382/
|
|
|
|
# "On-disk timestamps hitting the y2038 limit..."
|
|
|
|
# The problem seems to be being improved, but currently it actually fails on XFS on RHEL8
|
|
|
|
# https://rubyci.org/logs/rubyci.s3.amazonaws.com/rhel8/ruby-master/log/20201112T123004Z.fail.html.gz
|
2017-12-01 10:41:50 -05:00
|
|
|
time = Time.at(1<<44)
|
|
|
|
File.utime(time, time, @file1)
|
2020-11-12 09:36:47 -05:00
|
|
|
[559444, 2446, 2038].should.include? File.atime(@file1).year
|
|
|
|
[559444, 2446, 2038].should.include? File.mtime(@file1).year
|
2017-12-01 10:41:50 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|