mirror of
				https://github.com/IntQuant/noita_entangled_worlds.git
				synced 2025-10-25 18:04:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			170 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			170 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import json
 | |
| 
 | |
| path = "/home/quant/.local/share/Steam/steamapps/common/Noita/tools_modding/lua_api_documentation.html"
 | |
| 
 | |
| lines = open(path).readlines()
 | |
| 
 | |
| lines_iter = iter(lines)
 | |
| 
 | |
| parsed = []
 | |
| 
 | |
| def maybe_map_types(name, typ):
 | |
|     if typ == "multiple types":
 | |
|         raise ValueError("no 'multiple types' either")
 | |
|     if name == "entity_id":
 | |
|         typ = "entity_id"
 | |
|     if name == "component_id":
 | |
|         typ = "component_id"
 | |
|     if typ == "float":
 | |
|         typ = "number"
 | |
|     if typ == "uint":
 | |
|         typ = "color"
 | |
|     if typ == "uint32":
 | |
|         typ = "color"
 | |
|     if typ == "name":
 | |
|         typ = "string"
 | |
|     if typ == "bool_is_new":
 | |
|         typ = "bool"
 | |
|     if typ == "boolean":
 | |
|         typ = "bool"
 | |
|     if typ == "item_entity_id":
 | |
|         typ = "entity_id"
 | |
|     if typ == "physics_body_id":
 | |
|         raise ValueError(f"{typ} not supported")
 | |
|     return typ
 | |
| 
 | |
| def parse_arg(arg_s):
 | |
|     if "|" in arg_s:
 | |
|         raise ValueError("multiple argument types not supported")
 | |
|     if "{" in arg_s:
 | |
|         raise ValueError("no table support for now")
 | |
|     if "multiple_types" in arg_s:
 | |
|         raise ValueError("no 'multiple_types' either")
 | |
|     other, *default = arg_s.split("=", maxsplit=1)
 | |
|     other = other.strip()
 | |
|     if default:
 | |
|         default = default[0].strip()
 | |
|     else:
 | |
|         default = None
 | |
|     name, typ = other.split(":", maxsplit=1)
 | |
| 
 | |
|     typ = maybe_map_types(name, typ)
 | |
| 
 | |
|     return {
 | |
|         "name": name,
 | |
|         "typ": typ,
 | |
|         "default": default,
 | |
|     }
 | |
| 
 | |
| def parse_ret(ret_s):
 | |
|     if not ret_s:
 | |
|         return None
 | |
|     
 | |
|     optional = ret_s.endswith("|nil")
 | |
|     ret_s = ret_s.removesuffix("|nil")
 | |
|     
 | |
|     if "|" in ret_s:
 | |
|         raise ValueError("multiple return types not supported")
 | |
|     if "multiple_types" in ret_s:
 | |
|         raise ValueError("no 'multiple_types' either")
 | |
| 
 | |
|     returns_vec = False
 | |
|     if ret_s.startswith("{"):
 | |
|         ret_s = ret_s.removeprefix("{").removesuffix("}")
 | |
|         returns_vec = True
 | |
|     if "-" in ret_s:
 | |
|         raise ValueError("No support for key-value tables in returns")
 | |
|     
 | |
|     typ = ret_s
 | |
|     name = None
 | |
|     if ":" in ret_s:
 | |
|         name, typ = ret_s.split(":", maxsplit=1)
 | |
| 
 | |
|     if typ.endswith(" -"):
 | |
|         optional = True
 | |
|         typ = typ.removesuffix(" -")
 | |
| 
 | |
|     typ = maybe_map_types(name, typ)
 | |
| 
 | |
|     return {
 | |
|         "name": name,
 | |
|         "typ": typ,
 | |
|         "optional": optional,
 | |
|         "is_vec": returns_vec,
 | |
|     }
 | |
|     
 | |
| 
 | |
| ignore = {
 | |
|     # Those have some specifics that make generic way of handling things not work on them
 | |
|     "PhysicsApplyForceOnArea",
 | |
|     "GetRandomActionWithType",
 | |
|     "GetParallelWorldPosition",
 | |
|     "EntityGetFirstHitboxCenter",
 | |
|     "InputGetJoystickAnalogStick",
 | |
|     "PhysicsAddBodyImage",
 | |
|     "PhysicsBodyIDGetBodyAABB",
 | |
|     "GuiTextInput",
 | |
| }
 | |
| skipped = 0
 | |
| deprecated = 0
 | |
| 
 | |
| # 2 lazy 2 parse xml properly
 | |
| try:
 | |
|     while True:
 | |
|         line = next(lines_iter)
 | |
|         if line.startswith('<th><span class="function">'):
 | |
|             fn_line = line.strip()
 | |
|             ret_line = next(lines_iter).strip()
 | |
|             desc_line = next(lines_iter).strip()
 | |
| 
 | |
|             fn_name, other = fn_line.removeprefix('<th><span class="function">').split('</span>(<span class="field_name">', maxsplit=1)
 | |
|             args = other.removesuffix('</span><span class="func">)</span></th>').strip().split(", ")
 | |
|             try:
 | |
|                 args = [parse_arg(arg) for arg in args if ":" in arg]
 | |
|             except ValueError as e:
 | |
|                 skipped += 1
 | |
|                 print(f"Skipping {fn_name}: {e}")
 | |
|                 continue
 | |
| 
 | |
| 
 | |
|             rets = ret_line.removeprefix('<th><span class="field_name">').removesuffix('</span></th></th>').strip()
 | |
|             desc = desc_line.removeprefix('<th><span class="description">').removesuffix('</span></th></th>').strip().replace("</br>", "\n")
 | |
| 
 | |
|             if "Debugish" in rets:
 | |
|                 rets, desc = rets.split(" (", maxsplit=1)
 | |
|                 desc = desc.removesuffix(")")
 | |
|             rets = rets.split(", ")
 | |
|             try:
 | |
|                 rets = [parse_ret(ret) for ret in rets if ret]
 | |
|             except ValueError as e:
 | |
|                 print(f"Skipping {fn_name}: {e}")
 | |
|                 skipped += 1
 | |
|                 continue
 | |
|             if not desc:
 | |
|                 desc = "Nolla forgot to include a description :("
 | |
| 
 | |
|             if "Deprecated" in desc_line:
 | |
|                 deprecated += 1
 | |
|                 print(f"Skipping {fn_name}: deprecated")
 | |
|                 continue
 | |
| 
 | |
|             #print(fn_line, ret_line, desc_line)
 | |
|             
 | |
|             if fn_name not in ignore:
 | |
|                 #print(fn_name, args, "->", ret)
 | |
|                 parsed.append({
 | |
|                     "fn_name": fn_name,
 | |
|                     "args": args,
 | |
|                     "desc": desc,
 | |
|                     "rets": rets
 | |
|                 })
 | |
|             else:
 | |
|                 skipped += 1
 | |
| 
 | |
| except StopIteration:
 | |
|     pass
 | |
| 
 | |
| 
 | |
| print("Total skipped:", skipped, deprecated)
 | |
| print("Total parsed:", len(parsed))
 | |
| json.dump(parsed, open("ewext/noita_api_macro/src/lua_api.json", "w"), indent=2)
 | 
