Files
textmate/Frameworks/settings/src/settings.cc
Allan Odgaard 9894969e67 Initial commit
2012-08-09 16:25:56 +02:00

193 lines
6.1 KiB
C++

#include "settings.h"
#include "parser.h"
#include "track_paths.h"
#include <OakSystem/application.h>
#include <plist/plist.h>
#include <OakSystem/process.h>
#include <plist/uuid.h>
#include <oak/oak.h>
#include <regexp/format_string.h>
#include <regexp/glob.h>
#include <oak/debug.h>
#include <io/io.h>
OAK_DEBUG_VAR(Settings);
namespace
{
static std::vector< std::pair<std::string, std::string> > global_variables ()
{
std::vector< std::pair<std::string, std::string> > res;
static char const* const preserve[] = { "DIALOG", "DIALOG_1", "DIALOG_PORT_NAME", "DIALOG_1_PORT_NAME" };
iterate(key, preserve)
{
if(char const* value = getenv(*key))
res.push_back(std::make_pair(*key, value));
}
res.push_back(std::make_pair("TM_PID", text::format("%d", getpid())));
res.push_back(std::make_pair("TM_FULLNAME", getpwuid(getuid())->pw_gecos ?: "John Doe"));
citerate(pair, oak::basic_environment())
res.push_back(*pair);
if(CFPropertyListRef cfPlist = CFPreferencesCopyAppValue(CFSTR("environmentVariables"), kCFPreferencesCurrentApplication))
{
if(CFGetTypeID(cfPlist) == CFArrayGetTypeID())
{
CFArrayRef cfArray = (CFArrayRef)cfPlist;
for(CFIndex i = 0; i < CFArrayGetCount(cfArray); ++i)
{
std::string name, value; bool enabled;
plist::dictionary_t const& plist = plist::convert(CFArrayGetValueAtIndex(cfArray, i));
if(plist::get_key_path(plist, "name", name) && plist::get_key_path(plist, "value", value) && (!plist::get_key_path(plist, "enabled", enabled) || enabled))
res.push_back(std::make_pair(name, value));
}
}
CFRelease(cfPlist);
}
return res;
}
static void expand_variable (std::string const& name, std::string const& value, std::map<std::string, std::string>& environment)
{
std::string const expanded = format_string::expand(value, environment);
environment[name] = expanded;
}
static std::vector<std::string> paths (std::string const& directory)
{
std::vector<std::string> res;
for(std::string cwd = directory; true; cwd = cwd == "/" ? path::home() : path::parent(cwd))
{
res.push_back(path::join(cwd, ".tm_properties"));
if(cwd == path::home())
break;
}
res.push_back(oak::application_t::path("Contents/Resources/Default.tmProperties"));
std::reverse(res.begin(), res.end());
return res;
}
static bool is_scope_selector (std::string const& str)
{
static std::string const RootScopes[] = { "text", "source", "attr" };
iterate(scope, RootScopes)
{
if(str.find(*scope) == 0 && (str.size() == scope->size() || str[scope->size()] == '.' || str[scope->size()] == ' '))
return true;
}
return str == "";
}
struct section_t
{
section_t (path::glob_t const& fileGlob, scope::selector_t const& scopeSelector, std::vector< std::pair<std::string, std::string> > const& variables) : file_glob(fileGlob), scope_selector(scopeSelector), variables(variables) { }
path::glob_t file_glob;
scope::selector_t scope_selector;
std::vector< std::pair<std::string, std::string> > variables;
};
static std::vector<section_t> parse_sections (std::string const& path)
{
ini_file_t iniFile(path);
std::string content = path::content(path);
if(content != NULL_STR)
parse_ini(content.data(), content.data() + content.size(), iniFile);
std::vector<section_t> res;
iterate(section, iniFile.sections)
{
std::vector< std::pair<std::string, std::string> > variables;
variables.push_back(std::make_pair("CWD", path::parent(path)));
iterate(pair, section->values)
variables.push_back(std::make_pair(pair->name, pair->value));
if(section->names.empty())
{
res.push_back(section_t("{,.}*", scope::selector_t(), variables));
}
else
{
iterate(name, section->names)
{
bool scope = is_scope_selector(*name);
res.push_back(section_t(scope ? "{,.}*" : *name, scope ? *name : scope::selector_t(), variables));
}
}
}
return res;
}
static std::vector<section_t> const& sections (std::string const& path)
{
static track_paths_t tracked_paths;
static std::map<std::string, std::vector<section_t> > cache;
if(tracked_paths.is_changed(path))
{
if(cache.size() > 64)
{
D(DBF_Settings, bug("clear cache\n"););
iterate(pair, cache)
tracked_paths.remove(pair->first);
cache.clear();
}
D(DBF_Settings, if(cache.find(path) != cache.end()) bug("reload %s\n", path.c_str()););
cache[path] = parse_sections(path);
D(DBF_Settings, bug("%zu properties files cached\n", cache.size()););
}
return cache[path];
}
std::map<std::string, std::string> expanded_variables_for (std::string const& directory, std::string const& path, scope::scope_t const& scope, std::map<std::string, std::string> variables)
{
D(DBF_Settings, bug("%s, %s, %s\n", directory.c_str(), path.c_str(), to_s(scope).c_str()););
citerate(pair, global_variables())
expand_variable(pair->first, pair->second, variables);
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_mutex_lock(&mutex);
citerate(file, paths(directory))
{
citerate(section, sections(*file))
{
if(section->file_glob.does_match(path) && section->scope_selector.does_match(scope))
{
iterate(pair, section->variables)
expand_variable(pair->first, pair->second, variables);
}
}
}
pthread_mutex_unlock(&mutex);
variables.erase("CWD");
return variables;
}
}
settings_t settings_for_path (std::string const& path, scope::scope_t const& scope, std::string const& directory, oak::uuid_t const& uuid, std::map<std::string, std::string> variables)
{
return expanded_variables_for(directory != NULL_STR ? directory : (path != NULL_STR ? path::parent(path) : path::home()), path, scope, variables);
}
std::map<std::string, std::string> variables_for_path (std::string const& path, scope::scope_t const& scope, std::map<std::string, std::string> variables)
{
variables = expanded_variables_for(path == NULL_STR ? path::home() : path::parent(path), path, scope, variables);
auto it = variables.begin();
while(it != variables.end())
{
auto prev = it++;
if(!prev->first.empty() && islower(prev->first[0]))
variables.erase(prev);
}
return variables;
}