mirror of
https://github.com/textmate/textmate.git
synced 2026-02-14 16:34:57 -05:00
193 lines
6.1 KiB
C++
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;
|
|
}
|