Skip to content

Commit

Permalink
Fix change detection when source timestamps differ
Browse files Browse the repository at this point in the history
This is a follow-up fix to the fix for #25. False positives could
be caused when two data sources (eg. plugins folder and active
plugins file) have different timestamps before data is loaded, or
when a file is changed but no plugins folder changes are made.
  • Loading branch information
Ortham committed Nov 25, 2015
1 parent 7482090 commit 7f5f058
Show file tree
Hide file tree
Showing 3 changed files with 50 additions and 12 deletions.
24 changes: 13 additions & 11 deletions src/backend/LoadOrder.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ namespace liblo {
});
}
partitionMasters();
mtime = boost::filesystem::last_write_time(gameSettings.getPluginsFolder());
pluginsFolderModTime = boost::filesystem::last_write_time(gameSettings.getPluginsFolder());
}
if (fs::exists(gameSettings.getActivePluginsFile()))
loadActivePlugins();
Expand Down Expand Up @@ -270,20 +270,20 @@ namespace liblo {
try {
// Has the plugins folder been modified since the load order
// was last read or saved?
if (fs::last_write_time(gameSettings.getPluginsFolder()) != mtime)
if (fs::last_write_time(gameSettings.getPluginsFolder()) != pluginsFolderModTime)
return true;

// Has the active plugins file been changed since it was last
// read or saved?
if (fs::exists(gameSettings.getActivePluginsFile())
&& fs::last_write_time(gameSettings.getActivePluginsFile()) != mtime)
&& fs::last_write_time(gameSettings.getActivePluginsFile()) != activePluginsFileModTime)
return true;

// Has the full textfile load order been changed since it was
// last read or saved?
if (gameSettings.getLoadOrderMethod() == LIBLO_METHOD_TEXTFILE
&& fs::exists(gameSettings.getLoadOrderFile())
&& fs::last_write_time(gameSettings.getLoadOrderFile()) != mtime)
&& fs::last_write_time(gameSettings.getLoadOrderFile()) != loadOrderFileModTime)
return true;

// Plugins folder modification time not changed if a file in it is
Expand Down Expand Up @@ -324,7 +324,6 @@ namespace liblo {
}),
end(loadOrderFileLoadOrder));

//Compare the two LoadOrder objects: they should be identical (since mtimes for each have not been touched).
return PluginsFileLO.getLoadOrder() == loadOrderFileLoadOrder;
}

Expand Down Expand Up @@ -357,6 +356,11 @@ namespace liblo {
it = addToLoadOrder(line);
}
}

if (file == gameSettings.getActivePluginsFile())
activePluginsFileModTime = boost::filesystem::last_write_time(gameSettings.getActivePluginsFile());
else if (file == gameSettings.getLoadOrderFile()
loadOrderFileModTime = boost::filesystem::last_write_time(gameSettings.getLoadOrderFile());
}
catch (std::ifstream::failure& e) {
throw error(LIBLO_ERROR_FILE_READ_FAIL, "\"" + file.string() + "\" could not be read. Details: " + e.what());
Expand Down Expand Up @@ -408,6 +412,8 @@ namespace liblo {
it->activate(gameSettings.getPluginsFolder());
}
}

activePluginsFileModTime = boost::filesystem::last_write_time(gameSettings.getActivePluginsFile());
}
catch (std::exception& e) {
throw error(LIBLO_ERROR_FILE_READ_FAIL, "\"" + gameSettings.getActivePluginsFile().string() + "\" could not be read. Details: " + e.what());
Expand Down Expand Up @@ -465,8 +471,6 @@ namespace liblo {
loadOrder.at(i).setModTime(timestamp, gameSettings.getPluginsFolder());
++i;
}
//Now record new plugins folder mtime.
mtime = fs::last_write_time(gameSettings.getPluginsFolder());
}

