2016-02-09 20:07:48 +00:00
|
|
|
#include "args.hh"
|
|
|
|
#include "hash.hh"
|
|
|
|
|
2020-05-10 19:35:07 +00:00
|
|
|
#include <glob.h>
|
|
|
|
|
2020-08-17 15:44:52 +00:00
|
|
|
#include <nlohmann/json.hpp>
|
|
|
|
|
2016-02-09 20:07:48 +00:00
|
|
|
namespace nix {
|
|
|
|
|
2020-05-04 20:40:19 +00:00
|
|
|
void Args::addFlag(Flag && flag_)
|
2017-06-07 16:41:20 +00:00
|
|
|
{
|
2020-05-04 20:40:19 +00:00
|
|
|
auto flag = std::make_shared<Flag>(std::move(flag_));
|
|
|
|
if (flag->handler.arity != ArityAny)
|
|
|
|
assert(flag->handler.arity == flag->labels.size());
|
2017-06-07 16:41:20 +00:00
|
|
|
assert(flag->longName != "");
|
2020-05-04 20:40:19 +00:00
|
|
|
longFlags[flag->longName] = flag;
|
2021-02-07 19:44:56 +00:00
|
|
|
for (auto & alias : flag->aliases)
|
|
|
|
longFlags[alias] = flag;
|
2020-05-04 20:40:19 +00:00
|
|
|
if (flag->shortName) shortFlags[flag->shortName] = flag;
|
2017-06-07 16:41:20 +00:00
|
|
|
}
|
|
|
|
|
2021-02-26 13:55:54 +00:00
|
|
|
void Args::removeFlag(const std::string & longName)
|
|
|
|
{
|
|
|
|
auto flag = longFlags.find(longName);
|
|
|
|
assert(flag != longFlags.end());
|
|
|
|
if (flag->second->shortName) shortFlags.erase(flag->second->shortName);
|
|
|
|
longFlags.erase(flag);
|
|
|
|
}
|
|
|
|
|
2020-10-09 07:39:51 +00:00
|
|
|
void Completions::add(std::string completion, std::string description)
|
|
|
|
{
|
2020-10-09 19:55:59 +00:00
|
|
|
assert(description.find('\n') == std::string::npos);
|
|
|
|
insert(Completion {
|
|
|
|
.completion = completion,
|
|
|
|
.description = description
|
|
|
|
});
|
2020-10-09 07:39:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Completion::operator<(const Completion & other) const
|
|
|
|
{ return completion < other.completion || (completion == other.completion && description < other.description); }
|
|
|
|
|
2021-12-22 11:37:59 +00:00
|
|
|
CompletionType completionType = ctNormal;
|
2020-10-09 07:39:51 +00:00
|
|
|
std::shared_ptr<Completions> completions;
|
2020-05-10 18:32:21 +00:00
|
|
|
|
|
|
|
std::string completionMarker = "___COMPLETE___";
|
|
|
|
|
|
|
|
std::optional<std::string> needsCompletion(std::string_view s)
|
|
|
|
{
|
|
|
|
if (!completions) return {};
|
|
|
|
auto i = s.find(completionMarker);
|
|
|
|
if (i != std::string::npos)
|
|
|
|
return std::string(s.begin(), i);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2016-02-09 20:07:48 +00:00
|
|
|
void Args::parseCmdline(const Strings & _cmdline)
|
|
|
|
{
|
|
|
|
Strings pendingArgs;
|
|
|
|
bool dashDash = false;
|
|
|
|
|
|
|
|
Strings cmdline(_cmdline);
|
|
|
|
|
2020-05-10 18:32:21 +00:00
|
|
|
if (auto s = getEnv("NIX_GET_COMPLETIONS")) {
|
|
|
|
size_t n = std::stoi(*s);
|
|
|
|
assert(n > 0 && n <= cmdline.size());
|
|
|
|
*std::next(cmdline.begin(), n - 1) += completionMarker;
|
|
|
|
completions = std::make_shared<decltype(completions)::element_type>();
|
2020-05-11 19:38:17 +00:00
|
|
|
verbosity = lvlError;
|
2020-05-10 18:32:21 +00:00
|
|
|
}
|
|
|
|
|
2021-01-28 14:37:43 +00:00
|
|
|
bool argsSeen = false;
|
2016-02-09 20:07:48 +00:00
|
|
|
for (auto pos = cmdline.begin(); pos != cmdline.end(); ) {
|
|
|
|
|
|
|
|
auto arg = *pos;
|
|
|
|
|
|
|
|
/* Expand compound dash options (i.e., `-qlf' -> `-q -l -f',
|
|
|
|
`-j3` -> `-j 3`). */
|
|
|
|
if (!dashDash && arg.length() > 2 && arg[0] == '-' && arg[1] != '-' && isalpha(arg[1])) {
|
2022-02-25 15:00:00 +00:00
|
|
|
*pos = (std::string) "-" + arg[1];
|
2016-02-09 20:07:48 +00:00
|
|
|
auto next = pos; ++next;
|
|
|
|
for (unsigned int j = 2; j < arg.length(); j++)
|
|
|
|
if (isalpha(arg[j]))
|
2022-02-25 15:00:00 +00:00
|
|
|
cmdline.insert(next, (std::string) "-" + arg[j]);
|
2016-02-09 20:07:48 +00:00
|
|
|
else {
|
2022-02-25 15:00:00 +00:00
|
|
|
cmdline.insert(next, std::string(arg, j));
|
2016-02-09 20:07:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
arg = *pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!dashDash && arg == "--") {
|
|
|
|
dashDash = true;
|
|
|
|
++pos;
|
|
|
|
}
|
|
|
|
else if (!dashDash && std::string(arg, 0, 1) == "-") {
|
|
|
|
if (!processFlag(pos, cmdline.end()))
|
2020-04-21 23:07:07 +00:00
|
|
|
throw UsageError("unrecognised flag '%1%'", arg);
|
2016-02-09 20:07:48 +00:00
|
|
|
}
|
|
|
|
else {
|
2021-01-28 14:37:43 +00:00
|
|
|
if (!argsSeen) {
|
|
|
|
argsSeen = true;
|
|
|
|
initialFlagsProcessed();
|
|
|
|
}
|
2020-12-03 21:45:44 +00:00
|
|
|
pos = rewriteArgs(cmdline, pos);
|
2016-02-09 20:07:48 +00:00
|
|
|
pendingArgs.push_back(*pos++);
|
|
|
|
if (processArgs(pendingArgs, false))
|
|
|
|
pendingArgs.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
processArgs(pendingArgs, true);
|
2021-01-28 14:37:43 +00:00
|
|
|
|
|
|
|
if (!argsSeen)
|
|
|
|
initialFlagsProcessed();
|
2016-02-09 20:07:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Args::processFlag(Strings::iterator & pos, Strings::iterator end)
|
|
|
|
{
|
|
|
|
assert(pos != end);
|
|
|
|
|
|
|
|
auto process = [&](const std::string & name, const Flag & flag) -> bool {
|
|
|
|
++pos;
|
2017-10-24 10:45:11 +00:00
|
|
|
std::vector<std::string> args;
|
2020-07-01 18:31:39 +00:00
|
|
|
bool anyCompleted = false;
|
2020-05-04 20:40:19 +00:00
|
|
|
for (size_t n = 0 ; n < flag.handler.arity; ++n) {
|
2017-08-29 12:28:57 +00:00
|
|
|
if (pos == end) {
|
2022-06-20 02:15:38 +00:00
|
|
|
if (flag.handler.arity == ArityAny || anyCompleted) break;
|
2020-05-10 19:50:32 +00:00
|
|
|
throw UsageError("flag '%s' requires %d argument(s)", name, flag.handler.arity);
|
2017-08-29 12:28:57 +00:00
|
|
|
}
|
2022-02-18 12:24:39 +00:00
|
|
|
if (auto prefix = needsCompletion(*pos)) {
|
|
|
|
anyCompleted = true;
|
|
|
|
if (flag.completer)
|
2020-05-10 19:50:32 +00:00
|
|
|
flag.completer(n, *prefix);
|
2022-02-18 12:24:39 +00:00
|
|
|
}
|
2020-05-10 19:50:32 +00:00
|
|
|
args.push_back(*pos++);
|
2016-02-09 20:07:48 +00:00
|
|
|
}
|
2020-07-01 18:31:39 +00:00
|
|
|
if (!anyCompleted)
|
|
|
|
flag.handler.fun(std::move(args));
|
2016-02-09 20:07:48 +00:00
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
2022-02-25 15:00:00 +00:00
|
|
|
if (std::string(*pos, 0, 2) == "--") {
|
2020-05-10 18:32:21 +00:00
|
|
|
if (auto prefix = needsCompletion(*pos)) {
|
|
|
|
for (auto & [name, flag] : longFlags) {
|
|
|
|
if (!hiddenCategories.count(flag->category)
|
|
|
|
&& hasPrefix(name, std::string(*prefix, 2)))
|
2020-10-09 07:39:51 +00:00
|
|
|
completions->add("--" + name, flag->description);
|
2020-05-10 18:32:21 +00:00
|
|
|
}
|
2022-02-18 12:24:39 +00:00
|
|
|
return false;
|
2020-05-10 18:32:21 +00:00
|
|
|
}
|
2022-02-25 15:00:00 +00:00
|
|
|
auto i = longFlags.find(std::string(*pos, 2));
|
2016-02-09 20:07:48 +00:00
|
|
|
if (i == longFlags.end()) return false;
|
2017-06-07 16:41:20 +00:00
|
|
|
return process("--" + i->first, *i->second);
|
2016-02-09 20:07:48 +00:00
|
|
|
}
|
|
|
|
|
2022-02-25 15:00:00 +00:00
|
|
|
if (std::string(*pos, 0, 1) == "-" && pos->size() == 2) {
|
2016-02-09 20:07:48 +00:00
|
|
|
auto c = (*pos)[1];
|
|
|
|
auto i = shortFlags.find(c);
|
|
|
|
if (i == shortFlags.end()) return false;
|
2017-06-07 16:41:20 +00:00
|
|
|
return process(std::string("-") + c, *i->second);
|
2016-02-09 20:07:48 +00:00
|
|
|
}
|
|
|
|
|
2020-05-10 18:32:21 +00:00
|
|
|
if (auto prefix = needsCompletion(*pos)) {
|
|
|
|
if (prefix == "-") {
|
2020-10-09 07:39:51 +00:00
|
|
|
completions->add("--");
|
|
|
|
for (auto & [flagName, flag] : shortFlags)
|
|
|
|
completions->add(std::string("-") + flagName, flag->description);
|
2020-05-10 18:32:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-09 20:07:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Args::processArgs(const Strings & args, bool finish)
|
|
|
|
{
|
|
|
|
if (expectedArgs.empty()) {
|
|
|
|
if (!args.empty())
|
2020-04-21 23:07:07 +00:00
|
|
|
throw UsageError("unexpected argument '%1%'", args.front());
|
2016-02-09 20:07:48 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto & exp = expectedArgs.front();
|
|
|
|
|
|
|
|
bool res = false;
|
|
|
|
|
2020-05-11 13:46:18 +00:00
|
|
|
if ((exp.handler.arity == ArityAny && finish) ||
|
|
|
|
(exp.handler.arity != ArityAny && args.size() == exp.handler.arity))
|
2016-02-09 20:07:48 +00:00
|
|
|
{
|
2017-10-24 10:45:11 +00:00
|
|
|
std::vector<std::string> ss;
|
2020-05-11 13:46:18 +00:00
|
|
|
for (const auto &[n, s] : enumerate(args)) {
|
2022-02-18 12:24:39 +00:00
|
|
|
if (auto prefix = needsCompletion(s)) {
|
|
|
|
ss.push_back(*prefix);
|
|
|
|
if (exp.completer)
|
2020-05-11 13:46:18 +00:00
|
|
|
exp.completer(n, *prefix);
|
2022-02-18 12:24:39 +00:00
|
|
|
} else
|
|
|
|
ss.push_back(s);
|
2020-05-11 13:46:18 +00:00
|
|
|
}
|
|
|
|
exp.handler.fun(ss);
|
2016-02-09 20:07:48 +00:00
|
|
|
expectedArgs.pop_front();
|
|
|
|
res = true;
|
|
|
|
}
|
|
|
|
|
2017-07-14 11:44:45 +00:00
|
|
|
if (finish && !expectedArgs.empty() && !expectedArgs.front().optional)
|
2016-02-09 20:07:48 +00:00
|
|
|
throw UsageError("more arguments are required");
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2020-08-17 15:44:52 +00:00
|
|
|
nlohmann::json Args::toJSON()
|
|
|
|
{
|
|
|
|
auto flags = nlohmann::json::object();
|
|
|
|
|
|
|
|
for (auto & [name, flag] : longFlags) {
|
|
|
|
auto j = nlohmann::json::object();
|
2021-02-07 19:44:56 +00:00
|
|
|
if (flag->aliases.count(name)) continue;
|
2020-08-17 15:44:52 +00:00
|
|
|
if (flag->shortName)
|
|
|
|
j["shortName"] = std::string(1, flag->shortName);
|
|
|
|
if (flag->description != "")
|
2022-09-13 14:58:32 +00:00
|
|
|
j["description"] = trim(flag->description);
|
2021-01-25 18:03:13 +00:00
|
|
|
j["category"] = flag->category;
|
2020-08-17 15:44:52 +00:00
|
|
|
if (flag->handler.arity != ArityAny)
|
|
|
|
j["arity"] = flag->handler.arity;
|
|
|
|
if (!flag->labels.empty())
|
|
|
|
j["labels"] = flag->labels;
|
|
|
|
flags[name] = std::move(j);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto args = nlohmann::json::array();
|
|
|
|
|
|
|
|
for (auto & arg : expectedArgs) {
|
|
|
|
auto j = nlohmann::json::object();
|
|
|
|
j["label"] = arg.label;
|
|
|
|
j["optional"] = arg.optional;
|
|
|
|
if (arg.handler.arity != ArityAny)
|
|
|
|
j["arity"] = arg.handler.arity;
|
|
|
|
args.push_back(std::move(j));
|
|
|
|
}
|
|
|
|
|
|
|
|
auto res = nlohmann::json::object();
|
2022-09-13 14:58:32 +00:00
|
|
|
res["description"] = trim(description());
|
2020-08-17 15:44:52 +00:00
|
|
|
res["flags"] = std::move(flags);
|
|
|
|
res["args"] = std::move(args);
|
2020-12-07 12:04:24 +00:00
|
|
|
auto s = doc();
|
|
|
|
if (s != "") res.emplace("doc", stripIndentation(s));
|
2020-08-17 15:44:52 +00:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2020-10-09 07:39:51 +00:00
|
|
|
static void hashTypeCompleter(size_t index, std::string_view prefix)
|
2020-06-26 06:46:46 +00:00
|
|
|
{
|
|
|
|
for (auto & type : hashTypes)
|
|
|
|
if (hasPrefix(type, prefix))
|
2020-10-09 07:39:51 +00:00
|
|
|
completions->add(type);
|
2020-06-26 06:46:46 +00:00
|
|
|
}
|
|
|
|
|
2020-05-04 20:40:19 +00:00
|
|
|
Args::Flag Args::Flag::mkHashTypeFlag(std::string && longName, HashType * ht)
|
2016-02-09 20:07:48 +00:00
|
|
|
{
|
2020-05-04 20:40:19 +00:00
|
|
|
return Flag {
|
|
|
|
.longName = std::move(longName),
|
|
|
|
.description = "hash algorithm ('md5', 'sha1', 'sha256', or 'sha512')",
|
|
|
|
.labels = {"hash-algo"},
|
|
|
|
.handler = {[ht](std::string s) {
|
|
|
|
*ht = parseHashType(s);
|
2020-05-10 19:50:32 +00:00
|
|
|
}},
|
2020-06-26 06:46:46 +00:00
|
|
|
.completer = hashTypeCompleter
|
2020-05-04 20:40:19 +00:00
|
|
|
};
|
2016-02-09 20:07:48 +00:00
|
|
|
}
|
|
|
|
|
2020-06-02 18:25:32 +00:00
|
|
|
Args::Flag Args::Flag::mkHashTypeOptFlag(std::string && longName, std::optional<HashType> * oht)
|
|
|
|
{
|
|
|
|
return Flag {
|
|
|
|
.longName = std::move(longName),
|
|
|
|
.description = "hash algorithm ('md5', 'sha1', 'sha256', or 'sha512'). Optional as can also be gotten from SRI hash itself.",
|
|
|
|
.labels = {"hash-algo"},
|
|
|
|
.handler = {[oht](std::string s) {
|
|
|
|
*oht = std::optional<HashType> { parseHashType(s) };
|
2020-06-26 06:46:46 +00:00
|
|
|
}},
|
|
|
|
.completer = hashTypeCompleter
|
2020-05-04 20:40:19 +00:00
|
|
|
};
|
2016-02-09 20:07:48 +00:00
|
|
|
}
|
|
|
|
|
2020-10-06 11:36:55 +00:00
|
|
|
static void _completePath(std::string_view prefix, bool onlyDirs)
|
2020-05-10 19:35:07 +00:00
|
|
|
{
|
2021-12-22 11:37:59 +00:00
|
|
|
completionType = ctFilenames;
|
2020-05-11 13:46:18 +00:00
|
|
|
glob_t globbuf;
|
2022-02-19 13:26:34 +00:00
|
|
|
int flags = GLOB_NOESCAPE;
|
2020-05-12 09:08:59 +00:00
|
|
|
#ifdef GLOB_ONLYDIR
|
|
|
|
if (onlyDirs)
|
|
|
|
flags |= GLOB_ONLYDIR;
|
|
|
|
#endif
|
2022-02-19 13:26:34 +00:00
|
|
|
// using expandTilde here instead of GLOB_TILDE(_CHECK) so that ~<Tab> expands to /home/user/
|
|
|
|
if (glob((expandTilde(prefix) + "*").c_str(), flags, nullptr, &globbuf) == 0) {
|
2020-05-12 09:08:59 +00:00
|
|
|
for (size_t i = 0; i < globbuf.gl_pathc; ++i) {
|
|
|
|
if (onlyDirs) {
|
2022-02-18 12:26:40 +00:00
|
|
|
auto st = stat(globbuf.gl_pathv[i]);
|
2020-05-12 09:08:59 +00:00
|
|
|
if (!S_ISDIR(st.st_mode)) continue;
|
|
|
|
}
|
2020-10-09 07:39:51 +00:00
|
|
|
completions->add(globbuf.gl_pathv[i]);
|
2020-05-12 09:08:59 +00:00
|
|
|
}
|
2020-05-10 19:35:07 +00:00
|
|
|
}
|
2022-02-19 13:26:34 +00:00
|
|
|
globfree(&globbuf);
|
2020-05-10 19:35:07 +00:00
|
|
|
}
|
|
|
|
|
2020-05-11 20:04:13 +00:00
|
|
|
void completePath(size_t, std::string_view prefix)
|
|
|
|
{
|
2020-10-06 11:36:55 +00:00
|
|
|
_completePath(prefix, false);
|
2020-05-11 20:04:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void completeDir(size_t, std::string_view prefix)
|
|
|
|
{
|
2020-10-06 11:36:55 +00:00
|
|
|
_completePath(prefix, true);
|
2020-05-11 20:04:13 +00:00
|
|
|
}
|
|
|
|
|
2016-02-09 20:07:48 +00:00
|
|
|
Strings argvToStrings(int argc, char * * argv)
|
|
|
|
{
|
|
|
|
Strings args;
|
|
|
|
argc--; argv++;
|
|
|
|
while (argc--) args.push_back(*argv++);
|
|
|
|
return args;
|
|
|
|
}
|
|
|
|
|
2021-01-28 15:04:47 +00:00
|
|
|
MultiCommand::MultiCommand(const Commands & commands_)
|
|
|
|
: commands(commands_)
|
2018-11-22 15:03:31 +00:00
|
|
|
{
|
2020-05-11 13:46:18 +00:00
|
|
|
expectArgs({
|
2020-08-17 17:33:18 +00:00
|
|
|
.label = "subcommand",
|
2020-05-11 13:46:18 +00:00
|
|
|
.optional = true,
|
2022-10-22 13:25:35 +00:00
|
|
|
.handler = {[=,this](std::string s) {
|
2020-05-11 13:46:18 +00:00
|
|
|
assert(!command);
|
|
|
|
auto i = commands.find(s);
|
2022-03-03 09:50:35 +00:00
|
|
|
if (i == commands.end()) {
|
|
|
|
std::set<std::string> commandNames;
|
|
|
|
for (auto & [name, _] : commands)
|
|
|
|
commandNames.insert(name);
|
|
|
|
auto suggestions = Suggestions::bestMatches(commandNames, s);
|
|
|
|
throw UsageError(suggestions, "'%s' is not a recognised command", s);
|
|
|
|
}
|
2020-05-11 13:46:18 +00:00
|
|
|
command = {s, i->second()};
|
2021-09-13 12:41:28 +00:00
|
|
|
command->second->parent = this;
|
2022-02-18 12:24:39 +00:00
|
|
|
}},
|
|
|
|
.completer = {[&](size_t, std::string_view prefix) {
|
|
|
|
for (auto & [name, command] : commands)
|
|
|
|
if (hasPrefix(name, prefix))
|
|
|
|
completions->add(name);
|
2020-05-11 13:46:18 +00:00
|
|
|
}}
|
|
|
|
});
|
2020-05-05 13:18:23 +00:00
|
|
|
|
|
|
|
categories[Command::catDefault] = "Available commands";
|
2018-11-22 15:03:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool MultiCommand::processFlag(Strings::iterator & pos, Strings::iterator end)
|
|
|
|
{
|
|
|
|
if (Args::processFlag(pos, end)) return true;
|
2020-05-05 13:18:23 +00:00
|
|
|
if (command && command->second->processFlag(pos, end)) return true;
|
2018-11-22 15:03:31 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MultiCommand::processArgs(const Strings & args, bool finish)
|
|
|
|
{
|
|
|
|
if (command)
|
2020-05-05 13:18:23 +00:00
|
|
|
return command->second->processArgs(args, finish);
|
2018-11-22 15:03:31 +00:00
|
|
|
else
|
|
|
|
return Args::processArgs(args, finish);
|
|
|
|
}
|
|
|
|
|
2022-06-20 02:15:38 +00:00
|
|
|
void MultiCommand::completionHook()
|
|
|
|
{
|
|
|
|
if (command)
|
|
|
|
return command->second->completionHook();
|
|
|
|
else
|
|
|
|
return Args::completionHook();
|
|
|
|
}
|
|
|
|
|
2020-08-17 15:44:52 +00:00
|
|
|
nlohmann::json MultiCommand::toJSON()
|
|
|
|
{
|
|
|
|
auto cmds = nlohmann::json::object();
|
|
|
|
|
|
|
|
for (auto & [name, commandFun] : commands) {
|
|
|
|
auto command = commandFun();
|
|
|
|
auto j = command->toJSON();
|
2021-01-25 17:19:32 +00:00
|
|
|
auto cat = nlohmann::json::object();
|
|
|
|
cat["id"] = command->category();
|
2022-09-13 14:58:32 +00:00
|
|
|
cat["description"] = trim(categories[command->category()]);
|
2021-01-25 17:19:32 +00:00
|
|
|
j["category"] = std::move(cat);
|
2020-08-17 15:44:52 +00:00
|
|
|
cmds[name] = std::move(j);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto res = Args::toJSON();
|
|
|
|
res["commands"] = std::move(cmds);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2016-02-09 20:07:48 +00:00
|
|
|
}
|