Massive reformating and cleanup
This commit is contained in:
66
.clang-format
Normal file
66
.clang-format
Normal file
@ -0,0 +1,66 @@
|
||||
---
|
||||
Language: Cpp
|
||||
BasedOnStyle: Mozilla
|
||||
|
||||
AccessModifierOffset: -2
|
||||
AlignAfterOpenBracket: Align
|
||||
AlignConsecutiveMacros: true
|
||||
AlignConsecutiveAssignments: true
|
||||
AlignEscapedNewlines: Right
|
||||
AlignOperands: false
|
||||
AlignTrailingComments: true
|
||||
AllowAllArgumentsOnNextLine: true
|
||||
AllowAllConstructorInitializersOnNextLine: true
|
||||
AllowAllParametersOfDeclarationOnNextLine: true
|
||||
AllowShortBlocksOnASingleLine: true
|
||||
AllowShortCaseLabelsOnASingleLine: true
|
||||
AllowShortFunctionsOnASingleLine: true
|
||||
AllowShortIfStatementsOnASingleLine: Always
|
||||
AllowShortLambdasOnASingleLine: All
|
||||
AllowShortLoopsOnASingleLine: true
|
||||
AlwaysBreakAfterDefinitionReturnType: None
|
||||
AlwaysBreakAfterReturnType: None
|
||||
AlwaysBreakBeforeMultilineStrings: false
|
||||
AlwaysBreakTemplateDeclarations: Yes
|
||||
BreakBeforeBraces: Attach
|
||||
BreakBeforeTernaryOperators: false
|
||||
BreakConstructorInitializers: AfterColon
|
||||
ColumnLimit: 120
|
||||
CompactNamespaces: false
|
||||
ConstructorInitializerAllOnOneLineOrOnePerLine: false
|
||||
ExperimentalAutoDetectBinPacking: false
|
||||
FixNamespaceComments: false
|
||||
IncludeBlocks: Preserve
|
||||
IndentCaseLabels: true
|
||||
IndentWidth: 4
|
||||
PointerAlignment: Left
|
||||
ReflowComments: false
|
||||
SortIncludes: false
|
||||
SortUsingDeclarations: false
|
||||
SpaceAfterCStyleCast: false
|
||||
SpaceAfterLogicalNot: false
|
||||
SpaceAfterTemplateKeyword: true
|
||||
SpaceBeforeCtorInitializerColon: true
|
||||
SpaceBeforeInheritanceColon: true
|
||||
SpaceBeforeParens: ControlStatements
|
||||
SpaceBeforeRangeBasedForLoopColon: true
|
||||
SpaceInEmptyParentheses: false
|
||||
SpacesBeforeTrailingComments: 1
|
||||
SpacesInAngles: false
|
||||
SpacesInCStyleCastParentheses: false
|
||||
SpacesInContainerLiterals: false
|
||||
SpacesInParentheses: false
|
||||
SpacesInSquareBrackets: false
|
||||
Standard: Auto
|
||||
TabWidth: 4
|
||||
UseTab: Never
|
||||
|
||||
AllowShortEnumsOnASingleLine: false
|
||||
|
||||
BraceWrapping:
|
||||
AfterEnum: false
|
||||
|
||||
AlignConsecutiveDeclarations:
|
||||
Enabled: true
|
||||
|
||||
NamespaceIndentation: All
|
||||
@ -21,16 +21,14 @@
|
||||
#include "util/text.hpp"
|
||||
|
||||
#include "ast/ast_base.hpp"
|
||||
#include "ast/ast_token.hpp"
|
||||
#include "ast/ast_component.hpp"
|
||||
#include "ast/ast_executable.hpp"
|
||||
|
||||
#include "ast/ast_token.hpp"
|
||||
|
||||
namespace fsh {
|
||||
|
||||
class AstFactory {
|
||||
public:
|
||||
|
||||
// Generates an abstract syntax tree
|
||||
static std::shared_ptr<ExecutableNode> generate_ast(std::list<Token>& list) {
|
||||
auto it = list.begin();
|
||||
@ -43,4 +41,4 @@ class AstFactory {
|
||||
AstFactory(const AstFactory&) = default;
|
||||
};
|
||||
|
||||
}
|
||||
} // namespace fsh
|
||||
@ -1,9 +1,9 @@
|
||||
#pragma once
|
||||
|
||||
#include <string>
|
||||
#include <memory>
|
||||
#include <stdexcept>
|
||||
#include <sstream>
|
||||
#include <stdexcept>
|
||||
#include <string>
|
||||
|
||||
#include "lexer.hpp"
|
||||
#include "util/text.hpp"
|
||||
@ -24,6 +24,7 @@ enum NodeType {
|
||||
|
||||
class AstNode {
|
||||
friend class std::shared_ptr<AstNode>;
|
||||
|
||||
public:
|
||||
NodeType gtype() { return type; }
|
||||
const std::string& gname() { return name; };
|
||||
@ -32,16 +33,19 @@ class AstNode {
|
||||
using TokenType = Lexer::TokenType;
|
||||
AstNode(NodeType type, std::string name) : type(type), name(name) {}
|
||||
|
||||
|
||||
template <class T>
|
||||
static std::shared_ptr<T> Optional(std::list<Token>::iterator& it, std::shared_ptr<T>& node_ptr);
|
||||
|
||||
template <class T>
|
||||
static std::shared_ptr<T> Optional(std::list<Token>::iterator& it) {
|
||||
std::shared_ptr<T> node_ptr;
|
||||
return Optional<T>(it, node_ptr);
|
||||
}
|
||||
|
||||
template <class T>
|
||||
static std::shared_ptr<T> Mandatory(std::list<Token>::iterator& it) {return T::build(it);}
|
||||
static std::shared_ptr<T> Mandatory(std::list<Token>::iterator& it) {
|
||||
return T::build(it);
|
||||
}
|
||||
|
||||
private:
|
||||
AstNode();
|
||||
@ -54,8 +58,10 @@ class ExecutableNode : public AstNode {
|
||||
public:
|
||||
virtual void print(int indent) {};
|
||||
virtual void execute(std::istream& in, std::ostream& out) {}
|
||||
|
||||
protected:
|
||||
ExecutableNode(NodeType type, std::string Name) : AstNode(type, Name) {}
|
||||
|
||||
private:
|
||||
};
|
||||
|
||||
@ -64,4 +70,4 @@ public:
|
||||
AstBuildError(std::string err) : std::runtime_error(err) {}
|
||||
};
|
||||
|
||||
}
|
||||
} // namespace fsh
|
||||
@ -19,14 +19,11 @@ class CommandArgumentNode : public AstNode {
|
||||
std::string& gvalue() { return value; }
|
||||
|
||||
protected:
|
||||
CommandArgumentNode(TokenNode<TokenType::WORD>::shared word)
|
||||
: AstNode(NodeType::COMMAND_ARGUMENT, "StringArgument"),
|
||||
type(word->gtoken_type()),
|
||||
value(word->gvalue()) {}
|
||||
CommandArgumentNode(TokenNode<TokenType::STRING_LITERAL>::shared word)
|
||||
: AstNode(NodeType::COMMAND_ARGUMENT, "StringArgument"),
|
||||
type(word->gtoken_type()),
|
||||
value(word->gvalue()) {}
|
||||
CommandArgumentNode(TokenNode<TokenType::WORD>::shared word) :
|
||||
AstNode(NodeType::COMMAND_ARGUMENT, "StringArgument"), type(word->gtoken_type()), value(word->gvalue()) {}
|
||||
|
||||
CommandArgumentNode(TokenNode<TokenType::STRING_LITERAL>::shared word) :
|
||||
AstNode(NodeType::COMMAND_ARGUMENT, "StringArgument"), type(word->gtoken_type()), value(word->gvalue()) {}
|
||||
|
||||
private:
|
||||
Lexer::TokenType type;
|
||||
@ -40,15 +37,13 @@ class LRedirectNode : public AstNode {
|
||||
std::string gvalue() { return input; }
|
||||
|
||||
protected:
|
||||
LRedirectNode(TokenNode<TokenType::LREDIRECTION>::shared, TokenNode<TokenType::WORD>::shared in)
|
||||
: AstNode(NodeType::LREDIRECTS, "LRedirectNode"),
|
||||
input(in->gvalue()) {}
|
||||
LRedirectNode(TokenNode<TokenType::LREDIRECTION>::shared, TokenNode<TokenType::WORD>::shared in) :
|
||||
AstNode(NodeType::LREDIRECTS, "LRedirectNode"), input(in->gvalue()) {}
|
||||
|
||||
private:
|
||||
std::string input;
|
||||
};
|
||||
|
||||
|
||||
class RRedirectNode : public AstNode {
|
||||
public:
|
||||
static std::shared_ptr<RRedirectNode> build(std::list<Token>::iterator& it);
|
||||
@ -57,10 +52,8 @@ class RRedirectNode : public AstNode {
|
||||
bool gappend() { return append; }
|
||||
|
||||
protected:
|
||||
RRedirectNode(TokenNode<TokenType::RREDIRECTION>::shared rr, TokenNode<TokenType::WORD>::shared out)
|
||||
: AstNode(NodeType::RREDIRECTS, "LRedirectNode"),
|
||||
output(out->gvalue()),
|
||||
append(rr->gvalue().size()>1) {}
|
||||
RRedirectNode(TokenNode<TokenType::RREDIRECTION>::shared rr, TokenNode<TokenType::WORD>::shared out) :
|
||||
AstNode(NodeType::RREDIRECTS, "LRedirectNode"), output(out->gvalue()), append(rr->gvalue().size() > 1) {}
|
||||
|
||||
private:
|
||||
std::string output;
|
||||
@ -76,10 +69,10 @@ class RedirectsNode : public AstNode {
|
||||
bool gappend() { return append; }
|
||||
|
||||
protected:
|
||||
RedirectsNode(std::shared_ptr<LRedirectNode> in, std::shared_ptr<RRedirectNode> out)
|
||||
: AstNode(NodeType::REDIRECTS, "RedirectNode"),
|
||||
input(util::mk_optional(in)),
|
||||
output(util::mk_optional(out)) {if(out) append = out->gappend();}
|
||||
RedirectsNode(std::shared_ptr<LRedirectNode> in, std::shared_ptr<RRedirectNode> out) :
|
||||
AstNode(NodeType::REDIRECTS, "RedirectNode"), input(util::mk_optional(in)), output(util::mk_optional(out)) {
|
||||
if (out) append = out->gappend();
|
||||
}
|
||||
|
||||
private:
|
||||
std::optional<std::string> input;
|
||||
@ -87,4 +80,4 @@ class RedirectsNode : public AstNode {
|
||||
bool append = false;
|
||||
};
|
||||
|
||||
}
|
||||
} // namespace fsh
|
||||
@ -1,13 +1,12 @@
|
||||
#pragma once
|
||||
|
||||
#include <memory>
|
||||
#include <stdexcept>
|
||||
#include <optional>
|
||||
#include <stdexcept>
|
||||
|
||||
#include "ast/ast_base.hpp"
|
||||
#include "ast/ast_token.hpp"
|
||||
#include "ast/ast_component.hpp"
|
||||
|
||||
#include "ast/ast_token.hpp"
|
||||
|
||||
#include "lexer.hpp"
|
||||
#include "util/text.hpp"
|
||||
@ -16,22 +15,19 @@ namespace fsh {
|
||||
|
||||
class CommandNode : public ExecutableNode {
|
||||
using ArgumentVec = std::vector<std::shared_ptr<CommandArgumentNode> >;
|
||||
|
||||
public:
|
||||
static std::shared_ptr<CommandNode> build(std::list<Token>::iterator& it);
|
||||
virtual void print(int indent) override;
|
||||
virtual void execute(std::istream& in, std::ostream& out) override;
|
||||
|
||||
protected:
|
||||
CommandNode(
|
||||
TokenNode<TokenType::WORD>::shared cmd_name,
|
||||
CommandNode(TokenNode<TokenType::WORD>::shared cmd_name,
|
||||
TokenNode<TokenType::FLAG>::shared flag,
|
||||
ArgumentVec args,
|
||||
std::shared_ptr<RedirectsNode> redirects
|
||||
) : ExecutableNode(NodeType::COMMAND, "CommandNode"),
|
||||
cmd_name(cmd_name->gvalue()),
|
||||
flag(util::mk_optional(flag)),
|
||||
redirects(redirects),
|
||||
args(args) {}
|
||||
std::shared_ptr<RedirectsNode> redirects) :
|
||||
ExecutableNode(NodeType::COMMAND, "CommandNode"), cmd_name(cmd_name->gvalue()), flag(util::mk_optional(flag)),
|
||||
redirects(redirects), args(args) {}
|
||||
|
||||
private:
|
||||
std::string cmd_name;
|
||||
@ -47,17 +43,14 @@ class PipeLineNode : public ExecutableNode {
|
||||
virtual void execute(std::istream& in, std::ostream& out) override;
|
||||
|
||||
protected:
|
||||
PipeLineNode(std::shared_ptr<CommandNode> l, std::shared_ptr<ExecutableNode> r)
|
||||
: ExecutableNode(NodeType::PIPELINE_COMMAND, "PipeLine"),
|
||||
l_command(l),
|
||||
r_command(r) {}
|
||||
PipeLineNode(std::shared_ptr<CommandNode> l, std::shared_ptr<ExecutableNode> r) :
|
||||
ExecutableNode(NodeType::PIPELINE_COMMAND, "PipeLine"), l_command(l), r_command(r) {}
|
||||
|
||||
private:
|
||||
std::shared_ptr<ExecutableNode> l_command;
|
||||
std::shared_ptr<ExecutableNode> r_command;
|
||||
};
|
||||
|
||||
|
||||
class CommandLineNode : public ExecutableNode {
|
||||
public:
|
||||
static std::shared_ptr<CommandLineNode> build(std::list<Token>::iterator& it);
|
||||
@ -65,14 +58,12 @@ class CommandLineNode : public ExecutableNode {
|
||||
virtual void execute(std::istream& in, std::ostream& out) override;
|
||||
|
||||
protected:
|
||||
CommandLineNode(std::shared_ptr<ExecutableNode> command, std::shared_ptr<ExecutableNode> pipe = nullptr)
|
||||
: ExecutableNode(NodeType::COMMAND_LINE, "CommandLine"),
|
||||
command(command),
|
||||
pipe(pipe) {}
|
||||
CommandLineNode(std::shared_ptr<ExecutableNode> command, std::shared_ptr<ExecutableNode> pipe = nullptr) :
|
||||
ExecutableNode(NodeType::COMMAND_LINE, "CommandLine"), command(command), pipe(pipe) {}
|
||||
|
||||
private:
|
||||
std::shared_ptr<ExecutableNode> command;
|
||||
std::shared_ptr<ExecutableNode> pipe;
|
||||
};
|
||||
|
||||
}
|
||||
} // namespace fsh
|
||||
@ -4,7 +4,6 @@
|
||||
#include "ast/ast_base.hpp"
|
||||
#include "util/text.hpp"
|
||||
|
||||
|
||||
namespace fsh {
|
||||
|
||||
template <Lexer::TokenType T>
|
||||
@ -16,8 +15,7 @@ class TokenNode : public AstNode {
|
||||
static std::shared_ptr<TokenNode> build(std::list<Token>::iterator& it);
|
||||
|
||||
protected:
|
||||
TokenNode(std::string value)
|
||||
: AstNode(NodeType::TOKEN, util::tokens[T]), value(value) {}
|
||||
TokenNode(std::string value) : AstNode(NodeType::TOKEN, util::tokens[T]), value(value) {}
|
||||
|
||||
private:
|
||||
std::string value;
|
||||
|
||||
@ -21,22 +21,19 @@ public:
|
||||
using ArgNodes = std::vector<std::shared_ptr<CommandArgumentNode> >;
|
||||
using BuildFunc = std::shared_ptr<_Argument> (*)(void);
|
||||
|
||||
struct ArgRule
|
||||
{
|
||||
struct ArgRule {
|
||||
BuildFunc build;
|
||||
bool mandatory;
|
||||
bool extends;
|
||||
};
|
||||
|
||||
struct FlagRule
|
||||
{
|
||||
struct FlagRule {
|
||||
BuildFunc build;
|
||||
bool specialinput; //Not implemented
|
||||
bool capturing;
|
||||
bool extends; //Not implemented
|
||||
};
|
||||
|
||||
|
||||
void add_input_rule() {
|
||||
has_input = true;
|
||||
pos_arg_rules.push_back({ &_Argument::create<ArgInput>, false, false });
|
||||
@ -53,6 +50,7 @@ public:
|
||||
}
|
||||
|
||||
void parse(ArgManager& manager, ArgNodes& vec, FlagNode flag);
|
||||
|
||||
bool ghas_input() { return has_input; }
|
||||
|
||||
private:
|
||||
|
||||
@ -9,12 +9,13 @@ namespace fsh{
|
||||
class _Argument {
|
||||
public:
|
||||
virtual void svalue(const std::string& can, const Lexer::TokenType& type = Lexer::TokenType::FLAG) {}
|
||||
virtual void svalue(std::shared_ptr<CommandArgumentNode> can) {
|
||||
svalue(can->gvalue(), can->gtoken_type());
|
||||
}
|
||||
virtual void svalue(std::shared_ptr<CommandArgumentNode> can) { svalue(can->gvalue(), can->gtoken_type()); }
|
||||
|
||||
template <typename T>
|
||||
static std::shared_ptr<_Argument> create() {return std::make_shared<T>();}
|
||||
static std::shared_ptr<_Argument> create() {
|
||||
return std::make_shared<T>();
|
||||
}
|
||||
|
||||
protected:
|
||||
_Argument() {}
|
||||
};
|
||||
|
||||
@ -10,14 +10,13 @@ namespace fsh{
|
||||
template <typename T>
|
||||
class Argument : public _Argument {
|
||||
public:
|
||||
static T& get(std::shared_ptr<_Argument> a) {
|
||||
return std::dynamic_pointer_cast<Argument<T> >(a)->gvalue();
|
||||
}
|
||||
static T& get(std::shared_ptr<_Argument> a) { return std::dynamic_pointer_cast<Argument<T> >(a)->gvalue(); }
|
||||
|
||||
Argument() {}
|
||||
|
||||
virtual void svalue(const std::string& val, const Lexer::TokenType& type) override;
|
||||
virtual T& gvalue() { return value; }
|
||||
|
||||
private:
|
||||
bool is_string_literal; // Currently no getter
|
||||
T value;
|
||||
@ -29,9 +28,7 @@ void Argument<T>::svalue(const std::string& val, const Lexer::TokenType& type) {
|
||||
value = val;
|
||||
} else {
|
||||
std::stringstream ss_val(val);
|
||||
if(!(ss_val >> value)) {
|
||||
throw std::invalid_argument("Incorrect type");
|
||||
}
|
||||
if (!(ss_val >> value)) { throw std::invalid_argument("Incorrect type"); }
|
||||
}
|
||||
is_string_literal = type == Lexer::TokenType::STRING_LITERAL;
|
||||
}
|
||||
|
||||
@ -14,7 +14,6 @@ class ArgManager {
|
||||
public:
|
||||
ArgManager() {}
|
||||
|
||||
|
||||
std::istream& get_input(const unsigned int id) {
|
||||
if (id < pos_argument.size()) return ArgInput::get(pos_argument[id]);
|
||||
return util::cin;
|
||||
@ -25,21 +24,16 @@ public:
|
||||
if ((unsigned int)id < pos_argument.size()) return Argument<T>::get(pos_argument[id]);
|
||||
return std::make_optional<T>();
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
std::optional<T> get(const std::string& id) {
|
||||
if (flags.find(id) != flags.end()) return Argument<T>::get(flags[id]);
|
||||
return std::make_optional<T>();
|
||||
}
|
||||
bool get(const std::string& id) {
|
||||
return flags.find(id) != flags.end();
|
||||
}
|
||||
bool get(const std::string& id) { return flags.find(id) != flags.end(); }
|
||||
|
||||
void push_arg(std::shared_ptr<_Argument> arg) {
|
||||
pos_argument.push_back(arg);
|
||||
}
|
||||
void push_flag(const std::string &id,std::shared_ptr<_Argument> arg = nullptr) {
|
||||
flags[id] = arg;
|
||||
}
|
||||
void push_arg(std::shared_ptr<_Argument> arg) { pos_argument.push_back(arg); }
|
||||
void push_flag(const std::string& id, std::shared_ptr<_Argument> arg = nullptr) { flags[id] = arg; }
|
||||
|
||||
private:
|
||||
PosArgs pos_argument;
|
||||
|
||||
@ -3,36 +3,28 @@
|
||||
#include <unordered_map>
|
||||
|
||||
#include "cmd/cmd_base.hpp"
|
||||
#include "cmd/cmd_wc.hpp"
|
||||
#include "cmd/cmd_time.hpp"
|
||||
#include "cmd/cmd_date.hpp"
|
||||
#include "cmd/cmd_echo.hpp"
|
||||
#include "cmd/cmd_touch.hpp"
|
||||
#include "cmd/cmd_misc.hpp"
|
||||
#include "cmd/cmd_time.hpp"
|
||||
#include "cmd/cmd_touch.hpp"
|
||||
#include "cmd/cmd_wc.hpp"
|
||||
|
||||
namespace fsh
|
||||
{
|
||||
namespace fsh {
|
||||
|
||||
class CommandRegistry {
|
||||
|
||||
public:
|
||||
|
||||
static CommandRegistry& instance() {
|
||||
static CommandRegistry cmd_registry;
|
||||
return cmd_registry;
|
||||
}
|
||||
|
||||
Command& get(const std::string n) {
|
||||
if(cmds.find(n) == cmds.end()) {
|
||||
throw std::runtime_error("Command not found");
|
||||
}
|
||||
if (cmds.find(n) == cmds.end()) { throw std::runtime_error("Command not found"); }
|
||||
return *(cmds[n]);
|
||||
}
|
||||
|
||||
Command& operator[](const std::string n) {
|
||||
return get(n);
|
||||
}
|
||||
|
||||
Command& operator[](const std::string n) { return get(n); }
|
||||
|
||||
private:
|
||||
CommandRegistry() {
|
||||
@ -46,8 +38,6 @@ class CommandRegistry {
|
||||
}
|
||||
|
||||
std::unordered_map<std::string, std::unique_ptr<Command> > cmds;
|
||||
|
||||
};
|
||||
|
||||
|
||||
}
|
||||
} // namespace fsh
|
||||
@ -27,15 +27,12 @@ class Command {
|
||||
}
|
||||
|
||||
protected:
|
||||
|
||||
virtual void register_flags() {};
|
||||
virtual void run(std::istream& in, std::ostream& out, ArgManager& args) {};
|
||||
|
||||
ArgFactory& get_factory() { return arg_factory; }
|
||||
|
||||
private:
|
||||
ArgFactory arg_factory;
|
||||
};
|
||||
|
||||
|
||||
}
|
||||
@ -5,7 +5,6 @@
|
||||
|
||||
#include "cmd/cmd_base.hpp"
|
||||
|
||||
|
||||
namespace fsh {
|
||||
|
||||
class CmdDate : public Command {
|
||||
@ -15,7 +14,6 @@ class CmdDate : public Command {
|
||||
std::time_t time = std::time(nullptr);
|
||||
out << std::asctime(std::localtime(&time));
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
@ -15,12 +15,9 @@ class CmdEcho : public Command {
|
||||
virtual void run(std::istream& in, std::ostream& out, ArgManager& args) override {
|
||||
std::string s;
|
||||
std::string o;
|
||||
while(getline(in, s)) {
|
||||
o += s + "\n";
|
||||
}
|
||||
while (getline(in, s)) { o += s + "\n"; }
|
||||
out << o;
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
@ -6,7 +6,6 @@
|
||||
#include "fsh.hpp"
|
||||
#include "ast/ast.hpp"
|
||||
|
||||
|
||||
namespace fsh {
|
||||
|
||||
class CmdExit : public Command {
|
||||
@ -15,19 +14,16 @@ class CmdExit : public Command {
|
||||
virtual void run(std::istream& in, std::ostream& out, ArgManager& args) override {
|
||||
fsh::instance().environment["EXITING"] = "1";
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
class CmdPrintTree : public Command {
|
||||
|
||||
protected:
|
||||
|
||||
virtual void register_flags() override {
|
||||
auto& factory = get_factory();
|
||||
factory.add_input_rule();
|
||||
}
|
||||
|
||||
|
||||
virtual void run(std::istream& in, std::ostream& out, ArgManager& args) override {
|
||||
std::string line;
|
||||
std::getline(in, line);
|
||||
@ -37,7 +33,6 @@ class CmdPrintTree : public Command {
|
||||
|
||||
ast->print(0);
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
@ -14,7 +14,6 @@ class CmdTime : public Command {
|
||||
std::time_t time = std::time(nullptr);
|
||||
out << std::put_time(std::localtime(&time), "%T%n");
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
@ -18,7 +18,6 @@ class CmdTouch : public Command {
|
||||
throw std::runtime_error("File exists");
|
||||
}
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
@ -33,7 +33,6 @@ class CmdWc : public Command {
|
||||
}
|
||||
|
||||
virtual void run(std::istream& in, std::ostream& out, ArgManager& args) override {
|
||||
bool prev_space = true;
|
||||
if (args.get("-c")) {
|
||||
out << count_chars(in) << "\n";
|
||||
} else if (args.get("-w")) {
|
||||
@ -42,7 +41,6 @@ class CmdWc : public Command {
|
||||
throw std::invalid_argument("Didn't provide flag");
|
||||
}
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
@ -6,8 +6,7 @@
|
||||
#include "ast/ast.hpp"
|
||||
#include "util/input.hpp"
|
||||
|
||||
namespace fsh
|
||||
{
|
||||
namespace fsh {
|
||||
|
||||
class fsh {
|
||||
public:
|
||||
@ -26,8 +25,6 @@ private:
|
||||
environment["PROMPT"] = "$";
|
||||
environment["QUIT"] = "";
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
|
||||
}
|
||||
@ -1,7 +1,7 @@
|
||||
#pragma once
|
||||
|
||||
#include <list>
|
||||
#include <functional>
|
||||
#include <list>
|
||||
#include <string>
|
||||
|
||||
namespace fsh {
|
||||
@ -17,6 +17,7 @@ public:
|
||||
LREDIRECTION,
|
||||
RREDIRECTION,
|
||||
PIPE,
|
||||
|
||||
END_OF_STREAM
|
||||
};
|
||||
|
||||
@ -33,14 +34,12 @@ private:
|
||||
unsigned int idx;
|
||||
|
||||
Lexer(std::string str) : workspace(str), idx(0) {}
|
||||
|
||||
};
|
||||
using Token = Lexer::Token;
|
||||
|
||||
|
||||
|
||||
class LexerStateMachine {
|
||||
friend class Lexer;
|
||||
|
||||
private:
|
||||
enum State {
|
||||
START,
|
||||
@ -60,7 +59,8 @@ private:
|
||||
Lexer::TokenType type;
|
||||
std::vector<StateHandler> state_handlers;
|
||||
|
||||
LexerStateMachine(Lexer& parser) : parser(parser), token(""), type(Lexer::TokenType::WHITESPACE), state_handlers(STATE_COUNT) {
|
||||
LexerStateMachine(Lexer& parser) :
|
||||
parser(parser), token(""), type(Lexer::TokenType::WHITESPACE), state_handlers(STATE_COUNT) {
|
||||
state_handlers[State::START] = std::bind(&LexerStateMachine::handle_start, this);
|
||||
state_handlers[State::REDIRECT] = std::bind(&LexerStateMachine::handle_redirect, this);
|
||||
state_handlers[State::FLAG_OPT] = std::bind(&LexerStateMachine::handle_flag_opt, this);
|
||||
@ -81,12 +81,10 @@ private:
|
||||
State word_like_handler(Lexer::TokenType type, State state);
|
||||
};
|
||||
|
||||
inline LexerStateMachine::State LexerStateMachine::set_type(
|
||||
Lexer::TokenType type = Lexer::TokenType::WHITESPACE,
|
||||
State state = State::END
|
||||
){
|
||||
inline LexerStateMachine::State LexerStateMachine::set_type(Lexer::TokenType type = Lexer::TokenType::WHITESPACE,
|
||||
State state = State::END) {
|
||||
this->type = type;
|
||||
return state;
|
||||
}
|
||||
|
||||
}
|
||||
} // namespace fsh
|
||||
@ -7,9 +7,7 @@
|
||||
#include <iostream>
|
||||
#endif
|
||||
|
||||
|
||||
namespace fsh::util
|
||||
{
|
||||
namespace fsh::util {
|
||||
|
||||
/**
|
||||
* Applies some settings to terminals to fix some inconsistencies between
|
||||
@ -31,9 +29,8 @@ public:
|
||||
RIGHT_ARROW
|
||||
};
|
||||
|
||||
CursorIStreamBuffer() : valid(0), read_chars(0), cursor_pos(0), buffer() {
|
||||
setg(buffer,buffer,buffer);
|
||||
}
|
||||
CursorIStreamBuffer() : valid(0), read_chars(0), cursor_pos(0), buffer() { setg(buffer, buffer, buffer); }
|
||||
|
||||
protected:
|
||||
int underflow() override;
|
||||
// int uflow() override;
|
||||
@ -53,6 +50,7 @@ private:
|
||||
class CursorIStream : public std::istream {
|
||||
public:
|
||||
CursorIStream() : std::istream(&buffer) {}
|
||||
|
||||
private:
|
||||
CursorIStreamBuffer buffer;
|
||||
};
|
||||
@ -64,6 +62,4 @@ extern std::istream& cin;
|
||||
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
}
|
||||
@ -7,14 +7,11 @@
|
||||
|
||||
namespace fsh::util {
|
||||
|
||||
|
||||
extern const char* tokens[];
|
||||
|
||||
template <class T>
|
||||
static inline std::optional<std::string> mk_optional(T t) {
|
||||
if(t){
|
||||
return t->gvalue();
|
||||
}
|
||||
if (t) { return t->gvalue(); }
|
||||
return std::optional<std::string>();
|
||||
}
|
||||
|
||||
@ -25,30 +22,21 @@ static inline bool contains(char x, std::string text) {
|
||||
return 0;
|
||||
};
|
||||
|
||||
|
||||
static inline std::ifstream input(const std::string& x) {
|
||||
std::ifstream in(x, std::ios::in);
|
||||
if(!in) {
|
||||
throw std::runtime_error("Can't open file");
|
||||
}
|
||||
if (!in) { throw std::runtime_error("Can't open file"); }
|
||||
return in;
|
||||
}
|
||||
|
||||
|
||||
static inline std::ofstream output(const std::string& x) {
|
||||
std::ofstream out(x, std::ios::out);
|
||||
if(!out.is_open()) {
|
||||
throw std::runtime_error("Can't open file");
|
||||
}
|
||||
if (!out.is_open()) { throw std::runtime_error("Can't open file"); }
|
||||
return out;
|
||||
}
|
||||
|
||||
|
||||
static inline std::ofstream output_append(const std::string& x) {
|
||||
std::ofstream out(x, std::ios::app | std::ios::out);
|
||||
if(!out.is_open()) {
|
||||
throw std::runtime_error("Can't open file");
|
||||
}
|
||||
if (!out.is_open()) { throw std::runtime_error("Can't open file"); }
|
||||
return out;
|
||||
}
|
||||
|
||||
|
||||
@ -28,9 +28,9 @@ typename TokenNode<T>::shared TokenNode<T>::build(std::list<Token>::iterator& it
|
||||
|
||||
std::shared_ptr<CommandArgumentNode> CommandArgumentNode::build(std::list<Token>::iterator& it) {
|
||||
auto word = Optional<TokenNode<TokenType::WORD>>(it);
|
||||
if(word) {
|
||||
return std::shared_ptr<CommandArgumentNode>(new CommandArgumentNode(word));
|
||||
}
|
||||
|
||||
if (word) return std::shared_ptr<CommandArgumentNode>(new CommandArgumentNode(word));
|
||||
|
||||
auto sl = Mandatory<TokenNode<TokenType::STRING_LITERAL>>(it);
|
||||
return std::shared_ptr<CommandArgumentNode>(new CommandArgumentNode(sl));
|
||||
}
|
||||
@ -38,19 +38,23 @@ std::shared_ptr<CommandArgumentNode> CommandArgumentNode::build(std::list<Token>
|
||||
std::shared_ptr<RRedirectNode> RRedirectNode::build(std::list<Token>::iterator& it) {
|
||||
auto Rr = Mandatory<TokenNode<TokenType::RREDIRECTION>>(it);
|
||||
auto output = Mandatory<TokenNode<TokenType::WORD>>(it);
|
||||
|
||||
return std::shared_ptr<RRedirectNode>(new RRedirectNode(Rr, output));
|
||||
}
|
||||
|
||||
std::shared_ptr<LRedirectNode> LRedirectNode::build(std::list<Token>::iterator& it) {
|
||||
auto Lr = Mandatory<TokenNode<TokenType::LREDIRECTION>>(it);
|
||||
auto output = Mandatory<TokenNode<TokenType::WORD>>(it);
|
||||
|
||||
return std::shared_ptr<LRedirectNode>(new LRedirectNode(Lr, output));
|
||||
}
|
||||
|
||||
std::shared_ptr<RedirectsNode> RedirectsNode::build(std::list<Token>::iterator& it) {
|
||||
TokenNode<TokenType::WORD>::shared input, output;
|
||||
|
||||
auto Lr = Optional<LRedirectNode>(it);
|
||||
auto Rr = Optional<RRedirectNode>(it);
|
||||
|
||||
if (!Lr) Lr = Optional<LRedirectNode>(it);
|
||||
if (!Lr && !Rr) throw AstBuildError("No redirects where provided");
|
||||
return std::shared_ptr<RedirectsNode>(new RedirectsNode(Lr, Rr));
|
||||
@ -59,8 +63,9 @@ std::shared_ptr<RedirectsNode> RedirectsNode::build(std::list<Token>::iterator&
|
||||
std::shared_ptr<CommandNode> CommandNode::build(std::list<Token>::iterator& it) {
|
||||
auto cmd_name = Mandatory<TokenNode<TokenType::WORD>>(it);
|
||||
auto cmd_opts = Optional<TokenNode<TokenType::FLAG>>(it);
|
||||
ArgumentVec argv;
|
||||
auto cmd_argument = Optional<CommandArgumentNode>(it);
|
||||
ArgumentVec argv;
|
||||
|
||||
while (cmd_argument) {
|
||||
argv.push_back(cmd_argument);
|
||||
cmd_argument = Optional<CommandArgumentNode>(it);
|
||||
@ -69,7 +74,6 @@ std::shared_ptr<CommandNode> CommandNode::build(std::list<Token>::iterator& it)
|
||||
return std::shared_ptr<CommandNode>(new CommandNode(cmd_name, cmd_opts, argv, cmd_redr));
|
||||
}
|
||||
|
||||
|
||||
std::shared_ptr<PipeLineNode> PipeLineNode::build(std::list<Token>::iterator& it) {
|
||||
Mandatory<TokenNode<TokenType::PIPE>>(it);
|
||||
auto l_cmd = Mandatory<CommandNode>(it);
|
||||
|
||||
@ -12,8 +12,7 @@ void CommandNode::execute(std::istream& in, std::ostream& out) {
|
||||
std::ofstream f_o;
|
||||
if (redirects) {
|
||||
if (redirects->goutput()) {
|
||||
if(redirects->gappend())
|
||||
f_o = util::output_append(*(redirects->goutput()));
|
||||
if (redirects->gappend()) f_o = util::output_append(*(redirects->goutput()));
|
||||
else
|
||||
f_o = util::output(*(redirects->goutput()));
|
||||
o = &f_o;
|
||||
@ -24,14 +23,12 @@ void CommandNode::execute(std::istream& in, std::ostream& out) {
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// TODO: Error when input/output was overwritten and in and out weren't cin/cout
|
||||
|
||||
auto& cmd_registry = CommandRegistry::instance();
|
||||
cmd_registry[cmd_name].execute(flag, args, *i, *o);
|
||||
}
|
||||
|
||||
|
||||
void PipeLineNode::execute(std::istream& in, std::ostream& out) {
|
||||
std::stringstream buffer;
|
||||
if (r_command) {
|
||||
|
||||
@ -5,9 +5,7 @@ namespace fsh {
|
||||
|
||||
void CommandNode::print(int indent) {
|
||||
std::string t = "";
|
||||
for(int i=0; i<indent;i++) {
|
||||
t+="\t";
|
||||
}
|
||||
for (int i = 0; i < indent; i++) { t += "\t"; }
|
||||
std::cout << t << gname() << "\n";
|
||||
std::cout << t << "\tCommand_Name = " << cmd_name << "\n";
|
||||
if (flag) std::cout << t << "\tFlagOpt = " << *flag << "\n";
|
||||
@ -23,9 +21,7 @@ void CommandNode::print(int indent) {
|
||||
|
||||
void PipeLineNode::print(int indent) {
|
||||
std::string t = "";
|
||||
for(int i=0; i<indent;i++) {
|
||||
t+="\t";
|
||||
}
|
||||
for (int i = 0; i < indent; i++) { t += "\t"; }
|
||||
std::cout << t << gname() << "\n";
|
||||
std::cout << "left_cmd:\n";
|
||||
if (l_command) l_command->print(indent + 1);
|
||||
@ -35,9 +31,7 @@ void PipeLineNode::print(int indent) {
|
||||
|
||||
void CommandLineNode::print(int indent) {
|
||||
std::string t = "";
|
||||
for(int i=0; i<indent;i++) {
|
||||
t+="\t";
|
||||
}
|
||||
for (int i = 0; i < indent; i++) { t += "\t"; }
|
||||
std::cout << t << gname() << "\n";
|
||||
if (command) command->print(indent + 1);
|
||||
if (pipe) pipe->print(indent + 1);
|
||||
|
||||
@ -8,25 +8,17 @@ void ArgInput::svalue(const std::string& val, const Lexer::TokenType& type) {
|
||||
str = std::stringstream(txt);
|
||||
} else {
|
||||
file = std::ifstream(txt, std::ios::in);
|
||||
if(!*file) {
|
||||
throw std::runtime_error("Failed to open file");
|
||||
if (!*file) { throw std::runtime_error("Failed to open file"); }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
std::istream& ArgInput::gvalue() {
|
||||
if(str)
|
||||
return *str;
|
||||
if (str) return *str;
|
||||
return *file;
|
||||
}
|
||||
|
||||
|
||||
|
||||
void ArgFactory::parse(ArgManager& manager, ArgNodes& vec, FlagNode flag) {
|
||||
if(flag) {
|
||||
parse_flag(manager, flag);
|
||||
}
|
||||
if (flag) { parse_flag(manager, flag); }
|
||||
unsigned int i = 0;
|
||||
for (const auto& arg : vec) {
|
||||
std::shared_ptr<_Argument> a;
|
||||
@ -36,7 +28,6 @@ void ArgFactory::parse(ArgManager& manager, ArgNodes& vec, FlagNode flag) {
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void ArgFactory::parse_flag(ArgManager& manager, FlagNode flag) {
|
||||
const std::string f = *flag;
|
||||
int f_sz = f.size();
|
||||
@ -56,14 +47,14 @@ void ArgFactory::parse_flag(ArgManager& manager, FlagNode flag) {
|
||||
throw std::invalid_argument("Invalid flag");
|
||||
}
|
||||
|
||||
|
||||
std::shared_ptr<_Argument> ArgFactory::build_arg(BuildFunc build_func, const std::string& str) {
|
||||
auto arg = build_func();
|
||||
arg->svalue(str);
|
||||
return arg;
|
||||
}
|
||||
|
||||
std::shared_ptr<_Argument> ArgFactory::build_arg(BuildFunc build_func, std::shared_ptr<CommandArgumentNode> cmd_arg) {
|
||||
std::shared_ptr<_Argument> ArgFactory::build_arg(BuildFunc build_func,
|
||||
std::shared_ptr<CommandArgumentNode> cmd_arg) {
|
||||
auto arg = build_func();
|
||||
arg->svalue(cmd_arg);
|
||||
return arg;
|
||||
|
||||
@ -7,19 +7,16 @@ namespace fsh {
|
||||
ArgManager args;
|
||||
arg_factory.parse(args, vec, flag);
|
||||
if (arg_factory.ghas_input()) {
|
||||
|
||||
if (&args.get_input(0) != &util::cin && &in != &util::cin) {
|
||||
throw std::runtime_error("Tried to set input to command twice");
|
||||
}
|
||||
if(&args.get_input(0) != &util::cin)
|
||||
run(args.get_input(0), out, args);
|
||||
if (&args.get_input(0) != &util::cin) run(args.get_input(0), out, args);
|
||||
else
|
||||
run(in, out, args);
|
||||
}
|
||||
else {
|
||||
} else {
|
||||
run(in, out, args);
|
||||
}
|
||||
in.clear();
|
||||
}
|
||||
|
||||
}
|
||||
} // namespace fsh
|
||||
@ -23,9 +23,7 @@ namespace fsh {
|
||||
std::string line;
|
||||
std::getline(util::cin, line);
|
||||
run_line(line, util::cin, std::cout);
|
||||
} catch(const std::exception& e) {
|
||||
std::cerr << e.what() << "\n";
|
||||
}
|
||||
} catch (const std::exception& e) { std::cerr << e.what() << "\n"; }
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -9,14 +9,10 @@ namespace fsh {
|
||||
|
||||
Token LexerStateMachine::run() {
|
||||
State state = State::START;
|
||||
while (state != State::END) {
|
||||
state = state_handlers[state]();
|
||||
}
|
||||
while (state != State::END) { state = state_handlers[state](); }
|
||||
return { type, token };
|
||||
}
|
||||
|
||||
|
||||
|
||||
LexerStateMachine::State LexerStateMachine::handle_start() {
|
||||
const int c = parser.peek();
|
||||
|
||||
@ -48,13 +44,9 @@ LexerStateMachine::State LexerStateMachine::handle_flag_opt() {
|
||||
return word_like_handler(Lexer::TokenType::FLAG, State::FLAG_OPT);
|
||||
}
|
||||
|
||||
|
||||
|
||||
LexerStateMachine::State LexerStateMachine::handle_redirect() {
|
||||
const int c = parser.peek();
|
||||
if(c == '>') {
|
||||
token += parser.consume();
|
||||
}
|
||||
if (c == '>') { token += parser.consume(); }
|
||||
return set_type(Lexer::TokenType::RREDIRECTION);
|
||||
}
|
||||
|
||||
@ -62,7 +54,6 @@ LexerStateMachine::State LexerStateMachine::handle_potential_word() {
|
||||
return word_like_handler(Lexer::TokenType::WORD, State::POTENTIAL_WORD);
|
||||
}
|
||||
|
||||
|
||||
LexerStateMachine::State LexerStateMachine::handle_string_literal() {
|
||||
const int c = parser.peek();
|
||||
if (c == std::char_traits<char>::eof()) {
|
||||
@ -76,7 +67,6 @@ LexerStateMachine::State LexerStateMachine::handle_string_literal() {
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
LexerStateMachine::State LexerStateMachine::word_like_handler(Lexer::TokenType type, State state) {
|
||||
const int c = parser.peek();
|
||||
if (util::contains(c, "\t><| ") || c == std::char_traits<char>::eof()) {
|
||||
@ -93,20 +83,16 @@ std::list<Token> Lexer::process(std::string line) {
|
||||
Lexer parser(line);
|
||||
std::list<Token> tokens;
|
||||
tokens.push_back(parser.next_token());
|
||||
while(tokens.back().first != TokenType::END_OF_STREAM) {
|
||||
tokens.push_back(parser.next_token());
|
||||
}
|
||||
while (tokens.back().first != TokenType::END_OF_STREAM) { tokens.push_back(parser.next_token()); }
|
||||
return tokens;
|
||||
}
|
||||
|
||||
int Lexer::peek() {
|
||||
if(idx < workspace.size())
|
||||
return workspace[idx];
|
||||
if (idx < workspace.size()) return workspace[idx];
|
||||
return std::char_traits<char>::eof();
|
||||
}
|
||||
int Lexer::consume() {
|
||||
if(idx < workspace.size())
|
||||
return workspace[idx++];
|
||||
if (idx < workspace.size()) return workspace[idx++];
|
||||
return std::char_traits<char>::eof();
|
||||
}
|
||||
|
||||
|
||||
@ -5,7 +5,6 @@
|
||||
#include "fsh.hpp"
|
||||
#include "util/input.hpp"
|
||||
|
||||
|
||||
int main() {
|
||||
fsh::util::prepTerminal();
|
||||
fsh::fsh::instance().run();
|
||||
|
||||
@ -11,22 +11,16 @@
|
||||
|
||||
#endif
|
||||
|
||||
|
||||
#include <iostream>
|
||||
#include "util/input.hpp"
|
||||
|
||||
|
||||
namespace fsh::util
|
||||
{
|
||||
namespace fsh::util {
|
||||
|
||||
#ifndef _WIN32
|
||||
struct termios prev_attr;
|
||||
void resetTerminal() {
|
||||
tcsetattr(STDIN_FILENO, TCSANOW, &prev_attr);
|
||||
}
|
||||
void resetTerminal() { tcsetattr(STDIN_FILENO, TCSANOW, &prev_attr); }
|
||||
#endif
|
||||
|
||||
|
||||
void prepTerminal() {
|
||||
|
||||
#ifdef _WIN32
|
||||
@ -38,9 +32,7 @@ void prepTerminal() {
|
||||
strelica pod UNIX sistemima */
|
||||
|
||||
struct termios tattr;
|
||||
if(!isatty(STDIN_FILENO)) {
|
||||
return;
|
||||
}
|
||||
if (!isatty(STDIN_FILENO)) { return; }
|
||||
|
||||
tcgetattr(STDIN_FILENO, &prev_attr);
|
||||
atexit(resetTerminal);
|
||||
@ -52,7 +44,6 @@ void prepTerminal() {
|
||||
tcsetattr(STDIN_FILENO, TCSAFLUSH, &tattr);
|
||||
std::cout << "\e[?25h";
|
||||
#endif
|
||||
|
||||
}
|
||||
|
||||
#ifndef _WIN32
|
||||
@ -60,9 +51,7 @@ void prepTerminal() {
|
||||
int CursorIStreamBuffer::underflow() {
|
||||
if (gptr() < egptr()) return traits_type::to_int_type(*gptr());
|
||||
|
||||
if(read_chars >= input.size()) {
|
||||
readInputToNewLine();
|
||||
}
|
||||
if (read_chars >= input.size()) { readInputToNewLine(); }
|
||||
|
||||
int n = input.copy(buffer, sizeof(buffer) - 1, read_chars);
|
||||
read_chars += n;
|
||||
@ -74,12 +63,9 @@ int CursorIStreamBuffer::underflow() {
|
||||
CursorIStreamBuffer::EscapeSequence CursorIStreamBuffer::testForEscapeCodes(char chr) {
|
||||
if (valid == 2) {
|
||||
valid = 0;
|
||||
switch (chr)
|
||||
{
|
||||
case 'C':
|
||||
return RIGHT_ARROW;
|
||||
case 'D':
|
||||
return LEFT_ARROW;
|
||||
switch (chr) {
|
||||
case 'C': return RIGHT_ARROW;
|
||||
case 'D': return LEFT_ARROW;
|
||||
}
|
||||
}
|
||||
if (chr == '\033') {
|
||||
@ -94,7 +80,6 @@ CursorIStreamBuffer::EscapeSequence CursorIStreamBuffer::testForEscapeCodes(char
|
||||
return FAILED;
|
||||
}
|
||||
|
||||
|
||||
int CursorIStreamBuffer::readInputToNewLine() {
|
||||
read_chars = 0;
|
||||
cursor_pos = 0;
|
||||
@ -109,32 +94,27 @@ int CursorIStreamBuffer::readInputToNewLine(){
|
||||
std::cout << "\e[D\e[P";
|
||||
input.erase(--cursor_pos, 1);
|
||||
}
|
||||
}
|
||||
else if(e>0) {
|
||||
} else if (e > 0) {
|
||||
if (e == RIGHT_ARROW) {
|
||||
if (cursor_pos < input.size()) {
|
||||
cursor_pos++;
|
||||
std::cout << "\e[C";
|
||||
}
|
||||
}
|
||||
else if(e==LEFT_ARROW) {
|
||||
} else if (e == LEFT_ARROW) {
|
||||
if (cursor_pos > 0) {
|
||||
cursor_pos--;
|
||||
std::cout << "\e[D";
|
||||
}
|
||||
}
|
||||
continue;
|
||||
}
|
||||
else if(c == '\n') {
|
||||
} else if (c == '\n') {
|
||||
std::cout << '\n';
|
||||
input.push_back(c);
|
||||
return 0;
|
||||
}
|
||||
else{
|
||||
} else {
|
||||
input.insert(cursor_pos++, 1, c);
|
||||
std::cout << "\033[@" << c;
|
||||
}
|
||||
|
||||
}
|
||||
input.push_back(0);
|
||||
return 0;
|
||||
@ -145,5 +125,4 @@ CursorIStream cin;
|
||||
std::istream& cin = std::cin;
|
||||
#endif
|
||||
|
||||
|
||||
}
|
||||
@ -2,16 +2,7 @@
|
||||
|
||||
namespace fsh::util {
|
||||
|
||||
const char* tokens[] = {
|
||||
"WHITESPACE",
|
||||
"WORD",
|
||||
"STRING_LITERAL",
|
||||
"OPT",
|
||||
"FLAG",
|
||||
"LREDIRECTION",
|
||||
"RREDIRECTION",
|
||||
"PIPE",
|
||||
"END_OF_STREAM"
|
||||
};
|
||||
const char* tokens[] = { "WHITESPACE", "WORD", "STRING_LITERAL", "OPT", "FLAG",
|
||||
"LREDIRECTION", "RREDIRECTION", "PIPE", "END_OF_STREAM" };
|
||||
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user