#include #include "types/str.h" #include "io.h" #include "stdlib.h" #include "stdio.h" #include "stdint.h" #include "table.h" int _print_type = 0; int _max_depth = 2; int _start_nl_at = 3; int _collapse_all = 0; int _collapse_to_memory = 1; int _print_meta = 0; int _file_malloc_chunk = 512; lua_Debug i_get_debug(lua_State* L){ lua_Debug ar; lua_getstack(L, 1, &ar); lua_getinfo(L, "nSl", &ar); return ar; } int l_debug(lua_State* L){ size_t input_len = 0; char* input = (char*)luaL_checklstring(L, 1, &input_len); lua_Debug debug = i_get_debug(L); printf(color_gray"[ debug ] (%s:%i) %s\n"color_reset, debug.source, debug.currentline, input); return 0; } int l_log(lua_State* L){ size_t input_len = 0; char* input = (char*)luaL_checklstring(L, 1, &input_len); lua_Debug debug = i_get_debug(L); printf(color_green"[ log ] (%s:%i)"color_gray" %s\n"color_reset, debug.source, debug.currentline, input); return 0; } int l_warn(lua_State* L){ size_t input_len = 0; char* input = (char*)luaL_checklstring(L, 1, &input_len); lua_Debug debug = i_get_debug(L); printf(color_yellow"[ warn ] (%s:%i) %s\n"color_reset, debug.source, debug.currentline, input); return 0; } int l_error(lua_State* L){ size_t input_len = 0; char* input = (char*)luaL_checklstring(L, 1, &input_len); lua_Debug debug = i_get_debug(L); printf(color_red"[ error ] (%s:%i) %s\n"color_reset, debug.source, debug.currentline, input); return 0; } void print_indentation(int i){ for(int z = 0; z < i; z++) printf(" "); } void i_pprint(lua_State* L, int indent, int skip_indent){ int last_idx = lua_gettop(L); const char* type = lua_typename(L,lua_type(L,-1)); switch(lua_type(L,-1)){ case LUA_TTABLE: printf(" "); if(indent >= _max_depth && _max_depth >= 0) { printf("{"color_gray); if(_collapse_to_memory) printf(" %p ",lua_topointer(L, -1)); else printf(" ... "); printf(color_reset"}"); break; } int skip = i_len(L,last_idx) < _start_nl_at || _collapse_all; printf("{"); if(!skip) printf("\n"); lua_pushnil(L); for(;lua_next(L,last_idx) != 0;){ if(lua_type(L,-2) == LUA_TSTRING){ if(!skip) print_indentation(indent); printf(" '%s'"color_gray": "color_reset, lua_tostring(L,-2)); } else { if(!skip) print_indentation(indent + 1); //printf(" '%i'"color_gray": "color_reset, luaL_checknumber(L,-2)); } int tuwu = lua_gettop(L); i_pprint(L,indent+1,1); printf(","); if(!skip) printf("\n"); lua_settop(L, tuwu); lua_pop(L,1); } print_indentation(indent); printf("}"); break; case LUA_TSTRING: if(!skip_indent) print_indentation(indent); size_t len; char* wowa = (char*)luaL_tolstring(L, -1, &len); printf("\""); for(int i = 0; i != len; i++) { printf("%c",wowa[i]); if(wowa[i] == '\n') print_indentation(indent); } printf("\""); break; case LUA_TBOOLEAN: if(!skip_indent) print_indentation(indent); printf(color_blue"%s"color_reset, lua_toboolean(L,-1)?"true":"false"); break; case LUA_TFUNCTION: if(!skip_indent) print_indentation(indent); printf(color_yellow"(fn,%p)"color_reset, lua_topointer(L, -1)); break; case LUA_TUSERDATA: if(!skip_indent) print_indentation(indent); printf(color_yellow"(ud,%p)"color_reset, lua_touserdata(L,-1)); break; case LUA_TLIGHTUSERDATA: if(!skip_indent) print_indentation(indent); printf(color_yellow"(lud,%p)"color_reset, lua_topointer(L,-1)); break; //case LUA_TNUMBER: if(!skip_indent) print_indentation(indent); printf(color_yellow"%i"color_reset, (int)lua_tonumber(L,-1)); break; default: if(!skip_indent) print_indentation(indent); printf(color_yellow"%s"color_reset, lua_tostring(L,-1)); } if(_print_type){ if(lua_istable(L,last_idx)) printf(color_gray" : <%s:%lu>"color_reset,type,i_len(L,last_idx)); else printf(color_gray" : <%s>"color_reset,type); } if(_print_meta && lua_getmetatable(L, -1) != 0){ printf(color_lblue" (metatable "color_reset); int c = lua_gettop(L) - 1; i_pprint(L,indent+1,1); printf(color_lblue")"color_reset); lua_settop(L, c); } } int l_pprint(lua_State* L){ i_pprint(L,0,0); printf("\n"); return 0; } enum json_state { normal, reading }; #define push() \ if(is_array){\ lua_pushinteger(L, iter_count);\ } else {\ char kbuf[key->len];\ memset(kbuf, 0, key->len);\ strncpy(kbuf, key->c + 1, key->len - 2);\ lua_pushstring(L, kbuf);\ }\ char buf[value->len];\ memset(buf, 0, value->len);\ switch(value->c[0]){\ case '{': case '[':\ json_parse(L, value);\ break;\ case '"':\ strncpy(buf, value->c + 1, value->len - 2);\ lua_pushstring(L, buf);\ break;\ default:\ lua_pushnumber(L, atof(value->c));\ break;\ }\ lua_settable(L, last_idx); void json_parse(lua_State* L, str* raw){ enum json_state state = normal; char topush[2] = {0,0}; int count = 0; char last = '\0'; int token_depth = 0; str* key = str_init(""); str* value = str_init(""); lua_newtable(L); int last_idx = lua_gettop(L); int is_array = raw->c[0] == '['; int i = 1; int iter_count = 1; int escape_next = 0; for(i = 1; i != raw->len - 1; i++){ topush[0] = *(raw->c + i); if(state == normal && (topush[0] == ' ' || topush[0] == '\n' || topush[0] <= 10)) continue; if(escape_next > 0) escape_next--; switch(topush[0]){ case '\\': escape_next = 2; break; case '"': if(escape_next == 1) break; case '{': case '}': case '[': case ']': if((last == '{' && topush[0] == '}') || (last == '[' && topush[0] == ']')) token_depth--; if((last == '\0' || (last == '"' && topush[0] == '"') || (last == '{' && topush[0] == '}') || (last == '[' && topush[0] == ']'))){ if(token_depth == 0){ if(last == '\0'){ last = topush[0]; } else { last = '\0'; } if(state==reading) state = normal; else state = reading; } } break; case ',': if(state == normal){ push() str_clear(key); str_clear(value); count = 0; iter_count++; continue; } break; case ':': if(state == normal){ count++; continue; } break; } //if(escape_next == 2) continue; if((last == '{' && topush[0] == '{') || (last == '[' && topush[0] == '[')) token_depth++; if(count == 0 && !is_array) str_push(key, topush); else str_push(value, topush); } push() str_free(key); str_free(value); //printf("key: %s, value : %s\n",key.c,value.c); } int l_json_parse(lua_State* L){ str* raw_json = str_init((char*)luaL_checklstring(L, 1, NULL)); json_parse(L, raw_json); str_free(raw_json); return 1; } int resolve_path(lua_State* L){ return 0; }