2021-10-29 21:25:12 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/TypeUtils.h"
|
|
|
|
|
2022-08-18 22:32:08 +01:00
|
|
|
#include "Luau/Normalize.h"
|
2021-11-05 02:34:35 +00:00
|
|
|
#include "Luau/Scope.h"
|
2021-10-29 21:25:12 +01:00
|
|
|
#include "Luau/ToString.h"
|
|
|
|
#include "Luau/TypeInfer.h"
|
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
2022-08-18 22:32:08 +01:00
|
|
|
std::optional<TypeId> findMetatableEntry(ErrorVec& errors, TypeId type, const std::string& entry, Location location)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
type = follow(type);
|
|
|
|
|
|
|
|
std::optional<TypeId> metatable = getMetatable(type);
|
|
|
|
if (!metatable)
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
TypeId unwrapped = follow(*metatable);
|
|
|
|
|
|
|
|
if (get<AnyTypeVar>(unwrapped))
|
2021-12-10 22:05:05 +00:00
|
|
|
return getSingletonTypes().anyType;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
const TableTypeVar* mtt = getTableType(unwrapped);
|
|
|
|
if (!mtt)
|
|
|
|
{
|
2022-07-08 02:22:39 +01:00
|
|
|
errors.push_back(TypeError{location, GenericError{"Metatable was not a table"}});
|
2021-10-29 21:25:12 +01:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto it = mtt->props.find(entry);
|
|
|
|
if (it != mtt->props.end())
|
|
|
|
return it->second.type;
|
|
|
|
else
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:32:08 +01:00
|
|
|
std::optional<TypeId> findTablePropertyRespectingMeta(ErrorVec& errors, TypeId ty, const std::string& name, Location location)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
if (get<AnyTypeVar>(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;
|
|
|
|
}
|
|
|
|
|
2022-02-24 23:53:37 +00:00
|
|
|
std::optional<TypeId> mtIndex = findMetatableEntry(errors, ty, "__index", location);
|
2022-02-18 01:18:01 +00:00
|
|
|
int count = 0;
|
2021-10-29 21:25:12 +01:00
|
|
|
while (mtIndex)
|
|
|
|
{
|
|
|
|
TypeId index = follow(*mtIndex);
|
2022-02-18 01:18:01 +00:00
|
|
|
|
2022-05-20 01:02:24 +01:00
|
|
|
if (count >= 100)
|
|
|
|
return std::nullopt;
|
2022-02-18 01:18:01 +00:00
|
|
|
|
2022-05-20 01:02:24 +01:00
|
|
|
++count;
|
2022-02-18 01:18:01 +00:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
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<FunctionTypeVar>(index))
|
|
|
|
{
|
2022-06-17 02:05:14 +01:00
|
|
|
std::optional<TypeId> r = first(follow(itf->retTypes));
|
2021-10-29 21:25:12 +01:00
|
|
|
if (!r)
|
2021-12-10 22:05:05 +00:00
|
|
|
return getSingletonTypes().nilType;
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
|
|
|
return *r;
|
|
|
|
}
|
|
|
|
else if (get<AnyTypeVar>(index))
|
2021-12-10 22:05:05 +00:00
|
|
|
return getSingletonTypes().anyType;
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
|
|
|
errors.push_back(TypeError{location, GenericError{"__index should either be a function or table. Got " + toString(index)}});
|
|
|
|
|
2022-02-24 23:53:37 +00:00
|
|
|
mtIndex = findMetatableEntry(errors, *mtIndex, "__index", location);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2022-09-02 00:14:03 +01:00
|
|
|
std::optional<TypeId> getIndexTypeFromType(const ScopePtr& scope, ErrorVec& errors, TypeArena* arena, TypeId type, const std::string& prop,
|
|
|
|
const Location& location, bool addErrors, InternalErrorReporter& handle)
|
2022-08-18 22:32:08 +01:00
|
|
|
{
|
|
|
|
type = follow(type);
|
|
|
|
|
|
|
|
if (get<ErrorTypeVar>(type) || get<AnyTypeVar>(type) || get<NeverTypeVar>(type))
|
|
|
|
return type;
|
|
|
|
|
|
|
|
if (auto f = get<FreeTypeVar>(type))
|
|
|
|
*asMutable(type) = TableTypeVar{TableState::Free, f->level};
|
|
|
|
|
|
|
|
if (isString(type))
|
|
|
|
{
|
|
|
|
std::optional<TypeId> mtIndex = Luau::findMetatableEntry(errors, getSingletonTypes().stringType, "__index", location);
|
|
|
|
LUAU_ASSERT(mtIndex);
|
|
|
|
type = *mtIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getTableType(type))
|
|
|
|
{
|
|
|
|
return findTablePropertyRespectingMeta(errors, type, prop, location);
|
|
|
|
}
|
|
|
|
else if (const ClassTypeVar* cls = get<ClassTypeVar>(type))
|
|
|
|
{
|
|
|
|
if (const Property* p = lookupClassProp(cls, prop))
|
|
|
|
return p->type;
|
|
|
|
}
|
|
|
|
else if (const UnionTypeVar* utv = get<UnionTypeVar>(type))
|
|
|
|
{
|
|
|
|
std::vector<TypeId> goodOptions;
|
|
|
|
std::vector<TypeId> badOptions;
|
|
|
|
|
|
|
|
for (TypeId t : utv)
|
|
|
|
{
|
|
|
|
// TODO: we should probably limit recursion here?
|
|
|
|
// RecursionLimiter _rl(&recursionCount, FInt::LuauTypeInferRecursionLimit);
|
|
|
|
|
|
|
|
// Not needed when we normalize types.
|
|
|
|
if (get<AnyTypeVar>(follow(t)))
|
|
|
|
return t;
|
|
|
|
|
|
|
|
if (std::optional<TypeId> ty = getIndexTypeFromType(scope, errors, arena, t, prop, location, /* addErrors= */ false, handle))
|
|
|
|
goodOptions.push_back(*ty);
|
|
|
|
else
|
|
|
|
badOptions.push_back(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!badOptions.empty())
|
|
|
|
{
|
|
|
|
if (addErrors)
|
|
|
|
{
|
|
|
|
if (goodOptions.empty())
|
|
|
|
errors.push_back(TypeError{location, UnknownProperty{type, prop}});
|
|
|
|
else
|
|
|
|
errors.push_back(TypeError{location, MissingUnionProperty{type, badOptions, prop}});
|
|
|
|
}
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (goodOptions.empty())
|
|
|
|
return getSingletonTypes().neverType;
|
|
|
|
|
|
|
|
if (goodOptions.size() == 1)
|
|
|
|
return goodOptions[0];
|
|
|
|
|
|
|
|
// TODO: inefficient.
|
|
|
|
TypeId result = arena->addType(UnionTypeVar{std::move(goodOptions)});
|
|
|
|
auto [ty, ok] = normalize(result, NotNull{scope.get()}, *arena, handle);
|
|
|
|
if (!ok && addErrors)
|
|
|
|
errors.push_back(TypeError{location, NormalizationTooComplex{}});
|
|
|
|
return ok ? ty : getSingletonTypes().anyType;
|
|
|
|
}
|
|
|
|
else if (const IntersectionTypeVar* itv = get<IntersectionTypeVar>(type))
|
|
|
|
{
|
|
|
|
std::vector<TypeId> parts;
|
|
|
|
|
|
|
|
for (TypeId t : itv->parts)
|
|
|
|
{
|
|
|
|
// TODO: we should probably limit recursion here?
|
|
|
|
// RecursionLimiter _rl(&recursionCount, FInt::LuauTypeInferRecursionLimit);
|
|
|
|
|
|
|
|
if (std::optional<TypeId> ty = getIndexTypeFromType(scope, errors, arena, t, prop, location, /* addErrors= */ false, handle))
|
|
|
|
parts.push_back(*ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If no parts of the intersection had the property we looked up for, it never existed at all.
|
|
|
|
if (parts.empty())
|
|
|
|
{
|
|
|
|
if (addErrors)
|
|
|
|
errors.push_back(TypeError{location, UnknownProperty{type, prop}});
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts.size() == 1)
|
|
|
|
return parts[0];
|
|
|
|
|
|
|
|
return arena->addType(IntersectionTypeVar{std::move(parts)}); // Not at all correct.
|
|
|
|
}
|
|
|
|
|
|
|
|
if (addErrors)
|
|
|
|
errors.push_back(TypeError{location, UnknownProperty{type, prop}});
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2022-09-02 00:14:03 +01:00
|
|
|
std::pair<size_t, std::optional<size_t>> getParameterExtents(const TxnLog* log, TypePackId tp)
|
|
|
|
{
|
|
|
|
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())
|
|
|
|
return {minCount, std::nullopt};
|
|
|
|
else
|
|
|
|
return {minCount, minCount + optionalCount};
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
} // namespace Luau
|