[mwm set] Handle file system exceptions when building feature offsets index.

This commit is contained in:
vng 2015-09-25 16:22:48 +03:00
parent c56f4c20b9
commit 25aaa05cef
11 changed files with 82 additions and 16 deletions

View file

@ -71,7 +71,7 @@ namespace feature
{
LOG(LINFO, ("Creating features offset table file", storePath));
VERIFY(CountryIndexes::PreparePlaceOnDisk(localFile), ());
CountryIndexes::PreparePlaceOnDisk(localFile);
Builder builder;
FeaturesVector::ForEachOffset(cont.GetReader(DATA_FILE_TAG), [&builder] (uint32_t offset)

View file

@ -97,7 +97,7 @@ namespace feature
FilesContainerR baseContainer(pl.GetReader("minsk-pass" DATA_FILE_EXTENSION));
LocalCountryFile localFile = LocalCountryFile::MakeForTesting(testFileName);
TEST(CountryIndexes::PreparePlaceOnDisk(localFile), ());
CountryIndexes::PreparePlaceOnDisk(localFile);
string const indexFile = CountryIndexes::GetPath(localFile, CountryIndexes::Index::Offsets);
FileWriter::DeleteFileX(indexFile);

View file

@ -4,6 +4,7 @@
#include "defines.hpp"
#include "base/assert.hpp"
#include "base/exception.hpp"
#include "base/logging.hpp"
#include "base/stl_add.hpp"
@ -214,7 +215,18 @@ unique_ptr<MwmSet::MwmValueBase> MwmSet::LockValueImpl(MwmId const & id)
}
}
return CreateValue(*info);
try
{
return CreateValue(*info);
}
catch (exception const & ex)
{
LOG(LERROR, ("Can't create MWMValue for", info->GetCountryName(), "Reason", ex.what()));
--info->m_numRefs;
DeregisterImpl(id);
return nullptr;
}
}
void MwmSet::UnlockValue(MwmId const & id, unique_ptr<MwmValueBase> && p)

View file

@ -36,6 +36,7 @@ SOURCES += \
kmz_unarchive_test.cpp \
mwm_url_tests.cpp \
navigator_test.cpp \
mwm_set_test.cpp \
!linux* {
SOURCES += working_time_tests.cpp \

View file

@ -0,0 +1,49 @@
#include "testing/testing.hpp"
#include "indexer/index.hpp"
#include "platform/local_country_file_utils.hpp"
#include "platform/platform.hpp"
#ifndef OMIM_OS_WINDOWS
#include <sys/stat.h>
#endif
using namespace platform;
using namespace my;
#ifndef OMIM_OS_WINDOWS
UNIT_TEST(MwmSet_FileSystemErrors)
{
string const dir = GetPlatform().WritableDir();
CountryFile file("minsk-pass");
LocalCountryFile localFile(dir, file, 0);
TEST(CountryIndexes::DeleteFromDisk(localFile), ());
LogLevel oldLevel = g_LogAbortLevel;
g_LogAbortLevel = LCRITICAL;
// Remove writable permission.
int const readOnlyMode = S_IRUSR | S_IRGRP | S_IROTH | S_IXUSR | S_IXGRP | S_IXOTH;
TEST_EQUAL(chmod(dir.c_str(), readOnlyMode), 0, ());
Index index;
auto p = index.RegisterMap(localFile);
TEST_EQUAL(p.second, Index::RegResult::Success, ());
TEST(index.GetMwmIdByCountryFile(file) != Index::MwmId(), ());
TEST(!index.GetMwmHandleById(p.first).IsAlive(), ());
vector<shared_ptr<MwmInfo>> infos;
index.GetMwmsInfo(infos);
TEST(infos.empty(), ());
// Restore writable permission.
TEST_EQUAL(chmod(dir.c_str(), readOnlyMode | S_IWUSR), 0, ());
g_LogAbortLevel = oldLevel;
}
#endif

View file

@ -239,9 +239,11 @@ ModelReader * GetCountryReader(platform::LocalCountryFile const & file, MapOptio
}
// static
bool CountryIndexes::PreparePlaceOnDisk(LocalCountryFile const & localFile)
void CountryIndexes::PreparePlaceOnDisk(LocalCountryFile const & localFile)
{
return MkDirChecked(IndexesDir(localFile));
string const dir = IndexesDir(localFile);
if (!MkDirChecked(dir))
MYTHROW(FileSystemException, ("Can't create directory", dir));
}
// static
@ -302,6 +304,8 @@ string CountryIndexes::IndexesDir(LocalCountryFile const & localFile)
string dir = localFile.GetDirectory();
CountryFile const & file = localFile.GetCountryFile();
/// @todo It's a temporary code until we will put fIndex->fOffset into mwm container.
/// IndexesDir should not throw any exceptions.
if (dir.empty())
{
// Local file is stored in resources. Need to prepare index folder in the writable directory.
@ -309,7 +313,8 @@ string CountryIndexes::IndexesDir(LocalCountryFile const & localFile)
ASSERT_GREATER(version, 0, ());
dir = my::JoinFoldersToPath(GetPlatform().WritableDir(), strings::to_string(version));
VERIFY(MkDirChecked(dir), ());
if (!MkDirChecked(dir))
MYTHROW(FileSystemException, ("Can't create directory", dir));
}
return my::JoinFoldersToPath(dir, file.GetNameWithoutExt());

View file

@ -62,10 +62,10 @@ public:
Offsets
};
// Prepares (if necessary) directory for country indexes. Local file
// should point to existing local country files. Returns true on
// success, false otherwise.
static bool PreparePlaceOnDisk(LocalCountryFile const & localFile);
/// Prepares (if necessary) directory for country indexes. Local file
/// should point to existing local country files.
/// @throw FileSystemException if any file system error occured.
static void PreparePlaceOnDisk(LocalCountryFile const & localFile);
// Removes country indexes from disk including containing directory.
static bool DeleteFromDisk(LocalCountryFile const & localFile);

