823
832
def test_observed_sha1_cachable(self):
824
833
state, entry = self.get_state_with_a()
825
835
atime = time.time() - 10
826
836
self.build_tree(['a'])
827
statvalue = os.lstat('a')
828
statvalue = test_dirstate._FakeStat(statvalue.st_size, atime, atime,
829
statvalue.st_dev, statvalue.st_ino, statvalue.st_mode)
837
statvalue = test_dirstate._FakeStat.from_stat(os.lstat('a'))
838
statvalue.st_mtime = statvalue.st_ctime = atime
839
self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
840
state._dirblock_state)
830
841
state._observed_sha1(entry, "foo", statvalue)
831
842
self.assertEqual('foo', entry[1][0][1])
832
843
packed_stat = dirstate.pack_stat(statvalue)
833
844
self.assertEqual(packed_stat, entry[1][0][4])
845
self.assertEqual(dirstate.DirState.IN_MEMORY_HASH_MODIFIED,
846
state._dirblock_state)
835
848
def test_observed_sha1_not_cachable(self):
836
849
state, entry = self.get_state_with_a()
837
851
oldval = entry[1][0][1]
838
852
oldstat = entry[1][0][4]
839
853
self.build_tree(['a'])
840
854
statvalue = os.lstat('a')
855
self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
856
state._dirblock_state)
841
857
state._observed_sha1(entry, "foo", statvalue)
842
858
self.assertEqual(oldval, entry[1][0][1])
843
859
self.assertEqual(oldstat, entry[1][0][4])
860
self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
861
state._dirblock_state)
845
863
def test_update_entry(self):
846
864
state, _ = self.get_state_with_a()
871
889
stat_value=stat_value)
872
890
self.assertEqual(None, link_or_sha1)
874
# The dirblock entry should not have cached the file's sha1 (too new)
892
# The dirblock entry should not have computed or cached the file's
893
# sha1, but it did update the files' st_size. However, this is not
894
# worth writing a dirstate file for, so we leave the state UNMODIFIED
875
895
self.assertEqual(('f', '', 14, False, dirstate.DirState.NULLSTAT),
877
self.assertEqual(dirstate.DirState.IN_MEMORY_MODIFIED,
897
self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
878
898
state._dirblock_state)
879
899
mode = stat_value.st_mode
880
900
self.assertEqual([('is_exec', mode, False)], state._log)
956
977
# Dirblock is not updated (the link is too new)
957
978
self.assertEqual([('l', '', 6, False, dirstate.DirState.NULLSTAT)],
959
self.assertEqual(dirstate.DirState.IN_MEMORY_MODIFIED,
980
# The file entry turned into a symlink, that is considered
981
# HASH modified worthy.
982
self.assertEqual(dirstate.DirState.IN_MEMORY_HASH_MODIFIED,
960
983
state._dirblock_state)
962
985
# Because the stat_value looks new, we should re-read the target
963
987
link_or_sha1 = self.update_entry(state, entry, abspath='a',
964
988
stat_value=stat_value)
965
989
self.assertEqual('target', link_or_sha1)
966
self.assertEqual([('read_link', 'a', ''),
967
('read_link', 'a', ''),
990
self.assertEqual([('read_link', 'a', '')], state._log)
969
991
self.assertEqual([('l', '', 6, False, dirstate.DirState.NULLSTAT)],
971
994
state.adjust_time(+20) # Skip into the future, all files look old
972
996
link_or_sha1 = self.update_entry(state, entry, abspath='a',
973
997
stat_value=stat_value)
998
# The symlink stayed a symlink. So while it is new enough to cache, we
999
# don't bother setting the flag, because it is not really worth saving
1000
# (when we stat the symlink, we'll have paged in the target.)
1001
self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
1002
state._dirblock_state)
974
1003
self.assertEqual('target', link_or_sha1)
975
1004
# We need to re-read the link because only now can we cache it
976
self.assertEqual([('read_link', 'a', ''),
977
('read_link', 'a', ''),
978
('read_link', 'a', ''),
1005
self.assertEqual([('read_link', 'a', '')], state._log)
980
1006
self.assertEqual([('l', 'target', 6, False, packed_stat)],
983
1010
# Another call won't re-read the link
984
self.assertEqual([('read_link', 'a', ''),
985
('read_link', 'a', ''),
986
('read_link', 'a', ''),
1011
self.assertEqual([], state._log)
988
1012
link_or_sha1 = self.update_entry(state, entry, abspath='a',
989
1013
stat_value=stat_value)
990
1014
self.assertEqual('target', link_or_sha1)
1005
1029
self.build_tree(['a/'])
1006
1030
state.adjust_time(+20)
1007
1031
self.assertIs(None, self.do_update_entry(state, entry, 'a'))
1032
# a/ used to be a file, but is now a directory, worth saving
1008
1033
self.assertEqual(dirstate.DirState.IN_MEMORY_MODIFIED,
1009
1034
state._dirblock_state)
1011
1036
self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
1012
1037
state._dirblock_state)
1013
self.assertIs(None, self.do_update_entry(state, entry, 'a'))
1038
# No changes to a/ means not worth saving.
1039
self.assertIs(None, self.do_update_entry(state, entry, 'a'))
1040
self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
1041
state._dirblock_state)
1042
# Change the last-modified time for the directory
1043
t = time.time() - 100.0
1045
os.utime('a', (t, t))
1047
# It looks like Win32 + FAT doesn't allow to change times on a dir.
1048
raise tests.TestSkipped("can't update mtime of a dir on FAT")
1049
saved_packed_stat = entry[1][0][-1]
1050
self.assertIs(None, self.do_update_entry(state, entry, 'a'))
1051
# We *do* go ahead and update the information in the dirblocks, but we
1052
# don't bother setting IN_MEMORY_MODIFIED because it is trivial to
1054
self.assertNotEqual(saved_packed_stat, entry[1][0][-1])
1014
1055
self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
1015
1056
state._dirblock_state)
1302
1343
state._sha1_provider = UppercaseSHA1Provider()
1303
1344
self.assertChangedFileIds(['file-id'], tree)
1347
class TestPackStat(tests.TestCase):
1348
"""Check packed representaton of stat values is robust on all inputs"""
1350
scenarios = helper_scenarios
1352
def pack(self, statlike_tuple):
1353
return self.helpers.pack_stat(os.stat_result(statlike_tuple))
1356
def unpack_field(packed_string, stat_field):
1357
return _dirstate_helpers_py._unpack_stat(packed_string)[stat_field]
1359
def test_result(self):
1360
self.assertEqual("AAAQAAAAABAAAAARAAAAAgAAAAEAAIHk",
1361
self.pack((33252, 1, 2, 0, 0, 0, 4096, 15.5, 16.5, 17.5)))
1363
def test_giant_inode(self):
1364
packed = self.pack((33252, 0xF80000ABC, 0, 0, 0, 0, 0, 0, 0, 0))
1365
self.assertEqual(0x80000ABC, self.unpack_field(packed, "st_ino"))
1367
def test_giant_size(self):
1368
packed = self.pack((33252, 0, 0, 0, 0, 0, (1 << 33) + 4096, 0, 0, 0))
1369
self.assertEqual(4096, self.unpack_field(packed, "st_size"))
1371
def test_fractional_mtime(self):
1372
packed = self.pack((33252, 0, 0, 0, 0, 0, 0, 0, 16.9375, 0))
1373
self.assertEqual(16, self.unpack_field(packed, "st_mtime"))
1375
def test_ancient_mtime(self):
1376
packed = self.pack((33252, 0, 0, 0, 0, 0, 0, 0, -11644473600.0, 0))
1377
self.assertEqual(1240428288, self.unpack_field(packed, "st_mtime"))
1379
def test_distant_mtime(self):
1380
packed = self.pack((33252, 0, 0, 0, 0, 0, 0, 0, 64060588800.0, 0))
1381
self.assertEqual(3931046656, self.unpack_field(packed, "st_mtime"))
1383
def test_fractional_ctime(self):
1384
packed = self.pack((33252, 0, 0, 0, 0, 0, 0, 0, 0, 17.5625))
1385
self.assertEqual(17, self.unpack_field(packed, "st_ctime"))
1387
def test_ancient_ctime(self):
1388
packed = self.pack((33252, 0, 0, 0, 0, 0, 0, 0, 0, -11644473600.0))
1389
self.assertEqual(1240428288, self.unpack_field(packed, "st_ctime"))
1391
def test_distant_ctime(self):
1392
packed = self.pack((33252, 0, 0, 0, 0, 0, 0, 0, 0, 64060588800.0))
1393
self.assertEqual(3931046656, self.unpack_field(packed, "st_ctime"))
1395
def test_negative_dev(self):
1396
packed = self.pack((33252, 0, -0xFFFFFCDE, 0, 0, 0, 0, 0, 0, 0))
1397
self.assertEqual(0x322, self.unpack_field(packed, "st_dev"))