|
@@ -7,6 +7,7 @@
|
|
|
#include "Formatter.h"
|
|
|
#include "AST.h"
|
|
|
#include "Parser.h"
|
|
|
+#include <AK/Hex.h>
|
|
|
#include <AK/ScopedValueRollback.h>
|
|
|
#include <AK/TemporaryChange.h>
|
|
|
|
|
@@ -36,12 +37,14 @@ String Formatter::format()
|
|
|
|
|
|
node->visit(*this);
|
|
|
|
|
|
- auto string = m_builder.string_view();
|
|
|
+ VERIFY(m_builders.size() == 1);
|
|
|
+
|
|
|
+ auto string = current_builder().string_view();
|
|
|
|
|
|
if (!string.ends_with(" "))
|
|
|
- m_builder.append(m_trivia);
|
|
|
+ current_builder().append(m_trivia);
|
|
|
|
|
|
- return m_builder.to_string();
|
|
|
+ return current_builder().to_string();
|
|
|
}
|
|
|
|
|
|
void Formatter::with_added_indent(int indent, Function<void()> callback)
|
|
@@ -63,6 +66,13 @@ void Formatter::in_new_block(Function<void()> callback)
|
|
|
current_builder().append('}');
|
|
|
}
|
|
|
|
|
|
+String Formatter::in_new_builder(Function<void()> callback, StringBuilder new_builder)
|
|
|
+{
|
|
|
+ m_builders.append(move(new_builder));
|
|
|
+ callback();
|
|
|
+ return m_builders.take_last().to_string();
|
|
|
+}
|
|
|
+
|
|
|
void Formatter::test_and_update_output_cursor(const AST::Node* node)
|
|
|
{
|
|
|
if (!node)
|
|
@@ -96,9 +106,17 @@ void Formatter::will_visit(const AST::Node* node)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-void Formatter::insert_separator()
|
|
|
+void Formatter::insert_separator(bool escaped)
|
|
|
{
|
|
|
+ if (escaped)
|
|
|
+ current_builder().append('\\');
|
|
|
current_builder().append('\n');
|
|
|
+ if (!escaped && !m_heredocs_to_append_after_sequence.is_empty()) {
|
|
|
+ for (auto& entry : m_heredocs_to_append_after_sequence) {
|
|
|
+ current_builder().append(entry);
|
|
|
+ }
|
|
|
+ m_heredocs_to_append_after_sequence.clear();
|
|
|
+ }
|
|
|
insert_indent();
|
|
|
}
|
|
|
|
|
@@ -127,8 +145,8 @@ void Formatter::visit(const AST::And* node)
|
|
|
with_added_indent(should_indent ? 1 : 0, [&] {
|
|
|
node->left()->visit(*this);
|
|
|
|
|
|
- current_builder().append(" \\");
|
|
|
- insert_separator();
|
|
|
+ current_builder().append(' ');
|
|
|
+ insert_separator(true);
|
|
|
current_builder().append("&& ");
|
|
|
|
|
|
node->right()->visit(*this);
|
|
@@ -269,14 +287,17 @@ void Formatter::visit(const AST::DoubleQuotedString* node)
|
|
|
{
|
|
|
will_visit(node);
|
|
|
test_and_update_output_cursor(node);
|
|
|
- current_builder().append("\"");
|
|
|
+ auto not_in_heredoc = m_parent_node->kind() != AST::Node::Kind::Heredoc;
|
|
|
+ if (not_in_heredoc)
|
|
|
+ current_builder().append("\"");
|
|
|
|
|
|
TemporaryChange quotes { m_options.in_double_quotes, true };
|
|
|
TemporaryChange<const AST::Node*> parent { m_parent_node, node };
|
|
|
|
|
|
NodeVisitor::visit(node);
|
|
|
|
|
|
- current_builder().append("\"");
|
|
|
+ if (not_in_heredoc)
|
|
|
+ current_builder().append("\"");
|
|
|
visited(node);
|
|
|
}
|
|
|
|
|
@@ -355,6 +376,39 @@ void Formatter::visit(const AST::Glob* node)
|
|
|
visited(node);
|
|
|
}
|
|
|
|
|
|
+void Formatter::visit(const AST::Heredoc* node)
|
|
|
+{
|
|
|
+ will_visit(node);
|
|
|
+ test_and_update_output_cursor(node);
|
|
|
+
|
|
|
+ current_builder().append("<<");
|
|
|
+ if (node->deindent())
|
|
|
+ current_builder().append('~');
|
|
|
+ else
|
|
|
+ current_builder().append('-');
|
|
|
+
|
|
|
+ if (node->allow_interpolation())
|
|
|
+ current_builder().appendff("{}", node->end());
|
|
|
+ else
|
|
|
+ current_builder().appendff("'{}'", node->end());
|
|
|
+
|
|
|
+ auto content = in_new_builder([&] {
|
|
|
+ if (!node->contents())
|
|
|
+ return;
|
|
|
+
|
|
|
+ TemporaryChange<const AST::Node*> parent { m_parent_node, node };
|
|
|
+ TemporaryChange heredoc { m_options.in_heredoc, true };
|
|
|
+
|
|
|
+ auto& contents = *node->contents();
|
|
|
+ contents.visit(*this);
|
|
|
+ current_builder().appendff("\n{}\n", node->end());
|
|
|
+ });
|
|
|
+
|
|
|
+ m_heredocs_to_append_after_sequence.append(move(content));
|
|
|
+
|
|
|
+ visited(node);
|
|
|
+}
|
|
|
+
|
|
|
void Formatter::visit(const AST::HistoryEvent* node)
|
|
|
{
|
|
|
will_visit(node);
|
|
@@ -567,8 +621,8 @@ void Formatter::visit(const AST::Or* node)
|
|
|
with_added_indent(should_indent ? 1 : 0, [&] {
|
|
|
node->left()->visit(*this);
|
|
|
|
|
|
- current_builder().append(" \\");
|
|
|
- insert_separator();
|
|
|
+ current_builder().append(" ");
|
|
|
+ insert_separator(true);
|
|
|
current_builder().append("|| ");
|
|
|
|
|
|
node->right()->visit(*this);
|
|
@@ -584,10 +638,10 @@ void Formatter::visit(const AST::Pipe* node)
|
|
|
TemporaryChange<const AST::Node*> parent { m_parent_node, node };
|
|
|
|
|
|
node->left()->visit(*this);
|
|
|
- current_builder().append(" \\");
|
|
|
+ current_builder().append(" ");
|
|
|
|
|
|
with_added_indent(should_indent ? 1 : 0, [&] {
|
|
|
- insert_separator();
|
|
|
+ insert_separator(true);
|
|
|
current_builder().append("| ");
|
|
|
|
|
|
node->right()->visit(*this);
|
|
@@ -720,10 +774,10 @@ void Formatter::visit(const AST::StringLiteral* node)
|
|
|
{
|
|
|
will_visit(node);
|
|
|
test_and_update_output_cursor(node);
|
|
|
- if (!m_options.in_double_quotes)
|
|
|
+ if (!m_options.in_double_quotes && !m_options.in_heredoc)
|
|
|
current_builder().append("'");
|
|
|
|
|
|
- if (m_options.in_double_quotes) {
|
|
|
+ if (m_options.in_double_quotes && !m_options.in_heredoc) {
|
|
|
for (auto ch : node->text()) {
|
|
|
switch (ch) {
|
|
|
case '"':
|
|
@@ -761,7 +815,7 @@ void Formatter::visit(const AST::StringLiteral* node)
|
|
|
current_builder().append(node->text());
|
|
|
}
|
|
|
|
|
|
- if (!m_options.in_double_quotes)
|
|
|
+ if (!m_options.in_double_quotes && !m_options.in_heredoc)
|
|
|
current_builder().append("'");
|
|
|
visited(node);
|
|
|
}
|