void LoadOrder::saveTextfileLoadOrder() {
Expand All @@ -482,9 +486,7 @@ namespace liblo {
outfile << plugin.getName() << endl;
outfile.close();

//Now record new loadorder.txt mtime.
//Plugins.txt doesn't need its mtime updated as only the order of its contents has changed, and it is stored in memory as an unordered set.
mtime = fs::last_write_time(gameSettings.getLoadOrderFile());
loadOrderFileModTime = fs::last_write_time(gameSettings.getLoadOrderFile());
}
catch (std::ios_base::failure& e) {
throw error(LIBLO_ERROR_FILE_WRITE_FAIL, "\"" + gameSettings.getLoadOrderFile().string() + "\" cannot be written to. Details: " + e.what());
Expand Down Expand Up @@ -539,7 +541,7 @@ namespace liblo {
throw error(LIBLO_ERROR_FILE_WRITE_FAIL, "\"" + gameSettings.getActivePluginsFile().string() + "\" could not be written. Details: " + e.what());
}

mtime = fs::last_write_time(gameSettings.getActivePluginsFile());
activePluginsFileModTime = fs::last_write_time(gameSettings.getActivePluginsFile());

if (!badFilename.empty())
throw error(LIBLO_WARN_BAD_FILENAME, badFilename);
Expand Down
4 changes: 3 additions & 1 deletion src/backend/LoadOrder.h
Original file line number Diff line number Diff line change
Expand Up @@ -65,7 +65,9 @@ namespace liblo {

void clear();
private:
time_t mtime;
time_t pluginsFolderModTime;
time_t activePluginsFileModTime;
time_t loadOrderFileModTime;
std::vector<Plugin> loadOrder;
const GameSettings& gameSettings;

Expand Down
34 changes: 34 additions & 0 deletions src/tests/backend/LoadOrderTest.h
Original file line number Diff line number Diff line change
Expand Up @@ -1363,5 +1363,39 @@ namespace liblo {

EXPECT_TRUE(loadOrder.hasFilesystemChanged());
}

TEST_P(LoadOrderTest, shouldDetectFilesystemChangesIfNoChangesAreMadeButActivePluginsFileAndPluginsFolderTimestampsAreDifferent) {

time_t currentModTime = boost::filesystem::last_write_time(gameSettings.getPluginsFolder());
EXPECT_NO_THROW(boost::filesystem::last_write_time(gameSettings.getActivePluginsFile(), currentModTime + 2));

ASSERT_NO_THROW(loadOrder.load());

EXPECT_FALSE(loadOrder.hasFilesystemChanged());
}

TEST_P(LoadOrderTest, shouldDetectFilesystemChangesIfNoChangesAreMadeButLoadOrderFileAndPluginsFolderTimestampsAreDifferent) {
if (gameSettings.getLoadOrderMethod() != LIBLO_METHOD_TEXTFILE)
return;

time_t currentModTime = boost::filesystem::last_write_time(gameSettings.getPluginsFolder());
EXPECT_NO_THROW(boost::filesystem::last_write_time(gameSettings.getLoadOrderFile(), currentModTime + 2));

ASSERT_NO_THROW(loadOrder.load());

EXPECT_FALSE(loadOrder.hasFilesystemChanged());
}

TEST_P(LoadOrderTest, shouldDetectFilesystemChangesIfNoChangesAreMadeButLoadOrderAndActivePluginsFilesAreDifferent) {
if (gameSettings.getLoadOrderMethod() != LIBLO_METHOD_TEXTFILE)
return;

time_t currentModTime = boost::filesystem::last_write_time(gameSettings.getActivePluginsFile());
EXPECT_NO_THROW(boost::filesystem::last_write_time(gameSettings.getLoadOrderFile(), currentModTime + 2));

ASSERT_NO_THROW(loadOrder.load());

EXPECT_FALSE(loadOrder.hasFilesystemChanged());
}
}
}

0 comments on commit 7f5f058

Please sign in to comment.