/*
This file is part of Ingen.
Copyright 2007-2018 David Robillard
Ingen is free software: you can redistribute it and/or modify it under the
terms of the GNU Affero General Public License as published by the Free
Software Foundation, either version 3 of the License, or any later version.
Ingen 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 Affero General Public License for details.
You should have received a copy of the GNU Affero General Public License
along with Ingen. If not, see .
*/
#include "ingen/ClashAvoider.hpp"
#include "ingen/Store.hpp"
#include "ingen/URI.hpp"
#include "ingen/paths.hpp"
#include "raul/Path.hpp"
#include "raul/Symbol.hpp"
#include
#include
#include
#include
#include
#include
#include
#include
namespace ingen {
ClashAvoider::ClashAvoider(const Store& store)
: _store(store)
{}
URI
ClashAvoider::map_uri(const URI& in)
{
if (uri_is_path(in)) {
return path_to_uri(map_path(uri_to_path(in)));
}
return in;
}
raul::Path
ClashAvoider::map_path(const raul::Path& in)
{
unsigned offset = 0;
bool has_offset = false;
const size_t pos = in.find_last_of('_');
if (pos != std::string::npos && pos != (in.length()-1)) {
const std::string trailing = in.substr(pos + 1);
char* end = nullptr;
strtoul(trailing.c_str(), &end, 10);
has_offset = (*end == '\0');
}
// Path without _n suffix
std::string base_path_str = in;
if (has_offset) {
base_path_str.resize(base_path_str.find_last_of('_'));
}
raul::Path base_path(base_path_str);
auto m = _symbol_map.find(in);
if (m != _symbol_map.end()) {
return m->second;
}
// See if parent is mapped
raul::Path parent = in.parent();
do {
auto p = _symbol_map.find(parent);
if (p != _symbol_map.end()) {
const auto mapped = raul::Path{p->second.base() +
in.substr(parent.base().length())};
auto i = _symbol_map.emplace(in, mapped);
return i.first->second;
}
parent = parent.parent();
} while (!parent.is_root());
if (!exists(in) && _symbol_map.find(in) == _symbol_map.end()) {
// No clash, use symbol unmodified
auto i = _symbol_map.emplace(in, in);
assert(i.second);
return i.first->second;
}
// Append _2 _3 etc until an unused symbol is found
while (true) {
auto o = _offsets.find(base_path);
if (o != _offsets.end()) {
offset = ++o->second;
}
if (offset == 0) {
offset = 2;
}
std::stringstream ss;
ss << base_path << "_" << offset;
if (!exists(raul::Path(ss.str()))) {
std::string name = base_path.symbol();
if (name.empty()) {
name = "_";
}
const raul::Symbol sym{name};
const std::string str{ss.str()};
auto i = _symbol_map.emplace(in, raul::Path(str));
offset = _store.child_name_offset(in.parent(), sym, false);
_offsets.emplace(base_path, offset);
return i.first->second;
}
if (o != _offsets.end()) {
offset = ++o->second;
} else {
++offset;
}
}
}
bool
ClashAvoider::exists(const raul::Path& path) const
{
return _store.find(path) != _store.end();
}
static std::optional
numeric_suffix_start(const std::string& str)
{
if (!isdigit(str[str.length() - 1])) {
return {};
}
size_t i = str.length() - 1;
while (i > 0 && isdigit(str[i - 1])) {
--i;
}
return i;
}
std::string
ClashAvoider::adjust_name(const raul::Path& old_path,
const raul::Path& new_path,
std::string name)
{
const auto name_suffix_start = numeric_suffix_start(name);
if (!name_suffix_start) {
return name; // No numeric suffix, just re-use old label
}
const auto name_suffix = atoi(name.c_str() + *name_suffix_start);
const auto old_suffix_start = numeric_suffix_start(old_path);
const auto new_suffix_start = numeric_suffix_start(new_path);
if (old_suffix_start && new_suffix_start) {
// Add the offset applied to the symbol to the label suffix
const auto old_suffix = atoi(old_path.c_str() + *old_suffix_start);
const auto new_suffix = atoi(new_path.c_str() + *new_suffix_start);
const auto offset = new_suffix - old_suffix;
return (name.substr(0, *name_suffix_start) +
std::to_string(name_suffix + offset));
}
// Add 1 to previous label suffix
return (name.substr(0, *name_suffix_start) +
std::to_string(name_suffix + 1));
}
} // namespace ingen