astro-ghostcms/.pnpm-store/v3/files/3a/521278e2ee9a01bf1955896bf53...

226 lines
10 KiB
Plaintext

"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.isTypeReadonly = exports.readonlynessOptionsDefaults = exports.readonlynessOptionsSchema = void 0;
const utils_1 = require("@typescript-eslint/utils");
const tsutils = __importStar(require("ts-api-utils"));
const ts = __importStar(require("typescript"));
const propertyTypes_1 = require("./propertyTypes");
const TypeOrValueSpecifier_1 = require("./TypeOrValueSpecifier");
exports.readonlynessOptionsSchema = {
type: 'object',
additionalProperties: false,
properties: {
treatMethodsAsReadonly: {
type: 'boolean',
},
allow: {
type: 'array',
items: TypeOrValueSpecifier_1.typeOrValueSpecifierSchema,
},
},
};
exports.readonlynessOptionsDefaults = {
treatMethodsAsReadonly: false,
allow: [],
};
function hasSymbol(node) {
return Object.prototype.hasOwnProperty.call(node, 'symbol');
}
function isTypeReadonlyArrayOrTuple(program, type, options, seenTypes) {
const checker = program.getTypeChecker();
function checkTypeArguments(arrayType) {
const typeArguments = checker.getTypeArguments(arrayType);
// this shouldn't happen in reality as:
// - tuples require at least 1 type argument
// - ReadonlyArray requires at least 1 type argument
/* istanbul ignore if */ if (typeArguments.length === 0) {
return 3 /* Readonlyness.Readonly */;
}
// validate the element types are also readonly
if (typeArguments.some(typeArg => isTypeReadonlyRecurser(program, typeArg, options, seenTypes) ===
2 /* Readonlyness.Mutable */)) {
return 2 /* Readonlyness.Mutable */;
}
return 3 /* Readonlyness.Readonly */;
}
if (checker.isArrayType(type)) {
const symbol = utils_1.ESLintUtils.nullThrows(type.getSymbol(), utils_1.ESLintUtils.NullThrowsReasons.MissingToken('symbol', 'array type'));
const escapedName = symbol.getEscapedName();
// eslint-disable-next-line @typescript-eslint/no-unsafe-enum-comparison
if (escapedName === 'Array') {
return 2 /* Readonlyness.Mutable */;
}
return checkTypeArguments(type);
}
if (checker.isTupleType(type)) {
if (!type.target.readonly) {
return 2 /* Readonlyness.Mutable */;
}
return checkTypeArguments(type);
}
return 1 /* Readonlyness.UnknownType */;
}
function isTypeReadonlyObject(program, type, options, seenTypes) {
const checker = program.getTypeChecker();
function checkIndexSignature(kind) {
const indexInfo = checker.getIndexInfoOfType(type, kind);
if (indexInfo) {
if (!indexInfo.isReadonly) {
return 2 /* Readonlyness.Mutable */;
}
if (indexInfo.type === type || seenTypes.has(indexInfo.type)) {
return 3 /* Readonlyness.Readonly */;
}
return isTypeReadonlyRecurser(program, indexInfo.type, options, seenTypes);
}
return 1 /* Readonlyness.UnknownType */;
}
const properties = type.getProperties();
if (properties.length) {
// ensure the properties are marked as readonly
for (const property of properties) {
if (options.treatMethodsAsReadonly) {
if (property.valueDeclaration !== undefined &&
hasSymbol(property.valueDeclaration) &&
tsutils.isSymbolFlagSet(property.valueDeclaration.symbol, ts.SymbolFlags.Method)) {
continue;
}
const declarations = property.getDeclarations();
const lastDeclaration = declarations !== undefined && declarations.length > 0
? declarations[declarations.length - 1]
: undefined;
if (lastDeclaration !== undefined &&
hasSymbol(lastDeclaration) &&
tsutils.isSymbolFlagSet(lastDeclaration.symbol, ts.SymbolFlags.Method)) {
continue;
}
}
if (tsutils.isPropertyReadonlyInType(type, property.getEscapedName(), checker)) {
continue;
}
const name = ts.getNameOfDeclaration(property.valueDeclaration);
if (name && ts.isPrivateIdentifier(name)) {
continue;
}
return 2 /* Readonlyness.Mutable */;
}
// all properties were readonly
// now ensure that all of the values are readonly also.
// do this after checking property readonly-ness as a perf optimization,
// as we might be able to bail out early due to a mutable property before
// doing this deep, potentially expensive check.
for (const property of properties) {
const propertyType = utils_1.ESLintUtils.nullThrows((0, propertyTypes_1.getTypeOfPropertyOfType)(checker, type, property), utils_1.ESLintUtils.NullThrowsReasons.MissingToken(`property "${property.name}"`, 'type'));
// handle recursive types.
// we only need this simple check, because a mutable recursive type will break via the above prop readonly check
if (seenTypes.has(propertyType)) {
continue;
}
if (isTypeReadonlyRecurser(program, propertyType, options, seenTypes) ===
2 /* Readonlyness.Mutable */) {
return 2 /* Readonlyness.Mutable */;
}
}
}
const isStringIndexSigReadonly = checkIndexSignature(ts.IndexKind.String);
if (isStringIndexSigReadonly === 2 /* Readonlyness.Mutable */) {
return isStringIndexSigReadonly;
}
const isNumberIndexSigReadonly = checkIndexSignature(ts.IndexKind.Number);
if (isNumberIndexSigReadonly === 2 /* Readonlyness.Mutable */) {
return isNumberIndexSigReadonly;
}
return 3 /* Readonlyness.Readonly */;
}
// a helper function to ensure the seenTypes map is always passed down, except by the external caller
function isTypeReadonlyRecurser(program, type, options, seenTypes) {
const checker = program.getTypeChecker();
seenTypes.add(type);
if (options.allow?.some(specifier => (0, TypeOrValueSpecifier_1.typeMatchesSpecifier)(type, specifier, program))) {
return 3 /* Readonlyness.Readonly */;
}
if (tsutils.isUnionType(type)) {
// all types in the union must be readonly
const result = tsutils
.unionTypeParts(type)
.every(t => seenTypes.has(t) ||
isTypeReadonlyRecurser(program, t, options, seenTypes) ===
3 /* Readonlyness.Readonly */);
const readonlyness = result ? 3 /* Readonlyness.Readonly */ : 2 /* Readonlyness.Mutable */;
return readonlyness;
}
if (tsutils.isIntersectionType(type)) {
// Special case for handling arrays/tuples (as readonly arrays/tuples always have mutable methods).
if (type.types.some(t => checker.isArrayType(t) || checker.isTupleType(t))) {
const allReadonlyParts = type.types.every(t => seenTypes.has(t) ||
isTypeReadonlyRecurser(program, t, options, seenTypes) ===
3 /* Readonlyness.Readonly */);
return allReadonlyParts ? 3 /* Readonlyness.Readonly */ : 2 /* Readonlyness.Mutable */;
}
// Normal case.
const isReadonlyObject = isTypeReadonlyObject(program, type, options, seenTypes);
if (isReadonlyObject !== 1 /* Readonlyness.UnknownType */) {
return isReadonlyObject;
}
}
if (tsutils.isConditionalType(type)) {
const result = [type.root.node.trueType, type.root.node.falseType]
.map(checker.getTypeFromTypeNode)
.every(t => seenTypes.has(t) ||
isTypeReadonlyRecurser(program, t, options, seenTypes) ===
3 /* Readonlyness.Readonly */);
const readonlyness = result ? 3 /* Readonlyness.Readonly */ : 2 /* Readonlyness.Mutable */;
return readonlyness;
}
// all non-object, non-intersection types are readonly.
// this should only be primitive types
if (!tsutils.isObjectType(type)) {
return 3 /* Readonlyness.Readonly */;
}
// pure function types are readonly
if (type.getCallSignatures().length > 0 &&
type.getProperties().length === 0) {
return 3 /* Readonlyness.Readonly */;
}
const isReadonlyArray = isTypeReadonlyArrayOrTuple(program, type, options, seenTypes);
if (isReadonlyArray !== 1 /* Readonlyness.UnknownType */) {
return isReadonlyArray;
}
const isReadonlyObject = isTypeReadonlyObject(program, type, options, seenTypes);
/* istanbul ignore else */ if (isReadonlyObject !== 1 /* Readonlyness.UnknownType */) {
return isReadonlyObject;
}
throw new Error('Unhandled type');
}
/**
* Checks if the given type is readonly
*/
function isTypeReadonly(program, type, options = exports.readonlynessOptionsDefaults) {
return (isTypeReadonlyRecurser(program, type, options, new Set()) ===
3 /* Readonlyness.Readonly */);
}
exports.isTypeReadonly = isTypeReadonly;
//# sourceMappingURL=isTypeReadonly.js.map