// 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/Normalize.h" #include "Luau/Scope.h" #include "Luau/ToString.h" #include "Luau/TypeInfer.h" #include namespace Luau { std::optional findMetatableEntry( NotNull singletonTypes, ErrorVec& errors, TypeId type, const std::string& entry, Location location) { type = follow(type); std::optional metatable = getMetatable(type, singletonTypes); 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( NotNull singletonTypes, ErrorVec& errors, TypeId ty, const std::string& 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(singletonTypes, errors, ty, "__index", location); int count = 0; while (mtIndex) { TypeId index = follow(*mtIndex); if (count >= 100) return std::nullopt; ++count; 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->retTypes)); 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(singletonTypes, errors, *mtIndex, "__index", location); } return std::nullopt; } std::pair> getParameterExtents(const TxnLog* log, TypePackId tp, bool includeHiddenVariadics) { size_t minCount = 0; size_t optionalCount = 0; auto it = begin(tp, log); auto endIter = end(tp); while (it != endIter) { TypeId ty = *it; if (isOptional(ty)) ++optionalCount; else { minCount += optionalCount; optionalCount = 0; minCount++; } ++it; } if (it.tail() && isVariadicTail(*it.tail(), *log, includeHiddenVariadics)) return {minCount, std::nullopt}; else return {minCount, minCount + optionalCount}; } TypePack extendTypePack(TypeArena& arena, NotNull singletonTypes, TypePackId pack, size_t length) { TypePack result; while (true) { pack = follow(pack); if (const TypePack* p = get(pack)) { size_t i = 0; while (i < p->head.size() && result.head.size() < length) { result.head.push_back(p->head[i]); ++i; } if (result.head.size() == length) { if (i == p->head.size()) result.tail = p->tail; else { TypePackId newTail = arena.addTypePack(TypePack{}); TypePack* newTailPack = getMutable(newTail); newTailPack->head.insert(newTailPack->head.begin(), p->head.begin() + i, p->head.end()); newTailPack->tail = p->tail; result.tail = newTail; } return result; } else if (p->tail) { pack = *p->tail; continue; } else { // There just aren't enough types in this pack to satisfy the request. return result; } } else if (const VariadicTypePack* vtp = get(pack)) { while (result.head.size() < length) result.head.push_back(vtp->ty); result.tail = pack; return result; } else if (FreeTypePack* ftp = getMutable(pack)) { // If we need to get concrete types out of a free pack, we choose to // interpret this as proof that the pack must have at least 'length' // elements. We mint fresh types for each element we're extracting // and rebind the free pack to be a TypePack containing them. We // also have to create a new tail. TypePack newPack; newPack.tail = arena.freshTypePack(ftp->scope); while (result.head.size() < length) { newPack.head.push_back(arena.freshType(ftp->scope)); result.head.push_back(newPack.head.back()); } asMutable(pack)->ty.emplace(std::move(newPack)); return result; } else if (const Unifiable::Error* etp = getMutable(pack)) { while (result.head.size() < length) result.head.push_back(singletonTypes->errorRecoveryType()); result.tail = pack; return result; } else { // If the pack is blocked or generic, we can't extract. // Return whatever we've got with this pack as the tail. result.tail = pack; return result; } } } std::vector reduceUnion(const std::vector& types) { std::vector result; for (TypeId t : types) { t = follow(t); if (get(t)) continue; if (get(t) || get(t)) return {t}; if (const UnionTypeVar* utv = get(t)) { for (TypeId ty : utv) { ty = follow(ty); if (get(ty)) continue; if (get(ty) || get(ty)) return {ty}; if (result.end() == std::find(result.begin(), result.end(), ty)) result.push_back(ty); } } else if (std::find(result.begin(), result.end(), t) == result.end()) result.push_back(t); } return result; } static std::optional tryStripUnionFromNil(TypeArena& arena, TypeId ty) { if (const UnionTypeVar* utv = get(ty)) { if (!std::any_of(begin(utv), end(utv), isNil)) return ty; std::vector result; for (TypeId option : utv) { if (!isNil(option)) result.push_back(option); } if (result.empty()) return std::nullopt; return result.size() == 1 ? result[0] : arena.addType(UnionTypeVar{std::move(result)}); } return std::nullopt; } TypeId stripNil(NotNull singletonTypes, TypeArena& arena, TypeId ty) { ty = follow(ty); if (get(ty)) { std::optional cleaned = tryStripUnionFromNil(arena, ty); // If there is no union option without 'nil' if (!cleaned) return singletonTypes->nilType; return follow(*cleaned); } return follow(ty); } } // namespace Luau