View file

@ -15,7 +15,7 @@
#include "defines.hpp"
DECLARE_EXCEPTION(FileAbsentException, RootException);
DECLARE_EXCEPTION(NotImplementedException, RootException);
DECLARE_EXCEPTION(FileSystemException, RootException);
namespace platform
{

View file

@ -313,8 +313,7 @@ UNIT_TEST(LocalCountryFile_CountryIndexes)
TEST_EQUAL(
my::JoinFoldersToPath(germanyLocalFile.GetDirectory(), germanyFile.GetNameWithoutExt()),
CountryIndexes::IndexesDir(germanyLocalFile), ());
TEST(CountryIndexes::PreparePlaceOnDisk(germanyLocalFile),
("Can't prepare place for:", germanyLocalFile));
CountryIndexes::PreparePlaceOnDisk(germanyLocalFile);
string const bitsPath = CountryIndexes::GetPath(germanyLocalFile, CountryIndexes::Index::Bits);
TEST(!Platform::IsFileExistsByFullPath(bitsPath), (bitsPath));
@ -344,9 +343,8 @@ UNIT_TEST(LocalCountryFile_DoNotDeleteUserFiles)
CountryFile germanyFile("Germany");
LocalCountryFile germanyLocalFile(testDir.GetFullPath(), germanyFile, 101010 /* version */);
CountryIndexes::PreparePlaceOnDisk(germanyLocalFile);
TEST(CountryIndexes::PreparePlaceOnDisk(germanyLocalFile),
("Can't prepare place for:", germanyLocalFile));
string const userFilePath =
my::JoinFoldersToPath(CountryIndexes::IndexesDir(germanyLocalFile), "user-data.txt");
{

View file

@ -388,6 +388,7 @@ void OsrmFtSegBackwardIndex::Construct(OsrmFtSegMapping & mapping, uint32_t maxN
if (m_oldFormat)
LOG(LINFO, ("Using old format index for", localFile.GetCountryName()));
CountryIndexes::PreparePlaceOnDisk(localFile);
string const bitsFileName = CountryIndexes::GetPath(localFile, CountryIndexes::Index::Bits);
string const nodesFileName = CountryIndexes::GetPath(localFile, CountryIndexes::Index::Nodes);

View file

@ -598,7 +598,7 @@ UNIT_TEST(StorageTest_DeleteCountry)
LocalCountryFile file = LocalCountryFile::MakeForTesting("Wonderland");
TEST_EQUAL(MapOptions::MapWithCarRouting, file.GetFiles(), ());
TEST(CountryIndexes::PreparePlaceOnDisk(file), ());
CountryIndexes::PreparePlaceOnDisk(file);
string const bitsPath = CountryIndexes::GetPath(file, CountryIndexes::Index::Bits);
{
FileWriter writer(bitsPath);