// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/ConstraintSolverLogger.h" #include "Luau/JsonEmitter.h" namespace Luau { static void dumpScopeAndChildren(const Scope* scope, Json::JsonEmitter& emitter, ToStringOptions& opts) { emitter.writeRaw("{"); Json::write(emitter, "bindings"); emitter.writeRaw(":"); Json::ObjectEmitter o = emitter.writeObject(); for (const auto& [name, binding] : scope->bindings) { ToStringResult result = toStringDetailed(binding.typeId, opts); opts.nameMap = std::move(result.nameMap); o.writePair(name.c_str(), result.name); } o.finish(); emitter.writeRaw(","); Json::write(emitter, "children"); emitter.writeRaw(":"); Json::ArrayEmitter a = emitter.writeArray(); for (const Scope* child : scope->children) { dumpScopeAndChildren(child, emitter, opts); } a.finish(); emitter.writeRaw("}"); } static std::string dumpConstraintsToDot(std::vector>& constraints, ToStringOptions& opts) { std::string result = "digraph Constraints {\\n"; std::unordered_set> contained; for (NotNull c : constraints) { contained.insert(c); } for (NotNull c : constraints) { std::string id = std::to_string(reinterpret_cast(c.get())); result += id; result += " [label=\\\""; result += toString(*c, opts).c_str(); result += "\\\"];\\n"; for (NotNull dep : c->dependencies) { if (contained.count(dep) == 0) continue; result += std::to_string(reinterpret_cast(dep.get())); result += " -> "; result += id; result += ";\\n"; } } result += "}"; return result; } std::string ConstraintSolverLogger::compileOutput() { Json::JsonEmitter emitter; emitter.writeRaw("["); for (const std::string& snapshot : snapshots) { emitter.writeComma(); emitter.writeRaw(snapshot); } emitter.writeRaw("]"); return emitter.str(); } void ConstraintSolverLogger::captureBoundarySnapshot(const Scope* rootScope, std::vector>& unsolvedConstraints) { Json::JsonEmitter emitter; Json::ObjectEmitter o = emitter.writeObject(); o.writePair("type", "boundary"); o.writePair("constraintGraph", dumpConstraintsToDot(unsolvedConstraints, opts)); emitter.writeComma(); Json::write(emitter, "rootScope"); emitter.writeRaw(":"); dumpScopeAndChildren(rootScope, emitter, opts); o.finish(); snapshots.push_back(emitter.str()); } void ConstraintSolverLogger::prepareStepSnapshot( const Scope* rootScope, NotNull current, std::vector>& unsolvedConstraints) { Json::JsonEmitter emitter; Json::ObjectEmitter o = emitter.writeObject(); o.writePair("type", "step"); o.writePair("constraintGraph", dumpConstraintsToDot(unsolvedConstraints, opts)); o.writePair("currentId", std::to_string(reinterpret_cast(current.get()))); o.writePair("current", toString(*current, opts)); emitter.writeComma(); Json::write(emitter, "rootScope"); emitter.writeRaw(":"); dumpScopeAndChildren(rootScope, emitter, opts); o.finish(); preparedSnapshot = emitter.str(); } void ConstraintSolverLogger::commitPreparedStepSnapshot() { if (preparedSnapshot) { snapshots.push_back(std::move(*preparedSnapshot)); preparedSnapshot = std::nullopt; } } } // namespace Luau