// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/TypeUtils.h" #include "Luau/ToString.h" #include "Luau/TypeInfer.h" LUAU_FASTFLAG(LuauStringMetatable) namespace Luau { std::optional findMetatableEntry(ErrorVec& errors, const ScopePtr& globalScope, TypeId type, std::string entry, Location location) { type = follow(type); if (!FFlag::LuauStringMetatable) { if (const PrimitiveTypeVar* primType = get(type)) { if (primType->type != PrimitiveTypeVar::String || "__index" != entry) return std::nullopt; auto it = globalScope->bindings.find(AstName{"string"}); if (it != globalScope->bindings.end()) return it->second.typeId; else return std::nullopt; } } std::optional metatable = getMetatable(type); if (!metatable) return std::nullopt; TypeId unwrapped = follow(*metatable); if (get(unwrapped)) return singletonTypes.anyType; const TableTypeVar* mtt = getTableType(unwrapped); if (!mtt) { errors.push_back(TypeError{location, GenericError{"Metatable was not a table."}}); return std::nullopt; } auto it = mtt->props.find(entry); if (it != mtt->props.end()) return it->second.type; else return std::nullopt; } std::optional findTablePropertyRespectingMeta(ErrorVec& errors, const ScopePtr& globalScope, TypeId ty, Name name, Location location) { if (get(ty)) return ty; if (const TableTypeVar* tableType = getTableType(ty)) { const auto& it = tableType->props.find(name); if (it != tableType->props.end()) return it->second.type; } std::optional mtIndex = findMetatableEntry(errors, globalScope, ty, "__index", location); while (mtIndex) { TypeId index = follow(*mtIndex); if (const auto& itt = getTableType(index)) { const auto& fit = itt->props.find(name); if (fit != itt->props.end()) return fit->second.type; } else if (const auto& itf = get(index)) { std::optional r = first(follow(itf->retType)); if (!r) return singletonTypes.nilType; else return *r; } else if (get(index)) return singletonTypes.anyType; else errors.push_back(TypeError{location, GenericError{"__index should either be a function or table. Got " + toString(index)}}); mtIndex = findMetatableEntry(errors, globalScope, *mtIndex, "__index", location); } return std::nullopt; } } // namespace Luau