Fixes building anbox with clang 4.0 -- removes unused declarations, makes sure a template is visible at first instantiation, makes sure exception declarations match between headers and implementations Signed-off-by: Bernhard Rosenkränzer <bero@lindev.ch>
202 lines
4.9 KiB
C++
202 lines
4.9 KiB
C++
// Copyright (C) 2015 Thomas Voß <thomas.voss.bochum@gmail.com>
|
|
//
|
|
// This library is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published
|
|
// by the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#include <xdg.h>
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#include <cstdlib>
|
|
#include <stdexcept>
|
|
|
|
namespace fs = boost::filesystem;
|
|
|
|
namespace
|
|
{
|
|
|
|
fs::path throw_if_not_absolute(const fs::path& p)
|
|
{
|
|
if (p.has_root_directory())
|
|
return p;
|
|
|
|
throw std::runtime_error{"Directores MUST be absolute."};
|
|
}
|
|
|
|
namespace env
|
|
{
|
|
std::string get(const std::string& key, const std::string& default_value)
|
|
{
|
|
if (auto value = std::getenv(key.c_str()))
|
|
return value;
|
|
return default_value;
|
|
}
|
|
|
|
std::string get_or_throw(const std::string& key)
|
|
{
|
|
if (auto value = std::getenv(key.c_str()))
|
|
{
|
|
return value;
|
|
}
|
|
|
|
throw std::runtime_error{key + " not set in environment"};
|
|
}
|
|
|
|
constexpr const char* xdg_data_home{"XDG_DATA_HOME"};
|
|
constexpr const char* xdg_data_dirs{"XDG_DATA_DIRS"};
|
|
constexpr const char* xdg_config_home{"XDG_CONFIG_HOME"};
|
|
constexpr const char* xdg_config_dirs{"XDG_CONFIG_DIRS"};
|
|
constexpr const char* xdg_cache_home{"XDG_CACHE_HOME"};
|
|
}
|
|
|
|
namespace impl
|
|
{
|
|
class BaseDirSpecification : public xdg::BaseDirSpecification
|
|
{
|
|
public:
|
|
static const BaseDirSpecification& instance()
|
|
{
|
|
static const BaseDirSpecification spec;
|
|
return spec;
|
|
}
|
|
|
|
BaseDirSpecification()
|
|
{
|
|
}
|
|
|
|
const xdg::Data& data() const override
|
|
{
|
|
return data_;
|
|
}
|
|
|
|
const xdg::Config& config() const override
|
|
{
|
|
return config_;
|
|
}
|
|
|
|
const xdg::Cache& cache() const override
|
|
{
|
|
return cache_;
|
|
}
|
|
|
|
const xdg::Runtime& runtime() const override
|
|
{
|
|
return runtime_;
|
|
}
|
|
|
|
private:
|
|
xdg::Data data_;
|
|
xdg::Config config_;
|
|
xdg::Cache cache_;
|
|
xdg::Runtime runtime_;
|
|
};
|
|
}
|
|
}
|
|
|
|
fs::path xdg::Data::home() const
|
|
{
|
|
auto v = env::get(env::xdg_data_home, "");
|
|
if (v.empty())
|
|
return throw_if_not_absolute(fs::path{env::get_or_throw("HOME")} / ".local" / "share");
|
|
|
|
return throw_if_not_absolute(fs::path(v));
|
|
}
|
|
|
|
std::vector<fs::path> xdg::Data::dirs() const
|
|
{
|
|
auto v = env::get(env::xdg_data_dirs, "");
|
|
if (v.empty())
|
|
return {fs::path{"/usr/local/share"}, fs::path{"/usr/share"}};
|
|
|
|
std::vector<std::string> tokens;
|
|
tokens = boost::split(tokens, v, boost::is_any_of(":"));
|
|
std::vector<fs::path> result;
|
|
for (const auto& token : tokens)
|
|
{
|
|
result.push_back(throw_if_not_absolute(fs::path(token)));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
fs::path xdg::Config::home() const
|
|
{
|
|
auto v = env::get(env::xdg_config_home, "");
|
|
if (v.empty())
|
|
return throw_if_not_absolute(fs::path{env::get_or_throw("HOME")} / ".config");
|
|
|
|
return throw_if_not_absolute(fs::path(v));
|
|
}
|
|
|
|
std::vector<fs::path> xdg::Config::dirs() const
|
|
{
|
|
auto v = env::get(env::xdg_config_dirs, "");
|
|
if (v.empty())
|
|
return {fs::path{"/etc/xdg"}};
|
|
|
|
std::vector<std::string> tokens;
|
|
tokens = boost::split(tokens, v, boost::is_any_of(":"));
|
|
std::vector<fs::path> result;
|
|
for (const auto& token : tokens)
|
|
{
|
|
fs::path p(token);
|
|
result.push_back(throw_if_not_absolute(p));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
fs::path xdg::Cache::home() const
|
|
{
|
|
auto v = env::get(env::xdg_cache_home, "");
|
|
if (v.empty())
|
|
return throw_if_not_absolute(fs::path{env::get_or_throw("HOME")} / ".cache");
|
|
|
|
return throw_if_not_absolute(fs::path(v));
|
|
}
|
|
|
|
fs::path xdg::Runtime::dir() const
|
|
{
|
|
auto v = env::get(env::xdg_config_home, "");
|
|
if (v.empty())
|
|
{
|
|
// We do not fall back gracefully and instead throw, dispatching to calling
|
|
// code for handling the case of a safe user-specfic runtime directory missing.
|
|
throw std::runtime_error{"Runtime directory not set"};
|
|
}
|
|
|
|
return throw_if_not_absolute(fs::path(v));
|
|
}
|
|
|
|
std::shared_ptr<xdg::BaseDirSpecification> xdg::BaseDirSpecification::create()
|
|
{
|
|
return std::make_shared<impl::BaseDirSpecification>();
|
|
}
|
|
|
|
const xdg::Data& xdg::data()
|
|
{
|
|
return impl::BaseDirSpecification::instance().data();
|
|
}
|
|
|
|
const xdg::Config& xdg::config()
|
|
{
|
|
return impl::BaseDirSpecification::instance().config();
|
|
}
|
|
|
|
const xdg::Cache& xdg::cache()
|
|
{
|
|
return impl::BaseDirSpecification::instance().cache();
|
|
}
|
|
|
|
const xdg::Runtime& xdg::runtime()
|
|
{
|
|
return impl::BaseDirSpecification::instance().runtime();
|
|
}
|