| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106 |
- #pragma once
- #include "namedict.h"
- #include "str.h"
- #include "tuplelist.h"
- namespace pkpy{
- struct NeedMoreLines {
- NeedMoreLines(bool is_compiling_class) : is_compiling_class(is_compiling_class) {}
- bool is_compiling_class;
- };
- struct HandledException {};
- struct UnhandledException {};
- struct ToBeRaisedException {};
- enum CompileMode {
- EXEC_MODE,
- EVAL_MODE,
- REPL_MODE,
- JSON_MODE,
- CELL_MODE
- };
- struct SourceData {
- std::string source;
- Str filename;
- std::vector<const char*> line_starts;
- CompileMode mode;
- std::pair<const char*,const char*> get_line(int lineno) const {
- if(lineno == -1) return {nullptr, nullptr};
- lineno -= 1;
- if(lineno < 0) lineno = 0;
- const char* _start = line_starts.at(lineno);
- const char* i = _start;
- while(*i != '\n' && *i != '\0') i++;
- return {_start, i};
- }
- SourceData(const Str& source, const Str& filename, CompileMode mode) {
- int index = 0;
- // Skip utf8 BOM if there is any.
- if (strncmp(source.begin(), "\xEF\xBB\xBF", 3) == 0) index += 3;
- // Replace all '\r' with ' '
- std::stringstream ss;
- while(index < source.length()){
- if(source[index] == '\r') ss << ' ';
- else ss << source[index];
- index++;
- }
- this->filename = filename;
- this->source = ss.str();
- line_starts.push_back(this->source.c_str());
- this->mode = mode;
- }
- Str snapshot(int lineno, const char* cursor=nullptr){
- std::stringstream ss;
- ss << " " << "File \"" << filename << "\", line " << lineno << '\n';
- std::pair<const char*,const char*> pair = get_line(lineno);
- Str line = "<?>";
- int removed_spaces = 0;
- if(pair.first && pair.second){
- line = Str(pair.first, pair.second-pair.first).lstrip();
- removed_spaces = pair.second - pair.first - line.length();
- if(line.empty()) line = "<?>";
- }
- ss << " " << line;
- if(cursor && line != "<?>" && cursor >= pair.first && cursor <= pair.second){
- auto column = cursor - pair.first - removed_spaces;
- if(column >= 0) ss << "\n " << std::string(column, ' ') << "^";
- }
- return ss.str();
- }
- };
- class Exception {
- using StackTrace = stack<Str>;
- StrName type;
- Str msg;
- StackTrace stacktrace;
- public:
- Exception(StrName type, Str msg): type(type), msg(msg) {}
- bool match_type(StrName type) const { return this->type == type;}
- bool is_re = true;
- void st_push(Str snapshot){
- if(stacktrace.size() >= 8) return;
- stacktrace.push(snapshot);
- }
- Str summary() const {
- StackTrace st(stacktrace);
- std::stringstream ss;
- if(is_re) ss << "Traceback (most recent call last):\n";
- while(!st.empty()) { ss << st.top() << '\n'; st.pop(); }
- if (!msg.empty()) ss << type.sv() << ": " << msg;
- else ss << type.sv();
- return ss.str();
- }
- };
- } // namespace pkpy
|