1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169
|
#include "clicompleter.h"
#include "completionhelper.h"
#include "cli.h"
#include "cli_config.h"
#include "common/unused.h"
#include "commands/clicommand.h"
#include "parser/lexer.h"
#include "commands/clicommandfactory.h"
#include <QString>
#include <QDebug>
#if defined(Q_OS_WIN32)
#include "readline.h"
#elif defined(Q_OS_UNIX)
#include <readline/readline.h>
#endif
CliCompleter* CliCompleter::instance = nullptr;
CliCompleter::CliCompleter()
{
}
void CliCompleter::init(CLI* value)
{
rl_attempted_completion_function = CliCompleter::complete;
cli = value;
}
CliCompleter* CliCompleter::getInstance()
{
if (!instance)
instance = new CliCompleter();
return instance;
}
char** CliCompleter::complete(const char* text, int start, int end)
{
UNUSED(start);
#ifdef Q_OS_UNIX
// Unix readline needs this to disable the completion using rl_completion_entry_function
rl_attempted_completion_over = 1;
#endif
return toCharArray(getInstance()->completeInternal(QString::fromLocal8Bit(text), QString::fromLocal8Bit(rl_line_buffer), end));
}
QStringList CliCompleter::completeInternal(const QString& toBeReplaced, const QString& text, int curPos)
{
QString str;
if (!cli->getLine().isEmpty())
{
str += cli->getLine();
curPos += str.length();
}
str += text;
QStringList list;
if (str.startsWith(CFG_CLI.Console.CommandPrefixChar.get()))
list = completeCommand(str, curPos);
else
list = completeQuery(toBeReplaced, str, curPos);
list.removeDuplicates();
#ifdef Q_OS_WIN
if (list.size() == 1)
list[0] += " ";
#endif
#ifdef Q_OS_UNIX
// Unix readline treats first element in the list as a common value of all elements
if (list.size() > 0)
list.prepend(longestCommonPart(list));
#endif
return list;
}
QStringList CliCompleter::completeCommand(const QString& str, int curPos)
{
QStringList results;
QString text = str.mid(0, curPos);
QStringList cmdWords = tokenizeArgs(text);
if (cmdWords.size() == 0)
return results;
if (text[text.length()-1].isSpace())
cmdWords << "";
QString cmdStr = cmdWords[0].mid(1);
if (cmdWords.size() > 1)
{
CliCommand* command = CliCommandFactory::getCommand(cmdStr);
if (!command)
return results;
command->setup(cli);
results = command->complete(cmdWords.mid(1)).filter(QRegExp("^"+cmdWords.last()+".*"));
}
else
{
QStringList cmdNames = CliCommandFactory::getCommandNames().filter(QRegExp("^"+cmdStr+".*"));
cmdNames.sort(Qt::CaseInsensitive);
for (const QString& cmdName : cmdNames)
results << CFG_CLI.Console.CommandPrefixChar.get() + cmdName;
}
return results;
}
QStringList CliCompleter::completeQuery(const QString& toBeReplaced, const QString& str, int curPos)
{
QStringList list;
if (!cli->getCurrentDb())
return list;
bool keepOriginalStr = doKeepOriginalStr(str, curPos);
CompletionHelper completer(str, curPos, cli->getCurrentDb());
QList<ExpectedTokenPtr> expectedTokens = completer.getExpectedTokens().filtered();
for (const ExpectedTokenPtr& token : expectedTokens)
list << token->value;
list.removeAll("");
if (list.size() > 1)
list.removeOne(";"); // we don't want it together with other proposals, cause it introduces problems when proposed by completer
if (keepOriginalStr)
{
QMutableStringListIterator it(list);
while (it.hasNext())
it.next().prepend(toBeReplaced);
}
return list;
}
bool CliCompleter::doKeepOriginalStr(const QString& str, int curPos)
{
TokenList tokens = Lexer::tokenize(str.mid(0, curPos));
if (tokens.size() == 0)
return false;
return tokens.last()->isSeparating();
}
char** CliCompleter::toCharArray(const QStringList& list)
{
if (list.size() == 0)
return nullptr;
char** array = (char**)malloc((list.size() + 1) * sizeof(char*));
array[list.size()] = nullptr;
int i = 0;
for (const QString& str : list)
#if defined(Q_OS_WIN)
array[i++] = _strdup(str.toLocal8Bit().data());
#elif defined(Q_OS_UNIX)
array[i++] = strdup(str.toLocal8Bit().data());
#endif
return array;
}
|