| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2020, Andreas Kling <kling@serenityos.org> | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  * modification, are permitted provided that the following conditions are met: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 1. Redistributions of source code must retain the above copyright notice, this | 
					
						
							|  |  |  |  *    list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 2. Redistributions in binary form must reproduce the above copyright notice, | 
					
						
							|  |  |  |  *    this list of conditions and the following disclaimer in the documentation | 
					
						
							|  |  |  |  *    and/or other materials provided with the distribution. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
					
						
							|  |  |  |  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
					
						
							|  |  |  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
					
						
							|  |  |  |  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
					
						
							|  |  |  |  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
					
						
							|  |  |  |  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
					
						
							|  |  |  |  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
					
						
							|  |  |  |  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
					
						
							|  |  |  |  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
					
						
							|  |  |  |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 10:51:41 +01:00
										 |  |  | #include <AK/ByteBuffer.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | #include <AK/NonnullOwnPtr.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-12 10:51:41 +01:00
										 |  |  | #include <LibCore/ArgsParser.h>
 | 
					
						
							|  |  |  | #include <LibCore/File.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | #include <LibJS/AST.h>
 | 
					
						
							|  |  |  | #include <LibJS/Interpreter.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | #include <LibJS/Parser.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  | #include <LibJS/Runtime/Array.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-31 18:43:45 +01:00
										 |  |  | #include <LibJS/Runtime/Date.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-02 09:54:15 +02:00
										 |  |  | #include <LibJS/Runtime/Error.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-26 14:54:43 +01:00
										 |  |  | #include <LibJS/Runtime/Function.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-01 18:53:28 +02:00
										 |  |  | #include <LibJS/Runtime/GlobalObject.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 14:20:30 +01:00
										 |  |  | #include <LibJS/Runtime/Object.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/PrimitiveString.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-02 19:32:21 +02:00
										 |  |  | #include <LibJS/Runtime/Shape.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 14:20:30 +01:00
										 |  |  | #include <LibJS/Runtime/Value.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:57 +02:00
										 |  |  | #include <LibLine/Editor.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:10 +04:30
										 |  |  | #include <signal.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 11:52:40 -04:00
										 |  |  | Vector<String> repl_statements; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | class ReplObject : public JS::GlobalObject { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     ReplObject(); | 
					
						
							| 
									
										
										
										
											2020-04-18 13:18:06 +02:00
										 |  |  |     virtual void initialize() override; | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |     virtual ~ReplObject() override; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-12 22:49:04 -07:00
										 |  |  |     static JS::Value load_file(JS::Interpreter&); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | private: | 
					
						
							|  |  |  |     virtual const char* class_name() const override { return "ReplObject"; } | 
					
						
							|  |  |  |     static JS::Value exit_interpreter(JS::Interpreter&); | 
					
						
							|  |  |  |     static JS::Value repl_help(JS::Interpreter&); | 
					
						
							| 
									
										
										
										
											2020-04-02 11:52:40 -04:00
										 |  |  |     static JS::Value save_to_file(JS::Interpreter&); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  | static bool s_dump_ast = false; | 
					
						
							|  |  |  | static bool s_print_last_result = false; | 
					
						
							|  |  |  | static OwnPtr<Line::Editor> s_editor; | 
					
						
							|  |  |  | static int s_repl_line_level = 0; | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  | static String prompt_for_level(int level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     static StringBuilder prompt_builder; | 
					
						
							|  |  |  |     prompt_builder.clear(); | 
					
						
							|  |  |  |     prompt_builder.append("> "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto i = 0; i < level; ++i) | 
					
						
							|  |  |  |         prompt_builder.append("    "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return prompt_builder.build(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | String read_next_piece() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     StringBuilder piece; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     do { | 
					
						
							| 
									
										
										
										
											2020-03-30 22:44:23 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         String line = s_editor->get_line(prompt_for_level(s_repl_line_level)); | 
					
						
							|  |  |  |         s_editor->add_to_history(line); | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         piece.append(line); | 
					
						
							|  |  |  |         auto lexer = JS::Lexer(line); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (JS::Token token = lexer.next(); token.type() != JS::TokenType::Eof; token = lexer.next()) { | 
					
						
							|  |  |  |             switch (token.type()) { | 
					
						
							|  |  |  |             case JS::TokenType::BracketOpen: | 
					
						
							|  |  |  |             case JS::TokenType::CurlyOpen: | 
					
						
							|  |  |  |             case JS::TokenType::ParenOpen: | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |                 s_repl_line_level++; | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             case JS::TokenType::BracketClose: | 
					
						
							|  |  |  |             case JS::TokenType::CurlyClose: | 
					
						
							|  |  |  |             case JS::TokenType::ParenClose: | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |                 s_repl_line_level--; | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |     } while (s_repl_line_level > 0); | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return piece.to_string(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  | static void print_value(JS::Value value, HashTable<JS::Object*>& seen_objects); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  | static void print_array(const JS::Array& array, HashTable<JS::Object*>& seen_objects) | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     fputs("[ ", stdout); | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |     for (size_t i = 0; i < array.elements().size(); ++i) { | 
					
						
							|  |  |  |         print_value(array.elements()[i], seen_objects); | 
					
						
							|  |  |  |         if (i != array.elements().size() - 1) | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |             fputs(", ", stdout); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     fputs(" ]", stdout); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  | static void print_object(const JS::Object& object, HashTable<JS::Object*>& seen_objects) | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     fputs("{ ", stdout); | 
					
						
							| 
									
										
										
										
											2020-04-06 16:53:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (size_t i = 0; i < object.elements().size(); ++i) { | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         if (object.elements()[i].is_empty()) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2020-04-06 16:53:02 +02:00
										 |  |  |         printf("\"\033[33;1m%zu\033[0m\": ", i); | 
					
						
							|  |  |  |         print_value(object.elements()[i], seen_objects); | 
					
						
							|  |  |  |         if (i != object.elements().size() - 1) | 
					
						
							|  |  |  |             fputs(", ", stdout); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!object.elements().is_empty() && object.shape().property_count()) | 
					
						
							|  |  |  |         fputs(", ", stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |     size_t index = 0; | 
					
						
							| 
									
										
										
										
											2020-04-28 19:19:31 -07:00
										 |  |  |     for (auto& it : object.shape().property_table_ordered()) { | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |         printf("\"\033[33;1m%s\033[0m\": ", it.key.characters()); | 
					
						
							| 
									
										
										
										
											2020-04-02 19:32:21 +02:00
										 |  |  |         print_value(object.get_direct(it.value.offset), seen_objects); | 
					
						
							| 
									
										
										
										
											2020-04-06 16:53:02 +02:00
										 |  |  |         if (index != object.shape().property_count() - 1) | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |             fputs(", ", stdout); | 
					
						
							|  |  |  |         ++index; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     fputs(" }", stdout); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  | static void print_function(const JS::Object& function, HashTable<JS::Object*>&) | 
					
						
							| 
									
										
										
										
											2020-03-26 14:54:43 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |     printf("\033[34;1m[%s]\033[0m", function.class_name()); | 
					
						
							| 
									
										
										
										
											2020-03-26 14:54:43 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  | static void print_date(const JS::Object& date, HashTable<JS::Object*>&) | 
					
						
							| 
									
										
										
										
											2020-03-31 18:43:45 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |     printf("\033[34;1mDate %s\033[0m", static_cast<const JS::Date&>(date).string().characters()); | 
					
						
							| 
									
										
										
										
											2020-03-31 18:43:45 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 09:54:15 +02:00
										 |  |  | static void print_error(const JS::Object& object, HashTable<JS::Object*>&) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto& error = static_cast<const JS::Error&>(object); | 
					
						
							| 
									
										
										
										
											2020-04-02 14:18:19 +01:00
										 |  |  |     printf("\033[34;1m[%s]\033[0m", error.name().characters()); | 
					
						
							|  |  |  |     if (!error.message().is_empty()) | 
					
						
							|  |  |  |         printf(": %s", error.message().characters()); | 
					
						
							| 
									
										
										
										
											2020-04-02 09:54:15 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  | void print_value(JS::Value value, HashTable<JS::Object*>& seen_objects) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-08 17:38:09 +01:00
										 |  |  |     if (value.is_empty()) { | 
					
						
							|  |  |  |         printf("\033[34;1m<empty>\033[0m"); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |     if (value.is_object()) { | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |         if (seen_objects.contains(&value.as_object())) { | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |             // FIXME: Maybe we should only do this for circular references,
 | 
					
						
							|  |  |  |             //        not for all reoccurring objects.
 | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |             printf("<already printed Object %p>", &value.as_object()); | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |         seen_objects.set(&value.as_object()); | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (value.is_array()) | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |         return print_array(static_cast<const JS::Array&>(value.as_object()), seen_objects); | 
					
						
							| 
									
										
										
										
											2020-04-01 18:53:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 18:43:45 +01:00
										 |  |  |     if (value.is_object()) { | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |         auto& object = value.as_object(); | 
					
						
							|  |  |  |         if (object.is_function()) | 
					
						
							| 
									
										
										
										
											2020-03-31 18:43:45 +01:00
										 |  |  |             return print_function(object, seen_objects); | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |         if (object.is_date()) | 
					
						
							| 
									
										
										
										
											2020-03-31 18:43:45 +01:00
										 |  |  |             return print_date(object, seen_objects); | 
					
						
							| 
									
										
										
										
											2020-04-02 09:54:15 +02:00
										 |  |  |         if (object.is_error()) | 
					
						
							|  |  |  |             return print_error(object, seen_objects); | 
					
						
							| 
									
										
										
										
											2020-03-31 18:43:45 +01:00
										 |  |  |         return print_object(object, seen_objects); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (value.is_string()) | 
					
						
							| 
									
										
										
										
											2020-04-15 10:11:48 +02:00
										 |  |  |         printf("\033[32;1m"); | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |     else if (value.is_number()) | 
					
						
							|  |  |  |         printf("\033[35;1m"); | 
					
						
							|  |  |  |     else if (value.is_boolean()) | 
					
						
							| 
									
										
										
										
											2020-04-15 10:11:48 +02:00
										 |  |  |         printf("\033[33;1m"); | 
					
						
							| 
									
										
										
										
											2020-03-26 14:54:43 +01:00
										 |  |  |     else if (value.is_null()) | 
					
						
							|  |  |  |         printf("\033[33;1m"); | 
					
						
							|  |  |  |     else if (value.is_undefined()) | 
					
						
							| 
									
										
										
										
											2020-03-26 12:26:11 +01:00
										 |  |  |         printf("\033[34;1m"); | 
					
						
							|  |  |  |     if (value.is_string()) | 
					
						
							|  |  |  |         putchar('"'); | 
					
						
							|  |  |  |     printf("%s", value.to_string().characters()); | 
					
						
							|  |  |  |     if (value.is_string()) | 
					
						
							|  |  |  |         putchar('"'); | 
					
						
							|  |  |  |     printf("\033[0m"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void print(JS::Value value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     HashTable<JS::Object*> seen_objects; | 
					
						
							|  |  |  |     print_value(value, seen_objects); | 
					
						
							|  |  |  |     putchar('\n'); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | bool file_has_shebang(AK::ByteBuffer file_contents) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (file_contents.size() >= 2 && file_contents[0] == '#' && file_contents[1] == '!') | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | StringView strip_shebang(AK::ByteBuffer file_contents) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     size_t i = 0; | 
					
						
							|  |  |  |     for (i = 2; i < file_contents.size(); ++i) { | 
					
						
							|  |  |  |         if (file_contents[i] == '\n') | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return StringView((const char*)file_contents.data() + i, file_contents.size() - i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 11:52:40 -04:00
										 |  |  | bool write_to_file(const StringView& path) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int fd = open_with_path_length(path.characters_without_null_termination(), path.length(), O_WRONLY | O_CREAT | O_TRUNC, 0666); | 
					
						
							|  |  |  |     for (size_t i = 0; i < repl_statements.size(); i++) { | 
					
						
							|  |  |  |         auto line = repl_statements[i]; | 
					
						
							|  |  |  |         if (line.length() && i != repl_statements.size() - 1) { | 
					
						
							|  |  |  |             ssize_t nwritten = write(fd, line.characters(), line.length()); | 
					
						
							|  |  |  |             if (nwritten < 0) { | 
					
						
							|  |  |  |                 close(fd); | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (i != repl_statements.size() - 1) { | 
					
						
							|  |  |  |             char ch = '\n'; | 
					
						
							|  |  |  |             ssize_t nwritten = write(fd, &ch, 1); | 
					
						
							|  |  |  |             if (nwritten != 1) { | 
					
						
							|  |  |  |                 perror("write"); | 
					
						
							|  |  |  |                 close(fd); | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     close(fd); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | ReplObject::ReplObject() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-18 13:18:06 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ReplObject::initialize() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     GlobalObject::initialize(); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |     put_native_function("exit", exit_interpreter); | 
					
						
							|  |  |  |     put_native_function("help", repl_help); | 
					
						
							| 
									
										
										
										
											2020-04-04 14:13:53 +01:00
										 |  |  |     put_native_function("load", load_file, 1); | 
					
						
							|  |  |  |     put_native_function("save", save_to_file, 1); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ReplObject::~ReplObject() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-13 14:04:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 11:52:40 -04:00
										 |  |  | JS::Value ReplObject::save_to_file(JS::Interpreter& interpreter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!interpreter.argument_count()) | 
					
						
							|  |  |  |         return JS::Value(false); | 
					
						
							|  |  |  |     String save_path = interpreter.argument(0).to_string(); | 
					
						
							|  |  |  |     StringView path = StringView(save_path.characters()); | 
					
						
							|  |  |  |     if (write_to_file(path)) { | 
					
						
							|  |  |  |         return JS::Value(true); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return JS::Value(false); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-13 14:04:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | JS::Value ReplObject::exit_interpreter(JS::Interpreter& interpreter) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-01 22:38:59 +02:00
										 |  |  |     if (!interpreter.argument_count()) | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |         exit(0); | 
					
						
							| 
									
										
										
										
											2020-04-01 22:38:59 +02:00
										 |  |  |     int exit_code = interpreter.argument(0).to_number().as_double(); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |     exit(exit_code); | 
					
						
							|  |  |  |     return JS::js_undefined(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-13 14:04:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 21:08:54 +01:00
										 |  |  | JS::Value ReplObject::repl_help(JS::Interpreter&) | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-13 21:08:54 +01:00
										 |  |  |     printf("REPL commands:\n"); | 
					
						
							|  |  |  |     printf("    exit(code): exit the REPL with specified code. Defaults to 0.\n"); | 
					
						
							|  |  |  |     printf("    help(): display this menu\n"); | 
					
						
							|  |  |  |     printf("    load(files): Accepts file names as params to load into running session. For example load(\"js/1.js\", \"js/2.js\", \"js/3.js\")\n"); | 
					
						
							|  |  |  |     return JS::js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | JS::Value ReplObject::load_file(JS::Interpreter& interpreter) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-01 22:38:59 +02:00
										 |  |  |     if (!interpreter.argument_count()) | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |         return JS::Value(false); | 
					
						
							| 
									
										
										
										
											2020-04-01 22:38:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (auto& file : interpreter.call_frame().arguments) { | 
					
						
							| 
									
										
										
										
											2020-04-29 12:35:39 +02:00
										 |  |  |         String file_name = file.as_string().string(); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |         auto js_file = Core::File::construct(file_name); | 
					
						
							|  |  |  |         if (!js_file->open(Core::IODevice::ReadOnly)) { | 
					
						
							|  |  |  |             fprintf(stderr, "Failed to open %s: %s\n", file_name.characters(), js_file->error_string()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         auto file_contents = js_file->read_all(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         StringView source; | 
					
						
							|  |  |  |         if (file_has_shebang(file_contents)) { | 
					
						
							|  |  |  |             source = strip_shebang(file_contents); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             source = file_contents; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-13 02:05:21 +02:00
										 |  |  |         auto parser = JS::Parser(JS::Lexer(source)); | 
					
						
							|  |  |  |         auto program = parser.parse_program(); | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         if (s_dump_ast) | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |             program->dump(0); | 
					
						
							| 
									
										
										
										
											2020-04-13 02:05:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (parser.has_errors()) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-04 23:45:13 +02:00
										 |  |  |         interpreter.run(*program); | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         if (s_print_last_result) | 
					
						
							| 
									
										
										
										
											2020-04-12 22:49:04 -07:00
										 |  |  |             print(interpreter.last_value()); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     return JS::Value(true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  | void repl(JS::Interpreter& interpreter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (true) { | 
					
						
							|  |  |  |         String piece = read_next_piece(); | 
					
						
							|  |  |  |         if (piece.is_empty()) | 
					
						
							| 
									
										
										
										
											2020-03-31 13:29:42 +02:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2020-04-02 11:52:40 -04:00
										 |  |  |         repl_statements.append(piece); | 
					
						
							| 
									
										
										
										
											2020-04-13 02:05:21 +02:00
										 |  |  |         auto parser = JS::Parser(JS::Lexer(piece)); | 
					
						
							|  |  |  |         auto program = parser.parse_program(); | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         if (s_dump_ast) | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |             program->dump(0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 02:05:21 +02:00
										 |  |  |         if (parser.has_errors()) { | 
					
						
							|  |  |  |             printf("Parse error\n"); | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-04 23:45:13 +02:00
										 |  |  |         interpreter.run(*program); | 
					
						
							| 
									
										
										
										
											2020-04-02 09:54:15 +02:00
										 |  |  |         if (interpreter.exception()) { | 
					
						
							| 
									
										
										
										
											2020-04-02 15:25:08 +02:00
										 |  |  |             printf("Uncaught exception: "); | 
					
						
							| 
									
										
										
										
											2020-04-02 09:54:15 +02:00
										 |  |  |             print(interpreter.exception()->value()); | 
					
						
							|  |  |  |             interpreter.clear_exception(); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2020-04-04 23:45:13 +02:00
										 |  |  |             print(interpreter.last_value()); | 
					
						
							| 
									
										
										
										
											2020-04-02 09:54:15 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-09 21:58:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-12 22:49:04 -07:00
										 |  |  | void enable_test_mode(JS::Interpreter& interpreter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     interpreter.global_object().put_native_function("load", ReplObject::load_file); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 17:59:16 +01:00
										 |  |  | static Function<void()> interrupt_interpreter; | 
					
						
							|  |  |  | void sigint_handler() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     interrupt_interpreter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 10:51:41 +01:00
										 |  |  | int main(int argc, char** argv) | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-16 19:18:46 +01:00
										 |  |  |     bool gc_on_every_allocation = false; | 
					
						
							| 
									
										
										
										
											2020-04-05 18:33:12 +04:30
										 |  |  |     bool syntax_highlight = false; | 
					
						
							| 
									
										
										
										
											2020-04-05 05:49:46 -07:00
										 |  |  |     bool test_mode = false; | 
					
						
							| 
									
										
										
										
											2020-03-12 10:51:41 +01:00
										 |  |  |     const char* script_path = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Core::ArgsParser args_parser; | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |     args_parser.add_option(s_dump_ast, "Dump the AST", "dump-ast", 'A'); | 
					
						
							|  |  |  |     args_parser.add_option(s_print_last_result, "Print last result", "print-last-result", 'l'); | 
					
						
							| 
									
										
										
										
											2020-03-16 19:18:46 +01:00
										 |  |  |     args_parser.add_option(gc_on_every_allocation, "GC on every allocation", "gc-on-every-allocation", 'g'); | 
					
						
							| 
									
										
										
										
											2020-04-05 18:33:12 +04:30
										 |  |  |     args_parser.add_option(syntax_highlight, "Enable live syntax highlighting", "syntax-highlight", 's'); | 
					
						
							| 
									
										
										
										
											2020-04-12 22:49:04 -07:00
										 |  |  |     args_parser.add_option(test_mode, "Run the interpreter with added functionality for the test harness", "test-mode", 't'); | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |     args_parser.add_positional_argument(script_path, "Path to script file", "script", Core::ArgsParser::Required::No); | 
					
						
							| 
									
										
										
										
											2020-03-12 10:51:41 +01:00
										 |  |  |     args_parser.parse(argc, argv); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 17:59:16 +01:00
										 |  |  |     OwnPtr<JS::Interpreter> interpreter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     interrupt_interpreter = [&] { | 
					
						
							|  |  |  |         auto error = JS::Error::create(interpreter->global_object(), "Error", "Received SIGINT"); | 
					
						
							|  |  |  |         interpreter->throw_exception(error); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |     if (script_path == nullptr) { | 
					
						
							| 
									
										
										
										
											2020-04-24 17:59:16 +01:00
										 |  |  |         interpreter = JS::Interpreter::create<ReplObject>(); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |         interpreter->heap().set_should_collect_on_every_allocation(gc_on_every_allocation); | 
					
						
							| 
									
										
										
										
											2020-04-12 22:49:04 -07:00
										 |  |  |         if (test_mode) | 
					
						
							|  |  |  |             enable_test_mode(*interpreter); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         s_editor = make<Line::Editor>(); | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:10 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  |         signal(SIGINT, [](int) { | 
					
						
							| 
									
										
										
										
											2020-04-29 00:31:22 +04:30
										 |  |  |             if (!s_editor->is_editing()) | 
					
						
							|  |  |  |                 sigint_handler(); | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |             s_editor->interrupted(); | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:10 +04:30
										 |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         signal(SIGWINCH, [](int) { | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |             s_editor->resized(); | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:10 +04:30
										 |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         s_editor->on_display_refresh = [syntax_highlight](Line::Editor& editor) { | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  |             auto stylize = [&](Line::Span span, Line::Style styles) { | 
					
						
							|  |  |  |                 if (syntax_highlight) | 
					
						
							|  |  |  |                     editor.stylize(span, styles); | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |             editor.strip_styles(); | 
					
						
							|  |  |  |             StringBuilder builder; | 
					
						
							|  |  |  |             builder.append({ editor.buffer().data(), editor.buffer().size() }); | 
					
						
							|  |  |  |             // FIXME: The lexer returns weird position information without this
 | 
					
						
							|  |  |  |             builder.append(" "); | 
					
						
							|  |  |  |             String str = builder.build(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |             size_t open_indents = s_repl_line_level; | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  |             JS::Lexer lexer(str, false); | 
					
						
							|  |  |  |             bool indenters_starting_line = true; | 
					
						
							|  |  |  |             for (JS::Token token = lexer.next(); token.type() != JS::TokenType::Eof; token = lexer.next()) { | 
					
						
							|  |  |  |                 auto length = token.value().length(); | 
					
						
							|  |  |  |                 auto start = token.line_column() - 2; | 
					
						
							|  |  |  |                 auto end = start + length; | 
					
						
							|  |  |  |                 if (indenters_starting_line) { | 
					
						
							|  |  |  |                     if (token.type() != JS::TokenType::ParenClose && token.type() != JS::TokenType::BracketClose && token.type() != JS::TokenType::CurlyClose) { | 
					
						
							|  |  |  |                         indenters_starting_line = false; | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         --open_indents; | 
					
						
							| 
									
										
										
										
											2020-04-05 18:33:12 +04:30
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  |                 switch (token.type()) { | 
					
						
							|  |  |  |                 case JS::TokenType::Invalid: | 
					
						
							|  |  |  |                 case JS::TokenType::Eof: | 
					
						
							|  |  |  |                     stylize({ start, end }, { Line::Style::Foreground(Line::Style::Color::Red), Line::Style::Underline }); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case JS::TokenType::NumericLiteral: | 
					
						
							|  |  |  |                     stylize({ start, end }, { Line::Style::Foreground(Line::Style::Color::Magenta) }); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case JS::TokenType::StringLiteral: | 
					
						
							| 
									
										
										
										
											2020-04-24 00:20:02 +01:00
										 |  |  |                 case JS::TokenType::TemplateLiteral: | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  |                 case JS::TokenType::RegexLiteral: | 
					
						
							|  |  |  |                 case JS::TokenType::UnterminatedStringLiteral: | 
					
						
							| 
									
										
										
										
											2020-04-16 16:37:12 +02:00
										 |  |  |                     stylize({ start, end }, { Line::Style::Foreground(Line::Style::Color::Green), Line::Style::Bold }); | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  |                     break; | 
					
						
							|  |  |  |                 case JS::TokenType::BracketClose: | 
					
						
							|  |  |  |                 case JS::TokenType::BracketOpen: | 
					
						
							|  |  |  |                 case JS::TokenType::Caret: | 
					
						
							|  |  |  |                 case JS::TokenType::Comma: | 
					
						
							|  |  |  |                 case JS::TokenType::CurlyClose: | 
					
						
							|  |  |  |                 case JS::TokenType::CurlyOpen: | 
					
						
							|  |  |  |                 case JS::TokenType::ParenClose: | 
					
						
							|  |  |  |                 case JS::TokenType::ParenOpen: | 
					
						
							|  |  |  |                 case JS::TokenType::Semicolon: | 
					
						
							|  |  |  |                 case JS::TokenType::Period: | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case JS::TokenType::Ampersand: | 
					
						
							|  |  |  |                 case JS::TokenType::AmpersandEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::Asterisk: | 
					
						
							|  |  |  |                 case JS::TokenType::AsteriskAsteriskEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::AsteriskEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::DoubleAmpersand: | 
					
						
							|  |  |  |                 case JS::TokenType::DoubleAsterisk: | 
					
						
							|  |  |  |                 case JS::TokenType::DoublePipe: | 
					
						
							|  |  |  |                 case JS::TokenType::DoubleQuestionMark: | 
					
						
							|  |  |  |                 case JS::TokenType::Equals: | 
					
						
							|  |  |  |                 case JS::TokenType::EqualsEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::EqualsEqualsEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::ExclamationMark: | 
					
						
							|  |  |  |                 case JS::TokenType::ExclamationMarkEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::ExclamationMarkEqualsEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::GreaterThan: | 
					
						
							|  |  |  |                 case JS::TokenType::GreaterThanEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::LessThan: | 
					
						
							|  |  |  |                 case JS::TokenType::LessThanEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::Minus: | 
					
						
							|  |  |  |                 case JS::TokenType::MinusEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::MinusMinus: | 
					
						
							|  |  |  |                 case JS::TokenType::Percent: | 
					
						
							|  |  |  |                 case JS::TokenType::PercentEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::Pipe: | 
					
						
							|  |  |  |                 case JS::TokenType::PipeEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::Plus: | 
					
						
							|  |  |  |                 case JS::TokenType::PlusEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::PlusPlus: | 
					
						
							|  |  |  |                 case JS::TokenType::QuestionMark: | 
					
						
							|  |  |  |                 case JS::TokenType::QuestionMarkPeriod: | 
					
						
							|  |  |  |                 case JS::TokenType::ShiftLeft: | 
					
						
							|  |  |  |                 case JS::TokenType::ShiftLeftEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::ShiftRight: | 
					
						
							|  |  |  |                 case JS::TokenType::ShiftRightEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::Slash: | 
					
						
							|  |  |  |                 case JS::TokenType::SlashEquals: | 
					
						
							|  |  |  |                 case JS::TokenType::Tilde: | 
					
						
							|  |  |  |                 case JS::TokenType::UnsignedShiftRight: | 
					
						
							|  |  |  |                 case JS::TokenType::UnsignedShiftRightEquals: | 
					
						
							|  |  |  |                     break; | 
					
						
							| 
									
										
										
										
											2020-04-16 16:37:12 +02:00
										 |  |  |                 case JS::TokenType::BoolLiteral: | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  |                 case JS::TokenType::NullLiteral: | 
					
						
							|  |  |  |                     stylize({ start, end }, { Line::Style::Foreground(Line::Style::Color::Yellow), Line::Style::Bold }); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case JS::TokenType::Class: | 
					
						
							|  |  |  |                 case JS::TokenType::Const: | 
					
						
							|  |  |  |                 case JS::TokenType::Delete: | 
					
						
							|  |  |  |                 case JS::TokenType::Function: | 
					
						
							|  |  |  |                 case JS::TokenType::In: | 
					
						
							|  |  |  |                 case JS::TokenType::Instanceof: | 
					
						
							|  |  |  |                 case JS::TokenType::Interface: | 
					
						
							|  |  |  |                 case JS::TokenType::Let: | 
					
						
							|  |  |  |                 case JS::TokenType::New: | 
					
						
							| 
									
										
										
										
											2020-04-11 14:18:48 +01:00
										 |  |  |                 case JS::TokenType::Throw: | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  |                 case JS::TokenType::Typeof: | 
					
						
							|  |  |  |                 case JS::TokenType::Var: | 
					
						
							|  |  |  |                 case JS::TokenType::Void: | 
					
						
							|  |  |  |                     stylize({ start, end }, { Line::Style::Foreground(Line::Style::Color::Blue), Line::Style::Bold }); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case JS::TokenType::Await: | 
					
						
							| 
									
										
										
										
											2020-04-11 14:18:48 +01:00
										 |  |  |                 case JS::TokenType::Case: | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  |                 case JS::TokenType::Catch: | 
					
						
							|  |  |  |                 case JS::TokenType::Do: | 
					
						
							|  |  |  |                 case JS::TokenType::Else: | 
					
						
							|  |  |  |                 case JS::TokenType::Finally: | 
					
						
							|  |  |  |                 case JS::TokenType::For: | 
					
						
							|  |  |  |                 case JS::TokenType::If: | 
					
						
							|  |  |  |                 case JS::TokenType::Return: | 
					
						
							| 
									
										
										
										
											2020-04-11 14:18:48 +01:00
										 |  |  |                 case JS::TokenType::Switch: | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  |                 case JS::TokenType::Try: | 
					
						
							|  |  |  |                 case JS::TokenType::While: | 
					
						
							|  |  |  |                 case JS::TokenType::Yield: | 
					
						
							|  |  |  |                     stylize({ start, end }, { Line::Style::Foreground(Line::Style::Color::Cyan), Line::Style::Italic }); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case JS::TokenType::Identifier: | 
					
						
							| 
									
										
										
										
											2020-04-16 16:37:12 +02:00
										 |  |  |                     stylize({ start, end }, { Line::Style::Foreground(Line::Style::Color::White), Line::Style::Bold }); | 
					
						
							| 
									
										
										
										
											2020-04-09 08:00:45 +04:30
										 |  |  |                 default: | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             editor.set_prompt(prompt_for_level(open_indents)); | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         auto complete = [&interpreter, &editor = *s_editor](const String& token) -> Vector<Line::CompletionSuggestion> { | 
					
						
							| 
									
										
										
										
											2020-04-11 19:02:15 +04:30
										 |  |  |             if (token.length() == 0) | 
					
						
							|  |  |  |                 return {}; // nyeh
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  |             StringView line { editor.buffer().data(), editor.cursor() }; | 
					
						
							|  |  |  |             // we're only going to complete either
 | 
					
						
							|  |  |  |             //    - <N>
 | 
					
						
							|  |  |  |             //        where N is part of the name of a variable
 | 
					
						
							|  |  |  |             //    - <N>.<P>
 | 
					
						
							|  |  |  |             //        where N is the complete name of a variable and
 | 
					
						
							|  |  |  |             //        P is part of the name of one of its properties
 | 
					
						
							| 
									
										
										
										
											2020-04-19 18:32:28 +04:30
										 |  |  |             Vector<Line::CompletionSuggestion> results; | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  | 
 | 
					
						
							|  |  |  |             Function<void(const JS::Shape&, const StringView&)> list_all_properties = [&results, &list_all_properties](const JS::Shape& shape, auto& property_pattern) { | 
					
						
							|  |  |  |                 for (const auto& descriptor : shape.property_table()) { | 
					
						
							|  |  |  |                     if (descriptor.value.attributes & JS::Attribute::Enumerable) { | 
					
						
							| 
									
										
										
										
											2020-04-11 19:02:15 +04:30
										 |  |  |                         if (descriptor.key.view().starts_with(property_pattern)) { | 
					
						
							| 
									
										
										
										
											2020-04-19 18:32:28 +04:30
										 |  |  |                             Line::CompletionSuggestion completion { descriptor.key }; | 
					
						
							| 
									
										
										
										
											2020-04-11 19:02:15 +04:30
										 |  |  |                             if (!results.contains_slow(completion)) { // hide duplicates
 | 
					
						
							|  |  |  |                                 results.append(completion); | 
					
						
							|  |  |  |                             } | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (const auto* prototype = shape.prototype()) { | 
					
						
							|  |  |  |                     list_all_properties(prototype->shape(), property_pattern); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (token.contains(".")) { | 
					
						
							|  |  |  |                 auto parts = token.split('.', true); | 
					
						
							|  |  |  |                 // refuse either `.` or `a.b.c`
 | 
					
						
							|  |  |  |                 if (parts.size() > 2 || parts.size() == 0) | 
					
						
							|  |  |  |                     return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 auto name = parts[0]; | 
					
						
							|  |  |  |                 auto property_pattern = parts[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 auto maybe_variable = interpreter->get_variable(name); | 
					
						
							| 
									
										
										
										
											2020-04-25 18:43:34 +02:00
										 |  |  |                 if (maybe_variable.is_empty()) { | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  |                     maybe_variable = interpreter->global_object().get(name); | 
					
						
							| 
									
										
										
										
											2020-04-25 18:43:34 +02:00
										 |  |  |                     if (maybe_variable.is_empty()) | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  |                         return {}; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-25 18:43:34 +02:00
										 |  |  |                 auto variable = maybe_variable; | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  |                 if (!variable.is_object()) | 
					
						
							|  |  |  |                     return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 const auto* object = variable.to_object(interpreter->heap()); | 
					
						
							|  |  |  |                 const auto& shape = object->shape(); | 
					
						
							|  |  |  |                 list_all_properties(shape, property_pattern); | 
					
						
							| 
									
										
										
										
											2020-04-11 19:02:15 +04:30
										 |  |  |                 if (results.size()) | 
					
						
							|  |  |  |                     editor.suggest(property_pattern.length()); | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  |                 return results; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const auto& variable = interpreter->global_object(); | 
					
						
							|  |  |  |             list_all_properties(variable.shape(), token); | 
					
						
							| 
									
										
										
										
											2020-04-11 19:02:15 +04:30
										 |  |  |             if (results.size()) | 
					
						
							|  |  |  |                 editor.suggest(token.length()); | 
					
						
							| 
									
										
										
										
											2020-04-11 13:39:55 +04:30
										 |  |  |             return results; | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         s_editor->on_tab_complete_first_token = [complete](auto& value) { return complete(value); }; | 
					
						
							|  |  |  |         s_editor->on_tab_complete_other_token = [complete](auto& value) { return complete(value); }; | 
					
						
							| 
									
										
										
										
											2020-04-01 21:04:51 +02:00
										 |  |  |         repl(*interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2020-04-24 17:59:16 +01:00
										 |  |  |         interpreter = JS::Interpreter::create<JS::GlobalObject>(); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |         interpreter->heap().set_should_collect_on_every_allocation(gc_on_every_allocation); | 
					
						
							| 
									
										
										
										
											2020-04-12 22:49:04 -07:00
										 |  |  |         if (test_mode) | 
					
						
							|  |  |  |             enable_test_mode(*interpreter); | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 17:59:16 +01:00
										 |  |  |         signal(SIGINT, [](int) { | 
					
						
							|  |  |  |             sigint_handler(); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |         auto file = Core::File::construct(script_path); | 
					
						
							|  |  |  |         if (!file->open(Core::IODevice::ReadOnly)) { | 
					
						
							|  |  |  |             fprintf(stderr, "Failed to open %s: %s\n", script_path, file->error_string()); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         auto file_contents = file->read_all(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         StringView source; | 
					
						
							| 
									
										
										
										
											2020-04-01 15:20:32 -04:00
										 |  |  |         if (file_has_shebang(file_contents)) { | 
					
						
							|  |  |  |             source = strip_shebang(file_contents); | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             source = file_contents; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-13 02:05:21 +02:00
										 |  |  |         auto parser = JS::Parser(JS::Lexer(source)); | 
					
						
							|  |  |  |         auto program = parser.parse_program(); | 
					
						
							| 
									
										
										
										
											2020-03-12 10:51:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         if (s_dump_ast) | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |             program->dump(0); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 02:05:21 +02:00
										 |  |  |         if (parser.has_errors()) { | 
					
						
							|  |  |  |             printf("Parse Error\n"); | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 21:04:51 +02:00
										 |  |  |         auto result = interpreter->run(*program); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 10:55:37 -04:00
										 |  |  |         if (interpreter->exception()) { | 
					
						
							|  |  |  |             printf("Uncaught exception: "); | 
					
						
							|  |  |  |             print(interpreter->exception()->value()); | 
					
						
							|  |  |  |             interpreter->clear_exception(); | 
					
						
							| 
									
										
										
										
											2020-04-03 21:52:49 +01:00
										 |  |  |             return 1; | 
					
						
							| 
									
										
										
										
											2020-04-02 10:55:37 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-26 15:19:30 +01:00
										 |  |  |         if (s_print_last_result) | 
					
						
							| 
									
										
										
										
											2020-04-03 21:52:49 +01:00
										 |  |  |             print(result); | 
					
						
							| 
									
										
										
										
											2020-03-26 01:10:29 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-08 19:59:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } |