2025-10-15 10:12:44 +03:00

1723 lines
69 KiB
Go

package compiler
import (
"context"
"fmt"
"io"
"maps"
"slices"
"strings"
"sync"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/ast"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/binder"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/checker"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/collections"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/core"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/diagnostics"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/module"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/modulespecifiers"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/outputpaths"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/parser"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/printer"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/scanner"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/sourcemap"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/tsoptions"
"efprojects.com/kitten-ipc/kitcom/internal/tsgo/tspath"
"github.com/go-json-experiment/json"
)
type ProgramOptions struct {
Host CompilerHost
Config *tsoptions.ParsedCommandLine
UseSourceOfProjectReference bool
SingleThreaded core.Tristate
CreateCheckerPool func(*Program) CheckerPool
TypingsLocation string
ProjectName string
JSDocParsingMode ast.JSDocParsingMode
}
func (p *ProgramOptions) canUseProjectReferenceSource() bool {
return p.UseSourceOfProjectReference && !p.Config.CompilerOptions().DisableSourceOfProjectReferenceRedirect.IsTrue()
}
type Program struct {
opts ProgramOptions
checkerPool CheckerPool
comparePathsOptions tspath.ComparePathsOptions
processedFiles
usesUriStyleNodeCoreModules core.Tristate
commonSourceDirectory string
commonSourceDirectoryOnce sync.Once
declarationDiagnosticCache collections.SyncMap[*ast.SourceFile, []*ast.Diagnostic]
programDiagnostics []*ast.Diagnostic
hasEmitBlockingDiagnostics collections.Set[tspath.Path]
sourceFilesToEmitOnce sync.Once
sourceFilesToEmit []*ast.SourceFile
// Cached unresolved imports for ATA
unresolvedImportsOnce sync.Once
unresolvedImports *collections.Set[string]
}
// FileExists implements checker.Program.
func (p *Program) FileExists(path string) bool {
return p.Host().FS().FileExists(path)
}
// GetCurrentDirectory implements checker.Program.
func (p *Program) GetCurrentDirectory() string {
return p.Host().GetCurrentDirectory()
}
// GetGlobalTypingsCacheLocation implements checker.Program.
func (p *Program) GetGlobalTypingsCacheLocation() string {
return "" // !!! see src/tsserver/nodeServer.ts for strada's node-specific implementation
}
// GetNearestAncestorDirectoryWithPackageJson implements checker.Program.
func (p *Program) GetNearestAncestorDirectoryWithPackageJson(dirname string) string {
scoped := p.resolver.GetPackageScopeForPath(dirname)
if scoped != nil && scoped.Exists() {
return scoped.PackageDirectory
}
return ""
}
// GetPackageJsonInfo implements checker.Program.
func (p *Program) GetPackageJsonInfo(pkgJsonPath string) modulespecifiers.PackageJsonInfo {
scoped := p.resolver.GetPackageScopeForPath(pkgJsonPath)
if scoped != nil && scoped.Exists() && scoped.PackageDirectory == tspath.GetDirectoryPath(pkgJsonPath) {
return scoped
}
return nil
}
// GetRedirectTargets implements checker.Program.
func (p *Program) GetRedirectTargets(path tspath.Path) []string {
return nil // !!! TODO: project references support
}
// gets the original file that was included in program
// this returns original source file name when including output of project reference
// otherwise same name
// Equivalent to originalFileName on SourceFile in Strada
func (p *Program) GetSourceOfProjectReferenceIfOutputIncluded(file ast.HasFileName) string {
if source, ok := p.outputFileToProjectReferenceSource[file.Path()]; ok {
return source
}
return file.FileName()
}
// GetProjectReferenceFromSource implements checker.Program.
func (p *Program) GetProjectReferenceFromSource(path tspath.Path) *tsoptions.SourceOutputAndProjectReference {
return p.projectReferenceFileMapper.getProjectReferenceFromSource(path)
}
// IsSourceFromProjectReference implements checker.Program.
func (p *Program) IsSourceFromProjectReference(path tspath.Path) bool {
return p.projectReferenceFileMapper.isSourceFromProjectReference(path)
}
func (p *Program) GetProjectReferenceFromOutputDts(path tspath.Path) *tsoptions.SourceOutputAndProjectReference {
return p.projectReferenceFileMapper.getProjectReferenceFromOutputDts(path)
}
func (p *Program) GetResolvedProjectReferenceFor(path tspath.Path) (*tsoptions.ParsedCommandLine, bool) {
return p.projectReferenceFileMapper.getResolvedReferenceFor(path)
}
func (p *Program) GetRedirectForResolution(file ast.HasFileName) *tsoptions.ParsedCommandLine {
redirect, _ := p.projectReferenceFileMapper.getRedirectForResolution(file)
return redirect
}
func (p *Program) GetParseFileRedirect(fileName string) string {
return p.projectReferenceFileMapper.getParseFileRedirect(ast.NewHasFileName(fileName, p.toPath(fileName)))
}
func (p *Program) ForEachResolvedProjectReference(
fn func(path tspath.Path, config *tsoptions.ParsedCommandLine, parent *tsoptions.ParsedCommandLine, index int),
) {
p.projectReferenceFileMapper.forEachResolvedProjectReference(fn)
}
// UseCaseSensitiveFileNames implements checker.Program.
func (p *Program) UseCaseSensitiveFileNames() bool {
return p.Host().FS().UseCaseSensitiveFileNames()
}
func (p *Program) UsesUriStyleNodeCoreModules() bool {
return p.usesUriStyleNodeCoreModules.IsTrue()
}
var _ checker.Program = (*Program)(nil)
/** This should have similar behavior to 'processSourceFile' without diagnostics or mutation. */
func (p *Program) GetSourceFileFromReference(origin *ast.SourceFile, ref *ast.FileReference) *ast.SourceFile {
// TODO: The module loader in corsa is fairly different than strada, it should probably be able to expose this functionality at some point,
// rather than redoing the logic approximately here, since most of the related logic now lives in module.Resolver
// Still, without the failed lookup reporting that only the loader does, this isn't terribly complicated
fileName := tspath.ResolvePath(tspath.GetDirectoryPath(origin.FileName()), ref.FileName)
supportedExtensionsBase := tsoptions.GetSupportedExtensions(p.Options(), nil /*extraFileExtensions*/)
supportedExtensions := tsoptions.GetSupportedExtensionsWithJsonIfResolveJsonModule(p.Options(), supportedExtensionsBase)
allowNonTsExtensions := p.Options().AllowNonTsExtensions.IsTrue()
if tspath.HasExtension(fileName) {
if !allowNonTsExtensions {
canonicalFileName := tspath.GetCanonicalFileName(fileName, p.UseCaseSensitiveFileNames())
supported := false
for _, group := range supportedExtensions {
if tspath.FileExtensionIsOneOf(canonicalFileName, group) {
supported = true
break
}
}
if !supported {
return nil // unsupported extensions are forced to fail
}
}
return p.GetSourceFile(fileName)
}
if allowNonTsExtensions {
extensionless := p.GetSourceFile(fileName)
if extensionless != nil {
return extensionless
}
}
// Only try adding extensions from the first supported group (which should be .ts/.tsx/.d.ts)
for _, ext := range supportedExtensions[0] {
result := p.GetSourceFile(fileName + ext)
if result != nil {
return result
}
}
return nil
}
func NewProgram(opts ProgramOptions) *Program {
p := &Program{opts: opts}
p.initCheckerPool()
p.processedFiles = processAllProgramFiles(p.opts, p.SingleThreaded())
p.verifyCompilerOptions()
return p
}
// Return an updated program for which it is known that only the file with the given path has changed.
// In addition to a new program, return a boolean indicating whether the data of the old program was reused.
func (p *Program) UpdateProgram(changedFilePath tspath.Path, newHost CompilerHost) (*Program, bool) {
oldFile := p.filesByPath[changedFilePath]
newOpts := p.opts
newOpts.Host = newHost
newFile := newHost.GetSourceFile(oldFile.ParseOptions())
if !canReplaceFileInProgram(oldFile, newFile) {
return NewProgram(newOpts), false
}
// TODO: reverify compiler options when config has changed?
result := &Program{
opts: newOpts,
comparePathsOptions: p.comparePathsOptions,
processedFiles: p.processedFiles,
usesUriStyleNodeCoreModules: p.usesUriStyleNodeCoreModules,
programDiagnostics: p.programDiagnostics,
hasEmitBlockingDiagnostics: p.hasEmitBlockingDiagnostics,
unresolvedImports: p.unresolvedImports,
}
result.initCheckerPool()
index := core.FindIndex(result.files, func(file *ast.SourceFile) bool { return file.Path() == newFile.Path() })
result.files = slices.Clone(result.files)
result.files[index] = newFile
result.filesByPath = maps.Clone(result.filesByPath)
result.filesByPath[newFile.Path()] = newFile
updateFileIncludeProcessor(result)
return result, true
}
func (p *Program) initCheckerPool() {
if p.opts.CreateCheckerPool != nil {
p.checkerPool = p.opts.CreateCheckerPool(p)
} else {
p.checkerPool = newCheckerPool(core.IfElse(p.SingleThreaded(), 1, 4), p)
}
}
func canReplaceFileInProgram(file1 *ast.SourceFile, file2 *ast.SourceFile) bool {
return file2 != nil &&
file1.ParseOptions() == file2.ParseOptions() &&
file1.UsesUriStyleNodeCoreModules == file2.UsesUriStyleNodeCoreModules &&
slices.EqualFunc(file1.Imports(), file2.Imports(), equalModuleSpecifiers) &&
slices.EqualFunc(file1.ModuleAugmentations, file2.ModuleAugmentations, equalModuleAugmentationNames) &&
slices.Equal(file1.AmbientModuleNames, file2.AmbientModuleNames) &&
slices.EqualFunc(file1.ReferencedFiles, file2.ReferencedFiles, equalFileReferences) &&
slices.EqualFunc(file1.TypeReferenceDirectives, file2.TypeReferenceDirectives, equalFileReferences) &&
slices.EqualFunc(file1.LibReferenceDirectives, file2.LibReferenceDirectives, equalFileReferences) &&
equalCheckJSDirectives(file1.CheckJsDirective, file2.CheckJsDirective)
}
func equalModuleSpecifiers(n1 *ast.Node, n2 *ast.Node) bool {
return n1.Kind == n2.Kind && (!ast.IsStringLiteral(n1) || n1.Text() == n2.Text())
}
func equalModuleAugmentationNames(n1 *ast.Node, n2 *ast.Node) bool {
return n1.Kind == n2.Kind && n1.Text() == n2.Text()
}
func equalFileReferences(f1 *ast.FileReference, f2 *ast.FileReference) bool {
return f1.FileName == f2.FileName && f1.ResolutionMode == f2.ResolutionMode && f1.Preserve == f2.Preserve
}
func equalCheckJSDirectives(d1 *ast.CheckJsDirective, d2 *ast.CheckJsDirective) bool {
return d1 == nil && d2 == nil || d1 != nil && d2 != nil && d1.Enabled == d2.Enabled
}
func (p *Program) SourceFiles() []*ast.SourceFile { return p.files }
func (p *Program) Options() *core.CompilerOptions { return p.opts.Config.CompilerOptions() }
func (p *Program) CommandLine() *tsoptions.ParsedCommandLine { return p.opts.Config }
func (p *Program) Host() CompilerHost { return p.opts.Host }
func (p *Program) GetConfigFileParsingDiagnostics() []*ast.Diagnostic {
return slices.Clip(p.opts.Config.GetConfigFileParsingDiagnostics())
}
// GetUnresolvedImports returns the unresolved imports for this program.
// The result is cached and computed only once.
func (p *Program) GetUnresolvedImports() *collections.Set[string] {
p.unresolvedImportsOnce.Do(func() {
if p.unresolvedImports == nil {
p.unresolvedImports = p.extractUnresolvedImports()
}
})
return p.unresolvedImports
}
func (p *Program) extractUnresolvedImports() *collections.Set[string] {
unresolvedSet := &collections.Set[string]{}
for _, sourceFile := range p.files {
unresolvedImports := p.extractUnresolvedImportsFromSourceFile(sourceFile)
for _, imp := range unresolvedImports {
unresolvedSet.Add(imp)
}
}
return unresolvedSet
}
func (p *Program) extractUnresolvedImportsFromSourceFile(file *ast.SourceFile) []string {
var unresolvedImports []string
resolvedModules := p.resolvedModules[file.Path()]
for cacheKey, resolution := range resolvedModules {
resolved := resolution.IsResolved()
if (!resolved || !tspath.ExtensionIsOneOf(resolution.Extension, tspath.SupportedTSExtensionsWithJsonFlat)) &&
!tspath.IsExternalModuleNameRelative(cacheKey.Name) {
unresolvedImports = append(unresolvedImports, cacheKey.Name)
}
}
return unresolvedImports
}
func (p *Program) SingleThreaded() bool {
return p.opts.SingleThreaded.DefaultIfUnknown(p.Options().SingleThreaded).IsTrue()
}
func (p *Program) BindSourceFiles() {
wg := core.NewWorkGroup(p.SingleThreaded())
for _, file := range p.files {
if !file.IsBound() {
wg.Queue(func() {
binder.BindSourceFile(file)
})
}
}
wg.RunAndWait()
}
func (p *Program) CheckSourceFiles(ctx context.Context, files []*ast.SourceFile) {
wg := core.NewWorkGroup(p.SingleThreaded())
checkers, done := p.checkerPool.GetAllCheckers(ctx)
defer done()
for _, checker := range checkers {
wg.Queue(func() {
for file := range p.checkerPool.Files(checker) {
if files == nil || slices.Contains(files, file) {
checker.CheckSourceFile(ctx, file)
}
}
})
}
wg.RunAndWait()
}
// Return the type checker associated with the program.
func (p *Program) GetTypeChecker(ctx context.Context) (*checker.Checker, func()) {
return p.checkerPool.GetChecker(ctx)
}
func (p *Program) GetTypeCheckers(ctx context.Context) ([]*checker.Checker, func()) {
return p.checkerPool.GetAllCheckers(ctx)
}
// Return a checker for the given file. We may have multiple checkers in concurrent scenarios and this
// method returns the checker that was tasked with checking the file. Note that it isn't possible to mix
// types obtained from different checkers, so only non-type data (such as diagnostics or string
// representations of types) should be obtained from checkers returned by this method.
func (p *Program) GetTypeCheckerForFile(ctx context.Context, file *ast.SourceFile) (*checker.Checker, func()) {
return p.checkerPool.GetCheckerForFile(ctx, file)
}
func (p *Program) GetResolvedModule(file ast.HasFileName, moduleReference string, mode core.ResolutionMode) *module.ResolvedModule {
if resolutions, ok := p.resolvedModules[file.Path()]; ok {
if resolved, ok := resolutions[module.ModeAwareCacheKey{Name: moduleReference, Mode: mode}]; ok {
return resolved
}
}
return nil
}
func (p *Program) GetResolvedModuleFromModuleSpecifier(file ast.HasFileName, moduleSpecifier *ast.StringLiteralLike) *module.ResolvedModule {
if !ast.IsStringLiteralLike(moduleSpecifier) {
panic("moduleSpecifier must be a StringLiteralLike")
}
mode := p.GetModeForUsageLocation(file, moduleSpecifier)
return p.GetResolvedModule(file, moduleSpecifier.Text(), mode)
}
func (p *Program) GetResolvedModules() map[tspath.Path]module.ModeAwareCache[*module.ResolvedModule] {
return p.resolvedModules
}
func (p *Program) GetSyntacticDiagnostics(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
return p.getDiagnosticsHelper(ctx, sourceFile, false /*ensureBound*/, false /*ensureChecked*/, p.getSyntacticDiagnosticsForFile)
}
func (p *Program) GetBindDiagnostics(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
return p.getDiagnosticsHelper(ctx, sourceFile, true /*ensureBound*/, false /*ensureChecked*/, p.getBindDiagnosticsForFile)
}
func (p *Program) GetSemanticDiagnostics(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
return p.getDiagnosticsHelper(ctx, sourceFile, true /*ensureBound*/, true /*ensureChecked*/, p.getSemanticDiagnosticsForFile)
}
func (p *Program) GetSemanticDiagnosticsNoFilter(ctx context.Context, sourceFiles []*ast.SourceFile) map[*ast.SourceFile][]*ast.Diagnostic {
p.BindSourceFiles()
p.CheckSourceFiles(ctx, sourceFiles)
if ctx.Err() != nil {
return nil
}
result := make(map[*ast.SourceFile][]*ast.Diagnostic, len(sourceFiles))
for _, file := range sourceFiles {
result[file] = SortAndDeduplicateDiagnostics(p.getSemanticDiagnosticsForFileNotFilter(ctx, file))
}
return result
}
func (p *Program) GetSuggestionDiagnostics(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
return p.getDiagnosticsHelper(ctx, sourceFile, true /*ensureBound*/, true /*ensureChecked*/, p.getSuggestionDiagnosticsForFile)
}
func (p *Program) GetProgramDiagnostics() []*ast.Diagnostic {
return SortAndDeduplicateDiagnostics(slices.Concat(
p.programDiagnostics,
p.includeProcessor.getDiagnostics(p).GetGlobalDiagnostics()))
}
func (p *Program) GetIncludeProcessorDiagnostics(sourceFile *ast.SourceFile) []*ast.Diagnostic {
if checker.SkipTypeChecking(sourceFile, p.Options(), p, false) {
return nil
}
filtered, _ := p.getDiagnosticsWithPrecedingDirectives(sourceFile, p.includeProcessor.getDiagnostics(p).GetDiagnosticsForFile(sourceFile.FileName()))
return filtered
}
func (p *Program) getSourceFilesToEmit(targetSourceFile *ast.SourceFile, forceDtsEmit bool) []*ast.SourceFile {
if targetSourceFile == nil && !forceDtsEmit {
p.sourceFilesToEmitOnce.Do(func() {
p.sourceFilesToEmit = getSourceFilesToEmit(p, nil, false)
})
return p.sourceFilesToEmit
}
return getSourceFilesToEmit(p, targetSourceFile, forceDtsEmit)
}
func (p *Program) verifyCompilerOptions() {
options := p.Options()
sourceFile := core.Memoize(func() *ast.SourceFile {
configFile := p.opts.Config.ConfigFile
if configFile == nil {
return nil
}
return configFile.SourceFile
})
configFilePath := core.Memoize(func() string {
file := sourceFile()
if file != nil {
return file.FileName()
}
return ""
})
getCompilerOptionsPropertySyntax := core.Memoize(func() *ast.PropertyAssignment {
return tsoptions.ForEachTsConfigPropArray(sourceFile(), "compilerOptions", core.Identity)
})
getCompilerOptionsObjectLiteralSyntax := core.Memoize(func() *ast.ObjectLiteralExpression {
compilerOptionsProperty := getCompilerOptionsPropertySyntax()
if compilerOptionsProperty != nil &&
compilerOptionsProperty.Initializer != nil &&
ast.IsObjectLiteralExpression(compilerOptionsProperty.Initializer) {
return compilerOptionsProperty.Initializer.AsObjectLiteralExpression()
}
return nil
})
createOptionDiagnosticInObjectLiteralSyntax := func(objectLiteral *ast.ObjectLiteralExpression, onKey bool, key1 string, key2 string, message *diagnostics.Message, args ...any) *ast.Diagnostic {
diag := tsoptions.ForEachPropertyAssignment(objectLiteral, key1, func(property *ast.PropertyAssignment) *ast.Diagnostic {
return tsoptions.CreateDiagnosticForNodeInSourceFile(sourceFile(), core.IfElse(onKey, property.Name(), property.Initializer), message, args...)
}, key2)
if diag != nil {
p.programDiagnostics = append(p.programDiagnostics, diag)
}
return diag
}
createCompilerOptionsDiagnostic := func(message *diagnostics.Message, args ...any) *ast.Diagnostic {
compilerOptionsProperty := getCompilerOptionsPropertySyntax()
var diag *ast.Diagnostic
if compilerOptionsProperty != nil {
diag = tsoptions.CreateDiagnosticForNodeInSourceFile(sourceFile(), compilerOptionsProperty.Name(), message, args...)
} else {
diag = ast.NewCompilerDiagnostic(message, args...)
}
p.programDiagnostics = append(p.programDiagnostics, diag)
return diag
}
createDiagnosticForOption := func(onKey bool, option1 string, option2 string, message *diagnostics.Message, args ...any) *ast.Diagnostic {
diag := createOptionDiagnosticInObjectLiteralSyntax(getCompilerOptionsObjectLiteralSyntax(), onKey, option1, option2, message, args...)
if diag == nil {
diag = createCompilerOptionsDiagnostic(message, args...)
}
return diag
}
createDiagnosticForOptionName := func(message *diagnostics.Message, option1 string, option2 string, args ...any) {
newArgs := make([]any, 0, len(args)+2)
newArgs = append(newArgs, option1, option2)
newArgs = append(newArgs, args...)
createDiagnosticForOption(true /*onKey*/, option1, option2, message, newArgs...)
}
createOptionValueDiagnostic := func(option1 string, message *diagnostics.Message, args ...any) {
createDiagnosticForOption(false /*onKey*/, option1, "", message, args...)
}
createRemovedOptionDiagnostic := func(name string, value string, useInstead string) {
var message *diagnostics.Message
var args []any
if value == "" {
message = diagnostics.Option_0_has_been_removed_Please_remove_it_from_your_configuration
args = []any{name}
} else {
message = diagnostics.Option_0_1_has_been_removed_Please_remove_it_from_your_configuration
args = []any{name, value}
}
diag := createDiagnosticForOption(value == "", name, "", message, args...)
if useInstead != "" {
diag.AddMessageChain(ast.NewCompilerDiagnostic(diagnostics.Use_0_instead, useInstead))
}
}
// Removed in TS7
if options.BaseUrl != "" {
// BaseUrl will have been turned absolute by this point.
var useInstead string
if configFilePath() != "" {
relative := tspath.GetRelativePathFromFile(configFilePath(), options.BaseUrl, p.comparePathsOptions)
if !(strings.HasPrefix(relative, "./") || strings.HasPrefix(relative, "../")) {
relative = "./" + relative
}
suggestion := tspath.CombinePaths(relative, "*")
useInstead = fmt.Sprintf(`"paths": {"*": [%s]}`, core.Must(json.Marshal(suggestion)))
}
createRemovedOptionDiagnostic("baseUrl", "", useInstead)
}
if options.OutFile != "" {
createRemovedOptionDiagnostic("outFile", "", "")
}
// if options.Target == core.ScriptTargetES3 {
// createRemovedOptionDiagnostic("target", "ES3", "")
// }
// if options.Target == core.ScriptTargetES5 {
// createRemovedOptionDiagnostic("target", "ES5", "")
// }
if options.Module == core.ModuleKindAMD {
createRemovedOptionDiagnostic("module", "AMD", "")
}
if options.Module == core.ModuleKindSystem {
createRemovedOptionDiagnostic("module", "System", "")
}
if options.Module == core.ModuleKindUMD {
createRemovedOptionDiagnostic("module", "UMD", "")
}
if options.StrictPropertyInitialization.IsTrue() && !options.GetStrictOptionValue(options.StrictNullChecks) {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "strictPropertyInitialization", "strictNullChecks")
}
if options.ExactOptionalPropertyTypes.IsTrue() && !options.GetStrictOptionValue(options.StrictNullChecks) {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "exactOptionalPropertyTypes", "strictNullChecks")
}
if options.IsolatedDeclarations.IsTrue() {
if options.GetAllowJS() {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_with_option_1, "allowJs", "isolatedDeclarations")
}
if !options.GetEmitDeclarations() {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "isolatedDeclarations", "declaration", "composite")
}
}
if options.InlineSourceMap.IsTrue() {
if options.SourceMap.IsTrue() {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_with_option_1, "sourceMap", "inlineSourceMap")
}
if options.MapRoot != "" {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_with_option_1, "mapRoot", "inlineSourceMap")
}
}
if options.Composite.IsTrue() {
if options.Declaration.IsFalse() {
createDiagnosticForOptionName(diagnostics.Composite_projects_may_not_disable_declaration_emit, "declaration", "")
}
if options.Incremental.IsFalse() {
createDiagnosticForOptionName(diagnostics.Composite_projects_may_not_disable_incremental_compilation, "declaration", "")
}
}
p.verifyProjectReferences()
if options.Composite.IsTrue() {
var rootPaths collections.Set[tspath.Path]
for _, fileName := range p.opts.Config.FileNames() {
rootPaths.Add(p.toPath(fileName))
}
for _, file := range p.files {
if sourceFileMayBeEmitted(file, p, false) && !rootPaths.Has(file.Path()) {
p.includeProcessor.addProcessingDiagnostic(&processingDiagnostic{
kind: processingDiagnosticKindExplainingFileInclude,
data: &includeExplainingDiagnostic{
file: file.Path(),
message: diagnostics.File_0_is_not_listed_within_the_file_list_of_project_1_Projects_must_list_all_files_or_use_an_include_pattern,
args: []any{file.FileName(), configFilePath()},
},
})
}
}
}
forEachOptionPathsSyntax := func(callback func(*ast.PropertyAssignment) *ast.Diagnostic) *ast.Diagnostic {
return tsoptions.ForEachPropertyAssignment(getCompilerOptionsObjectLiteralSyntax(), "paths", callback)
}
createDiagnosticForOptionPaths := func(onKey bool, key string, message *diagnostics.Message, args ...any) *ast.Diagnostic {
diag := forEachOptionPathsSyntax(func(pathProp *ast.PropertyAssignment) *ast.Diagnostic {
if ast.IsObjectLiteralExpression(pathProp.Initializer) {
return createOptionDiagnosticInObjectLiteralSyntax(pathProp.Initializer.AsObjectLiteralExpression(), onKey, key, "", message, args...)
}
return nil
})
if diag == nil {
diag = createCompilerOptionsDiagnostic(message, args...)
}
return diag
}
createDiagnosticForOptionPathKeyValue := func(key string, valueIndex int, message *diagnostics.Message, args ...any) *ast.Diagnostic {
diag := forEachOptionPathsSyntax(func(pathProp *ast.PropertyAssignment) *ast.Diagnostic {
if ast.IsObjectLiteralExpression(pathProp.Initializer) {
return tsoptions.ForEachPropertyAssignment(pathProp.Initializer.AsObjectLiteralExpression(), key, func(keyProps *ast.PropertyAssignment) *ast.Diagnostic {
initializer := keyProps.Initializer
if ast.IsArrayLiteralExpression(initializer) {
elements := initializer.AsArrayLiteralExpression().Elements
if elements != nil && len(elements.Nodes) > valueIndex {
diag := tsoptions.CreateDiagnosticForNodeInSourceFile(sourceFile(), elements.Nodes[valueIndex], message, args...)
p.programDiagnostics = append(p.programDiagnostics, diag)
return diag
}
}
return nil
})
}
return nil
})
if diag == nil {
diag = createCompilerOptionsDiagnostic(message, args...)
}
return diag
}
for key, value := range options.Paths.Entries() {
// !!! This code does not handle cases where where the path mappings have the wrong types,
// as that information is mostly lost during the parsing process.
if !hasZeroOrOneAsteriskCharacter(key) {
createDiagnosticForOptionPaths(true /*onKey*/, key, diagnostics.Pattern_0_can_have_at_most_one_Asterisk_character, key)
}
if value == nil {
createDiagnosticForOptionPaths(false /*onKey*/, key, diagnostics.Substitutions_for_pattern_0_should_be_an_array, key)
} else if len(value) == 0 {
createDiagnosticForOptionPaths(false /*onKey*/, key, diagnostics.Substitutions_for_pattern_0_shouldn_t_be_an_empty_array, key)
}
for i, subst := range value {
if !hasZeroOrOneAsteriskCharacter(subst) {
createDiagnosticForOptionPathKeyValue(key, i, diagnostics.Substitution_0_in_pattern_1_can_have_at_most_one_Asterisk_character, subst, key)
}
if !tspath.PathIsRelative(subst) && !tspath.PathIsAbsolute(subst) {
createDiagnosticForOptionPathKeyValue(key, i, diagnostics.Non_relative_paths_are_not_allowed_Did_you_forget_a_leading_Slash)
}
}
}
if options.SourceMap.IsFalseOrUnknown() && options.InlineSourceMap.IsFalseOrUnknown() {
if options.InlineSources.IsTrue() {
createDiagnosticForOptionName(diagnostics.Option_0_can_only_be_used_when_either_option_inlineSourceMap_or_option_sourceMap_is_provided, "inlineSources", "")
}
if options.SourceRoot != "" {
createDiagnosticForOptionName(diagnostics.Option_0_can_only_be_used_when_either_option_inlineSourceMap_or_option_sourceMap_is_provided, "sourceRoot", "")
}
}
if options.MapRoot != "" && !(options.SourceMap.IsTrue() || options.DeclarationMap.IsTrue()) {
// Error to specify --mapRoot without --sourcemap
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "mapRoot", "sourceMap", "declarationMap")
}
if options.DeclarationDir != "" {
if !options.GetEmitDeclarations() {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "declarationDir", "declaration", "composite")
}
}
if options.DeclarationMap.IsTrue() && !options.GetEmitDeclarations() {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "declarationMap", "declaration", "composite")
}
if options.Lib != nil && options.NoLib.IsTrue() {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_with_option_1, "lib", "noLib")
}
languageVersion := options.GetEmitScriptTarget()
firstNonAmbientExternalModuleSourceFile := core.Find(p.files, func(f *ast.SourceFile) bool { return ast.IsExternalModule(f) && !f.IsDeclarationFile })
if options.IsolatedModules.IsTrue() || options.VerbatimModuleSyntax.IsTrue() {
if options.Module == core.ModuleKindNone && languageVersion < core.ScriptTargetES2015 && options.IsolatedModules.IsTrue() {
// !!!
// createDiagnosticForOptionName(diagnostics.Option_isolatedModules_can_only_be_used_when_either_option_module_is_provided_or_option_target_is_ES2015_or_higher, "isolatedModules", "target")
}
if options.PreserveConstEnums.IsFalse() {
createDiagnosticForOptionName(diagnostics.Option_preserveConstEnums_cannot_be_disabled_when_0_is_enabled, core.IfElse(options.VerbatimModuleSyntax.IsTrue(), "verbatimModuleSyntax", "isolatedModules"), "preserveConstEnums")
}
} else if firstNonAmbientExternalModuleSourceFile != nil && languageVersion < core.ScriptTargetES2015 && options.Module == core.ModuleKindNone {
// !!!
}
if options.OutDir != "" ||
options.RootDir != "" ||
options.SourceRoot != "" ||
options.MapRoot != "" ||
(options.GetEmitDeclarations() && options.DeclarationDir != "") {
// !!! sheetal checkSourceFilesBelongToPath - for root Dir and configFile - explaining why file is in the program
dir := p.CommonSourceDirectory()
if options.OutDir != "" && dir == "" && core.Some(p.files, func(f *ast.SourceFile) bool { return tspath.GetRootLength(f.FileName()) > 1 }) {
createDiagnosticForOptionName(diagnostics.Cannot_find_the_common_subdirectory_path_for_the_input_files, "outDir", "")
}
}
if options.CheckJs.IsTrue() && !options.GetAllowJS() {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "checkJs", "allowJs")
}
if options.EmitDeclarationOnly.IsTrue() {
if !options.GetEmitDeclarations() {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "emitDeclarationOnly", "declaration", "composite")
}
}
// !!! emitDecoratorMetadata
if options.JsxFactory != "" {
if options.ReactNamespace != "" {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_with_option_1, "reactNamespace", "jsxFactory")
}
if options.Jsx == core.JsxEmitReactJSX || options.Jsx == core.JsxEmitReactJSXDev {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_when_option_jsx_is_1, "jsxFactory", tsoptions.InverseJsxOptionMap.GetOrZero(options.Jsx))
}
if parser.ParseIsolatedEntityName(options.JsxFactory) == nil {
createOptionValueDiagnostic("jsxFactory", diagnostics.Invalid_value_for_jsxFactory_0_is_not_a_valid_identifier_or_qualified_name, options.JsxFactory)
}
} else if options.ReactNamespace != "" && !scanner.IsIdentifierText(options.ReactNamespace, core.LanguageVariantStandard) {
createOptionValueDiagnostic("reactNamespace", diagnostics.Invalid_value_for_reactNamespace_0_is_not_a_valid_identifier, options.ReactNamespace)
}
if options.JsxFragmentFactory != "" {
if options.JsxFactory == "" {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "jsxFragmentFactory", "jsxFactory")
}
if options.Jsx == core.JsxEmitReactJSX || options.Jsx == core.JsxEmitReactJSXDev {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_when_option_jsx_is_1, "jsxFragmentFactory", tsoptions.InverseJsxOptionMap.GetOrZero(options.Jsx))
}
if parser.ParseIsolatedEntityName(options.JsxFragmentFactory) == nil {
createOptionValueDiagnostic("jsxFragmentFactory", diagnostics.Invalid_value_for_jsxFragmentFactory_0_is_not_a_valid_identifier_or_qualified_name, options.JsxFragmentFactory)
}
}
if options.ReactNamespace != "" {
if options.Jsx == core.JsxEmitReactJSX || options.Jsx == core.JsxEmitReactJSXDev {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_when_option_jsx_is_1, "reactNamespace", tsoptions.InverseJsxOptionMap.GetOrZero(options.Jsx))
}
}
if options.JsxImportSource != "" {
if options.Jsx == core.JsxEmitReact {
createDiagnosticForOptionName(diagnostics.Option_0_cannot_be_specified_when_option_jsx_is_1, "jsxImportSource", tsoptions.InverseJsxOptionMap.GetOrZero(options.Jsx))
}
}
moduleKind := options.GetEmitModuleKind()
if options.AllowImportingTsExtensions.IsTrue() && !(options.NoEmit.IsTrue() || options.EmitDeclarationOnly.IsTrue() || options.RewriteRelativeImportExtensions.IsTrue()) {
createOptionValueDiagnostic("allowImportingTsExtensions", diagnostics.Option_allowImportingTsExtensions_can_only_be_used_when_either_noEmit_or_emitDeclarationOnly_is_set)
}
moduleResolution := options.GetModuleResolutionKind()
if options.ResolvePackageJsonExports.IsTrue() && !moduleResolutionSupportsPackageJsonExportsAndImports(moduleResolution) {
createDiagnosticForOptionName(diagnostics.Option_0_can_only_be_used_when_moduleResolution_is_set_to_node16_nodenext_or_bundler, "resolvePackageJsonExports", "")
}
if options.ResolvePackageJsonImports.IsTrue() && !moduleResolutionSupportsPackageJsonExportsAndImports(moduleResolution) {
createDiagnosticForOptionName(diagnostics.Option_0_can_only_be_used_when_moduleResolution_is_set_to_node16_nodenext_or_bundler, "resolvePackageJsonImports", "")
}
if options.CustomConditions != nil && !moduleResolutionSupportsPackageJsonExportsAndImports(moduleResolution) {
createDiagnosticForOptionName(diagnostics.Option_0_can_only_be_used_when_moduleResolution_is_set_to_node16_nodenext_or_bundler, "customConditions", "")
}
// !!! Reenable once we don't map old moduleResolution kinds to bundler.
// if moduleResolution == core.ModuleResolutionKindBundler && !emitModuleKindIsNonNodeESM(moduleKind) && moduleKind != core.ModuleKindPreserve {
// createOptionValueDiagnostic("moduleResolution", diagnostics.Option_0_can_only_be_used_when_module_is_set_to_preserve_or_to_es2015_or_later, "bundler")
// }
if core.ModuleKindNode16 <= moduleKind && moduleKind <= core.ModuleKindNodeNext &&
!(core.ModuleResolutionKindNode16 <= moduleResolution && moduleResolution <= core.ModuleResolutionKindNodeNext) {
moduleKindName := moduleKind.String()
var moduleResolutionName string
if v, ok := core.ModuleKindToModuleResolutionKind[moduleKind]; ok {
moduleResolutionName = v.String()
} else {
moduleResolutionName = "Node16"
}
createOptionValueDiagnostic("moduleResolution", diagnostics.Option_moduleResolution_must_be_set_to_0_or_left_unspecified_when_option_module_is_set_to_1, moduleResolutionName, moduleKindName)
} else if core.ModuleResolutionKindNode16 <= moduleResolution && moduleResolution <= core.ModuleResolutionKindNodeNext &&
!(core.ModuleKindNode16 <= moduleKind && moduleKind <= core.ModuleKindNodeNext) {
moduleResolutionName := moduleResolution.String()
createOptionValueDiagnostic("module", diagnostics.Option_module_must_be_set_to_0_when_option_moduleResolution_is_set_to_1, moduleResolutionName, moduleResolutionName)
}
// !!! The below needs filesByName, which is not equivalent to p.filesByPath.
// If the emit is enabled make sure that every output file is unique and not overwriting any of the input files
if !options.NoEmit.IsTrue() && !options.SuppressOutputPathCheck.IsTrue() {
var emitFilesSeen collections.Set[string]
// Verify that all the emit files are unique and don't overwrite input files
verifyEmitFilePath := func(emitFileName string) {
if emitFileName != "" {
emitFilePath := p.toPath(emitFileName)
// Report error if the output overwrites input file
if _, ok := p.filesByPath[emitFilePath]; ok {
diag := ast.NewCompilerDiagnostic(diagnostics.Cannot_write_file_0_because_it_would_overwrite_input_file, emitFileName)
if configFilePath() == "" {
// The program is from either an inferred project or an external project
diag.AddMessageChain(ast.NewCompilerDiagnostic(diagnostics.Adding_a_tsconfig_json_file_will_help_organize_projects_that_contain_both_TypeScript_and_JavaScript_files_Learn_more_at_https_Colon_Slash_Slashaka_ms_Slashtsconfig))
}
p.blockEmittingOfFile(emitFileName, diag)
}
var emitFileKey string
if !p.Host().FS().UseCaseSensitiveFileNames() {
emitFileKey = tspath.ToFileNameLowerCase(string(emitFilePath))
} else {
emitFileKey = string(emitFilePath)
}
// Report error if multiple files write into same file
if emitFilesSeen.Has(emitFileKey) {
// Already seen the same emit file - report error
p.blockEmittingOfFile(emitFileName, ast.NewCompilerDiagnostic(diagnostics.Cannot_write_file_0_because_it_would_be_overwritten_by_multiple_input_files, emitFileName))
} else {
emitFilesSeen.Add(emitFileKey)
}
}
}
outputpaths.ForEachEmittedFile(p, options, func(emitFileNames *outputpaths.OutputPaths, sourceFile *ast.SourceFile) bool {
verifyEmitFilePath(emitFileNames.JsFilePath())
verifyEmitFilePath(emitFileNames.SourceMapFilePath())
verifyEmitFilePath(emitFileNames.DeclarationFilePath())
verifyEmitFilePath(emitFileNames.DeclarationMapPath())
return false
}, p.getSourceFilesToEmit(nil, false), false)
verifyEmitFilePath(p.opts.Config.GetBuildInfoFileName())
}
}
func (p *Program) blockEmittingOfFile(emitFileName string, diag *ast.Diagnostic) {
p.hasEmitBlockingDiagnostics.Add(p.toPath(emitFileName))
p.programDiagnostics = append(p.programDiagnostics, diag)
}
func (p *Program) IsEmitBlocked(emitFileName string) bool {
return p.hasEmitBlockingDiagnostics.Has(p.toPath(emitFileName))
}
func (p *Program) verifyProjectReferences() {
buildInfoFileName := core.IfElse(!p.Options().SuppressOutputPathCheck.IsTrue(), p.opts.Config.GetBuildInfoFileName(), "")
createDiagnosticForReference := func(config *tsoptions.ParsedCommandLine, index int, message *diagnostics.Message, args ...any) {
diag := tsoptions.CreateDiagnosticAtReferenceSyntax(config, index, message, args...)
if diag == nil {
diag = ast.NewCompilerDiagnostic(message, args...)
}
p.programDiagnostics = append(p.programDiagnostics, diag)
}
p.ForEachResolvedProjectReference(func(path tspath.Path, config *tsoptions.ParsedCommandLine, parent *tsoptions.ParsedCommandLine, index int) {
ref := parent.ProjectReferences()[index]
// !!! Deprecated in 5.0 and removed since 5.5
// verifyRemovedProjectReference(ref, parent, index);
if config == nil {
createDiagnosticForReference(parent, index, diagnostics.File_0_not_found, ref.Path)
return
}
refOptions := config.CompilerOptions()
if !refOptions.Composite.IsTrue() || refOptions.NoEmit.IsTrue() {
if len(parent.FileNames()) > 0 {
if !refOptions.Composite.IsTrue() {
createDiagnosticForReference(parent, index, diagnostics.Referenced_project_0_must_have_setting_composite_Colon_true, ref.Path)
}
if refOptions.NoEmit.IsTrue() {
createDiagnosticForReference(parent, index, diagnostics.Referenced_project_0_may_not_disable_emit, ref.Path)
}
}
}
if buildInfoFileName != "" && buildInfoFileName == config.GetBuildInfoFileName() {
createDiagnosticForReference(parent, index, diagnostics.Cannot_write_file_0_because_it_will_overwrite_tsbuildinfo_file_generated_by_referenced_project_1, buildInfoFileName, ref.Path)
p.hasEmitBlockingDiagnostics.Add(p.toPath(buildInfoFileName))
}
})
}
func hasZeroOrOneAsteriskCharacter(str string) bool {
seenAsterisk := false
for _, ch := range str {
if ch == '*' {
if !seenAsterisk {
seenAsterisk = true
} else {
// have already seen asterisk
return false
}
}
}
return true
}
func moduleResolutionSupportsPackageJsonExportsAndImports(moduleResolution core.ModuleResolutionKind) bool {
return moduleResolution >= core.ModuleResolutionKindNode16 && moduleResolution <= core.ModuleResolutionKindNodeNext ||
moduleResolution == core.ModuleResolutionKindBundler
}
func emitModuleKindIsNonNodeESM(moduleKind core.ModuleKind) bool {
return moduleKind >= core.ModuleKindES2015 && moduleKind <= core.ModuleKindESNext
}
func (p *Program) GetGlobalDiagnostics(ctx context.Context) []*ast.Diagnostic {
if len(p.files) == 0 {
return nil
}
var globalDiagnostics []*ast.Diagnostic
checkers, done := p.checkerPool.GetAllCheckers(ctx)
defer done()
for _, checker := range checkers {
globalDiagnostics = append(globalDiagnostics, checker.GetGlobalDiagnostics()...)
}
return SortAndDeduplicateDiagnostics(globalDiagnostics)
}
func (p *Program) GetDeclarationDiagnostics(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
return p.getDiagnosticsHelper(ctx, sourceFile, true /*ensureBound*/, true /*ensureChecked*/, p.getDeclarationDiagnosticsForFile)
}
func (p *Program) GetOptionsDiagnostics(ctx context.Context) []*ast.Diagnostic {
return SortAndDeduplicateDiagnostics(append(p.GetGlobalDiagnostics(ctx), p.getOptionsDiagnosticsOfConfigFile()...))
}
func (p *Program) getOptionsDiagnosticsOfConfigFile() []*ast.Diagnostic {
// todo update p.configParsingDiagnostics when updateAndGetProgramDiagnostics is implemented
if p.Options() == nil || p.Options().ConfigFilePath == "" {
return nil
}
return p.GetConfigFileParsingDiagnostics() // TODO: actually call getDiagnosticsHelper on config path
}
func (p *Program) getSyntacticDiagnosticsForFile(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
return core.Concatenate(sourceFile.Diagnostics(), sourceFile.JSDiagnostics())
}
func (p *Program) getBindDiagnosticsForFile(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
// TODO: restore this; tsgo's main depends on this function binding all files for timing.
// if checker.SkipTypeChecking(sourceFile, p.compilerOptions) {
// return nil
// }
return sourceFile.BindDiagnostics()
}
func FilterNoEmitSemanticDiagnostics(diagnostics []*ast.Diagnostic, options *core.CompilerOptions) []*ast.Diagnostic {
if !options.NoEmit.IsTrue() {
return diagnostics
}
return core.Filter(diagnostics, func(d *ast.Diagnostic) bool {
return !d.SkippedOnNoEmit()
})
}
func (p *Program) getSemanticDiagnosticsForFile(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
return slices.Concat(
FilterNoEmitSemanticDiagnostics(p.getSemanticDiagnosticsForFileNotFilter(ctx, sourceFile), p.Options()),
p.GetIncludeProcessorDiagnostics(sourceFile),
)
}
func (p *Program) getSemanticDiagnosticsForFileNotFilter(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
compilerOptions := p.Options()
if checker.SkipTypeChecking(sourceFile, compilerOptions, p, false) {
return nil
}
var fileChecker *checker.Checker
var done func()
if sourceFile != nil {
fileChecker, done = p.checkerPool.GetCheckerForFile(ctx, sourceFile)
defer done()
}
diags := slices.Clip(sourceFile.BindDiagnostics())
checkers, closeCheckers := p.checkerPool.GetAllCheckers(ctx)
defer closeCheckers()
// Ask for diags from all checkers; checking one file may add diagnostics to other files.
// These are deduplicated later.
for _, checker := range checkers {
if sourceFile == nil || checker == fileChecker {
diags = append(diags, checker.GetDiagnostics(ctx, sourceFile)...)
} else {
diags = append(diags, checker.GetDiagnosticsWithoutCheck(sourceFile)...)
}
}
if ctx.Err() != nil {
return nil
}
// !!! This should be rewritten to work like getBindAndCheckDiagnosticsForFileNoCache.
isPlainJS := ast.IsPlainJSFile(sourceFile, compilerOptions.CheckJs)
if isPlainJS {
return core.Filter(diags, func(d *ast.Diagnostic) bool {
return plainJSErrors.Has(d.Code())
})
}
filtered, directivesByLine := p.getDiagnosticsWithPrecedingDirectives(sourceFile, diags)
for _, directive := range directivesByLine {
// Above we changed all used directive kinds to @ts-ignore, so any @ts-expect-error directives that
// remain are unused and thus errors.
if directive.Kind == ast.CommentDirectiveKindExpectError {
filtered = append(filtered, ast.NewDiagnostic(sourceFile, directive.Loc, diagnostics.Unused_ts_expect_error_directive))
}
}
return filtered
}
func (p *Program) getDiagnosticsWithPrecedingDirectives(sourceFile *ast.SourceFile, diags []*ast.Diagnostic) ([]*ast.Diagnostic, map[int]ast.CommentDirective) {
if len(sourceFile.CommentDirectives) == 0 {
return diags, nil
}
// Build map of directives by line number
directivesByLine := make(map[int]ast.CommentDirective)
for _, directive := range sourceFile.CommentDirectives {
line, _ := scanner.GetECMALineAndCharacterOfPosition(sourceFile, directive.Loc.Pos())
directivesByLine[line] = directive
}
lineStarts := scanner.GetECMALineStarts(sourceFile)
filtered := make([]*ast.Diagnostic, 0, len(diags))
for _, diagnostic := range diags {
ignoreDiagnostic := false
for line := scanner.ComputeLineOfPosition(lineStarts, diagnostic.Pos()) - 1; line >= 0; line-- {
// If line contains a @ts-ignore or @ts-expect-error directive, ignore this diagnostic and change
// the directive kind to @ts-ignore to indicate it was used.
if directive, ok := directivesByLine[line]; ok {
ignoreDiagnostic = true
directive.Kind = ast.CommentDirectiveKindIgnore
directivesByLine[line] = directive
break
}
// Stop searching backwards when we encounter a line that isn't blank or a comment.
if !isCommentOrBlankLine(sourceFile.Text(), int(lineStarts[line])) {
break
}
}
if !ignoreDiagnostic {
filtered = append(filtered, diagnostic)
}
}
return filtered, directivesByLine
}
func (p *Program) getDeclarationDiagnosticsForFile(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
if sourceFile.IsDeclarationFile {
return []*ast.Diagnostic{}
}
if cached, ok := p.declarationDiagnosticCache.Load(sourceFile); ok {
return cached
}
host, done := newEmitHost(ctx, p, sourceFile)
defer done()
diagnostics := getDeclarationDiagnostics(host, sourceFile)
diagnostics, _ = p.declarationDiagnosticCache.LoadOrStore(sourceFile, diagnostics)
return diagnostics
}
func (p *Program) getSuggestionDiagnosticsForFile(ctx context.Context, sourceFile *ast.SourceFile) []*ast.Diagnostic {
if checker.SkipTypeChecking(sourceFile, p.Options(), p, false) {
return nil
}
var fileChecker *checker.Checker
var done func()
if sourceFile != nil {
fileChecker, done = p.checkerPool.GetCheckerForFile(ctx, sourceFile)
defer done()
}
diags := slices.Clip(sourceFile.BindSuggestionDiagnostics)
checkers, closeCheckers := p.checkerPool.GetAllCheckers(ctx)
defer closeCheckers()
// Ask for diags from all checkers; checking one file may add diagnostics to other files.
// These are deduplicated later.
for _, checker := range checkers {
if sourceFile == nil || checker == fileChecker {
diags = append(diags, checker.GetSuggestionDiagnostics(ctx, sourceFile)...)
} else {
// !!! is there any case where suggestion diagnostics are produced in other checkers?
}
}
if ctx.Err() != nil {
return nil
}
return diags
}
func isCommentOrBlankLine(text string, pos int) bool {
for pos < len(text) && (text[pos] == ' ' || text[pos] == '\t') {
pos++
}
return pos == len(text) ||
pos < len(text) && (text[pos] == '\r' || text[pos] == '\n') ||
pos+1 < len(text) && text[pos] == '/' && text[pos+1] == '/'
}
func SortAndDeduplicateDiagnostics(diagnostics []*ast.Diagnostic) []*ast.Diagnostic {
diagnostics = slices.Clone(diagnostics)
slices.SortFunc(diagnostics, ast.CompareDiagnostics)
return compactAndMergeRelatedInfos(diagnostics)
}
// Remove duplicate diagnostics and, for sequences of diagnostics that differ only by related information,
// create a single diagnostic with sorted and deduplicated related information.
func compactAndMergeRelatedInfos(diagnostics []*ast.Diagnostic) []*ast.Diagnostic {
if len(diagnostics) < 2 {
return diagnostics
}
i := 0
j := 0
for i < len(diagnostics) {
d := diagnostics[i]
n := 1
for i+n < len(diagnostics) && ast.EqualDiagnosticsNoRelatedInfo(d, diagnostics[i+n]) {
n++
}
if n > 1 {
var relatedInfos []*ast.Diagnostic
for k := range n {
relatedInfos = append(relatedInfos, diagnostics[i+k].RelatedInformation()...)
}
if relatedInfos != nil {
slices.SortFunc(relatedInfos, ast.CompareDiagnostics)
relatedInfos = slices.CompactFunc(relatedInfos, ast.EqualDiagnostics)
d = d.Clone().SetRelatedInfo(relatedInfos)
}
}
diagnostics[j] = d
i += n
j++
}
clear(diagnostics[j:])
return diagnostics[:j]
}
func (p *Program) getDiagnosticsHelper(ctx context.Context, sourceFile *ast.SourceFile, ensureBound bool, ensureChecked bool, getDiagnostics func(context.Context, *ast.SourceFile) []*ast.Diagnostic) []*ast.Diagnostic {
if sourceFile != nil {
if ensureBound {
binder.BindSourceFile(sourceFile)
}
return SortAndDeduplicateDiagnostics(getDiagnostics(ctx, sourceFile))
}
if ensureBound {
p.BindSourceFiles()
}
if ensureChecked {
p.CheckSourceFiles(ctx, nil)
if ctx.Err() != nil {
return nil
}
}
var result []*ast.Diagnostic
for _, file := range p.files {
result = append(result, getDiagnostics(ctx, file)...)
}
return SortAndDeduplicateDiagnostics(result)
}
func (p *Program) LineCount() int {
var count int
for _, file := range p.files {
count += len(file.ECMALineMap())
}
return count
}
func (p *Program) IdentifierCount() int {
var count int
for _, file := range p.files {
count += file.IdentifierCount
}
return count
}
func (p *Program) SymbolCount() int {
var count int
for _, file := range p.files {
count += file.SymbolCount
}
checkers, done := p.checkerPool.GetAllCheckers(context.Background())
defer done()
for _, checker := range checkers {
count += int(checker.SymbolCount)
}
return count
}
func (p *Program) TypeCount() int {
var count int
checkers, done := p.checkerPool.GetAllCheckers(context.Background())
defer done()
for _, checker := range checkers {
count += int(checker.TypeCount)
}
return count
}
func (p *Program) InstantiationCount() int {
var count int
checkers, done := p.checkerPool.GetAllCheckers(context.Background())
defer done()
for _, checker := range checkers {
count += int(checker.TotalInstantiationCount)
}
return count
}
func (p *Program) GetSourceFileMetaData(path tspath.Path) ast.SourceFileMetaData {
return p.sourceFileMetaDatas[path]
}
func (p *Program) GetEmitModuleFormatOfFile(sourceFile ast.HasFileName) core.ModuleKind {
return ast.GetEmitModuleFormatOfFileWorker(sourceFile.FileName(), p.projectReferenceFileMapper.getCompilerOptionsForFile(sourceFile), p.GetSourceFileMetaData(sourceFile.Path()))
}
func (p *Program) GetEmitSyntaxForUsageLocation(sourceFile ast.HasFileName, location *ast.StringLiteralLike) core.ResolutionMode {
return getEmitSyntaxForUsageLocationWorker(sourceFile.FileName(), p.sourceFileMetaDatas[sourceFile.Path()], location, p.projectReferenceFileMapper.getCompilerOptionsForFile(sourceFile))
}
func (p *Program) GetImpliedNodeFormatForEmit(sourceFile ast.HasFileName) core.ResolutionMode {
return ast.GetImpliedNodeFormatForEmitWorker(sourceFile.FileName(), p.projectReferenceFileMapper.getCompilerOptionsForFile(sourceFile).GetEmitModuleKind(), p.GetSourceFileMetaData(sourceFile.Path()))
}
func (p *Program) GetModeForUsageLocation(sourceFile ast.HasFileName, location *ast.StringLiteralLike) core.ResolutionMode {
return getModeForUsageLocation(sourceFile.FileName(), p.sourceFileMetaDatas[sourceFile.Path()], location, p.projectReferenceFileMapper.getCompilerOptionsForFile(sourceFile))
}
func (p *Program) GetDefaultResolutionModeForFile(sourceFile ast.HasFileName) core.ResolutionMode {
return getDefaultResolutionModeForFile(sourceFile.FileName(), p.sourceFileMetaDatas[sourceFile.Path()], p.projectReferenceFileMapper.getCompilerOptionsForFile(sourceFile))
}
func (p *Program) IsSourceFileDefaultLibrary(path tspath.Path) bool {
_, ok := p.libFiles[path]
return ok
}
func (p *Program) GetDefaultLibFile(path tspath.Path) *LibFile {
if libFile, ok := p.libFiles[path]; ok {
return libFile
}
return nil
}
func (p *Program) CommonSourceDirectory() string {
p.commonSourceDirectoryOnce.Do(func() {
p.commonSourceDirectory = outputpaths.GetCommonSourceDirectory(
p.Options(),
func() []string {
var files []string
for _, file := range p.files {
if sourceFileMayBeEmitted(file, p, false /*forceDtsEmit*/) {
files = append(files, file.FileName())
}
}
return files
},
p.GetCurrentDirectory(),
p.UseCaseSensitiveFileNames(),
)
})
return p.commonSourceDirectory
}
type WriteFileData struct {
SourceMapUrlPos int
BuildInfo any
Diagnostics []*ast.Diagnostic
SkippedDtsWrite bool
}
type WriteFile func(fileName string, text string, writeByteOrderMark bool, data *WriteFileData) error
type EmitOptions struct {
TargetSourceFile *ast.SourceFile // Single file to emit. If `nil`, emits all files
EmitOnly EmitOnly
WriteFile WriteFile
}
type EmitResult struct {
EmitSkipped bool
Diagnostics []*ast.Diagnostic // Contains declaration emit diagnostics
EmittedFiles []string // Array of files the compiler wrote to disk
SourceMaps []*SourceMapEmitResult // Array of sourceMapData if compiler emitted sourcemaps
}
type SourceMapEmitResult struct {
InputSourceFileNames []string // Input source file (which one can use on program to get the file), 1:1 mapping with the sourceMap.sources list
SourceMap *sourcemap.RawSourceMap
GeneratedFile string
}
func (p *Program) Emit(ctx context.Context, options EmitOptions) *EmitResult {
// !!! performance measurement
p.BindSourceFiles()
if options.EmitOnly != EmitOnlyForcedDts {
result := HandleNoEmitOnError(
ctx,
p,
options.TargetSourceFile,
)
if result != nil || ctx.Err() != nil {
return result
}
}
writerPool := &sync.Pool{
New: func() any {
return printer.NewTextWriter(p.Options().NewLine.GetNewLineCharacter())
},
}
wg := core.NewWorkGroup(p.SingleThreaded())
var emitters []*emitter
sourceFiles := p.getSourceFilesToEmit(options.TargetSourceFile, options.EmitOnly == EmitOnlyForcedDts)
for _, sourceFile := range sourceFiles {
emitter := &emitter{
writer: nil,
sourceFile: sourceFile,
emitOnly: options.EmitOnly,
writeFile: options.WriteFile,
}
emitters = append(emitters, emitter)
wg.Queue(func() {
host, done := newEmitHost(ctx, p, sourceFile)
defer done()
emitter.host = host
// take an unused writer
writer := writerPool.Get().(printer.EmitTextWriter)
writer.Clear()
// attach writer and perform emit
emitter.writer = writer
emitter.paths = outputpaths.GetOutputPathsFor(sourceFile, host.Options(), host, options.EmitOnly == EmitOnlyForcedDts)
emitter.emit()
emitter.writer = nil
// put the writer back in the pool
writerPool.Put(writer)
})
}
// wait for emit to complete
wg.RunAndWait()
// collect results from emit, preserving input order
return CombineEmitResults(core.Map(emitters, func(e *emitter) *EmitResult {
return &e.emitResult
}))
}
func CombineEmitResults(results []*EmitResult) *EmitResult {
result := &EmitResult{}
for _, emitResult := range results {
if emitResult == nil {
continue // Skip nil results
}
if emitResult.EmitSkipped {
result.EmitSkipped = true
}
result.Diagnostics = append(result.Diagnostics, emitResult.Diagnostics...)
result.EmittedFiles = append(result.EmittedFiles, emitResult.EmittedFiles...)
if emitResult.SourceMaps != nil {
result.SourceMaps = append(result.SourceMaps, emitResult.SourceMaps...)
}
}
return result
}
type ProgramLike interface {
Options() *core.CompilerOptions
GetSourceFiles() []*ast.SourceFile
GetConfigFileParsingDiagnostics() []*ast.Diagnostic
GetSyntacticDiagnostics(ctx context.Context, file *ast.SourceFile) []*ast.Diagnostic
GetBindDiagnostics(ctx context.Context, file *ast.SourceFile) []*ast.Diagnostic
GetOptionsDiagnostics(ctx context.Context) []*ast.Diagnostic
GetProgramDiagnostics() []*ast.Diagnostic
GetGlobalDiagnostics(ctx context.Context) []*ast.Diagnostic
GetSemanticDiagnostics(ctx context.Context, file *ast.SourceFile) []*ast.Diagnostic
GetDeclarationDiagnostics(ctx context.Context, file *ast.SourceFile) []*ast.Diagnostic
Emit(ctx context.Context, options EmitOptions) *EmitResult
}
func HandleNoEmitOnError(ctx context.Context, program ProgramLike, file *ast.SourceFile) *EmitResult {
if !program.Options().NoEmitOnError.IsTrue() {
return nil // No emit on error is not set, so we can proceed with emitting
}
diagnostics := GetDiagnosticsOfAnyProgram(
ctx,
program,
file,
true,
program.GetBindDiagnostics,
program.GetSemanticDiagnostics,
)
if len(diagnostics) == 0 {
return nil // No diagnostics, so we can proceed with emitting
}
return &EmitResult{
Diagnostics: diagnostics,
EmitSkipped: true,
}
}
func GetDiagnosticsOfAnyProgram(
ctx context.Context,
program ProgramLike,
file *ast.SourceFile,
skipNoEmitCheckForDtsDiagnostics bool,
getBindDiagnostics func(context.Context, *ast.SourceFile) []*ast.Diagnostic,
getSemanticDiagnostics func(context.Context, *ast.SourceFile) []*ast.Diagnostic,
) []*ast.Diagnostic {
allDiagnostics := slices.Clip(program.GetConfigFileParsingDiagnostics())
configFileParsingDiagnosticsLength := len(allDiagnostics)
allDiagnostics = append(allDiagnostics, program.GetSyntacticDiagnostics(ctx, file)...)
allDiagnostics = append(allDiagnostics, program.GetProgramDiagnostics()...)
if len(allDiagnostics) == configFileParsingDiagnosticsLength {
// Options diagnostics include global diagnostics (even though we collect them separately),
// and global diagnostics create checkers, which then bind all of the files. Do this binding
// early so we can track the time.
getBindDiagnostics(ctx, file)
allDiagnostics = append(allDiagnostics, program.GetOptionsDiagnostics(ctx)...)
if program.Options().ListFilesOnly.IsFalseOrUnknown() {
allDiagnostics = append(allDiagnostics, program.GetGlobalDiagnostics(ctx)...)
if len(allDiagnostics) == configFileParsingDiagnosticsLength {
allDiagnostics = append(allDiagnostics, getSemanticDiagnostics(ctx, file)...)
}
if (skipNoEmitCheckForDtsDiagnostics || program.Options().NoEmit.IsTrue()) && program.Options().GetEmitDeclarations() && len(allDiagnostics) == configFileParsingDiagnosticsLength {
allDiagnostics = append(allDiagnostics, program.GetDeclarationDiagnostics(ctx, file)...)
}
}
}
return allDiagnostics
}
func (p *Program) toPath(filename string) tspath.Path {
return tspath.ToPath(filename, p.GetCurrentDirectory(), p.UseCaseSensitiveFileNames())
}
func (p *Program) GetSourceFile(filename string) *ast.SourceFile {
path := p.toPath(filename)
return p.GetSourceFileByPath(path)
}
func (p *Program) GetSourceFileForResolvedModule(fileName string) *ast.SourceFile {
file := p.GetSourceFile(fileName)
if file == nil {
filename := p.GetParseFileRedirect(fileName)
if filename != "" {
return p.GetSourceFile(filename)
}
}
return file
}
func (p *Program) GetSourceFileByPath(path tspath.Path) *ast.SourceFile {
return p.filesByPath[path]
}
func (p *Program) HasSameFileNames(other *Program) bool {
return maps.EqualFunc(p.filesByPath, other.filesByPath, func(a, b *ast.SourceFile) bool {
// checks for casing differences on case-insensitive file systems
return a.FileName() == b.FileName()
})
}
func (p *Program) GetSourceFiles() []*ast.SourceFile {
return p.files
}
// Testing only
func (p *Program) GetIncludeReasons() map[tspath.Path][]*fileIncludeReason {
return p.includeProcessor.fileIncludeReasons
}
// Testing only
func (p *Program) IsMissingPath(path tspath.Path) bool {
return slices.ContainsFunc(p.missingFiles, func(missingPath string) bool {
return p.toPath(missingPath) == path
})
}
func (p *Program) ExplainFiles(w io.Writer) {
toRelativeFileName := func(fileName string) string {
return tspath.GetRelativePathFromDirectory(p.GetCurrentDirectory(), fileName, p.comparePathsOptions)
}
for _, file := range p.GetSourceFiles() {
fmt.Fprintln(w, toRelativeFileName(file.FileName()))
for _, reason := range p.includeProcessor.fileIncludeReasons[file.Path()] {
fmt.Fprintln(w, " ", reason.toDiagnostic(p, true).Message())
}
for _, diag := range p.includeProcessor.explainRedirectAndImpliedFormat(p, file, toRelativeFileName) {
fmt.Fprintln(w, " ", diag.Message())
}
}
}
func (p *Program) GetLibFileFromReference(ref *ast.FileReference) *ast.SourceFile {
path, ok := tsoptions.GetLibFileName(ref.FileName)
if !ok {
return nil
}
if sourceFile, ok := p.filesByPath[tspath.Path(path)]; ok {
return sourceFile
}
return nil
}
func (p *Program) GetResolvedTypeReferenceDirectiveFromTypeReferenceDirective(typeRef *ast.FileReference, sourceFile *ast.SourceFile) *module.ResolvedTypeReferenceDirective {
if resolutions, ok := p.typeResolutionsInFile[sourceFile.Path()]; ok {
if resolved, ok := resolutions[module.ModeAwareCacheKey{Name: typeRef.FileName, Mode: p.getModeForTypeReferenceDirectiveInFile(typeRef, sourceFile)}]; ok {
return resolved
}
}
return nil
}
func (p *Program) GetResolvedTypeReferenceDirectives() map[tspath.Path]module.ModeAwareCache[*module.ResolvedTypeReferenceDirective] {
return p.typeResolutionsInFile
}
func (p *Program) getModeForTypeReferenceDirectiveInFile(ref *ast.FileReference, sourceFile *ast.SourceFile) core.ResolutionMode {
if ref.ResolutionMode != core.ResolutionModeNone {
return ref.ResolutionMode
}
return p.GetDefaultResolutionModeForFile(sourceFile)
}
func (p *Program) IsSourceFileFromExternalLibrary(file *ast.SourceFile) bool {
return p.sourceFilesFoundSearchingNodeModules.Has(file.Path())
}
// UnsupportedExtensions returns a list of all present "unsupported" extensions,
// e.g. extensions that are not yet supported by the port.
func (p *Program) UnsupportedExtensions() []string {
return p.unsupportedExtensions
}
func (p *Program) GetJSXRuntimeImportSpecifier(path tspath.Path) (moduleReference string, specifier *ast.Node) {
if result := p.jsxRuntimeImportSpecifiers[path]; result != nil {
return result.moduleReference, result.specifier
}
return "", nil
}
func (p *Program) GetImportHelpersImportSpecifier(path tspath.Path) *ast.Node {
return p.importHelpersImportSpecifiers[path]
}
func (p *Program) SourceFileMayBeEmitted(sourceFile *ast.SourceFile, forceDtsEmit bool) bool {
return sourceFileMayBeEmitted(sourceFile, p, forceDtsEmit)
}
var plainJSErrors = collections.NewSetFromItems(
// binder errors
diagnostics.Cannot_redeclare_block_scoped_variable_0.Code(),
diagnostics.A_module_cannot_have_multiple_default_exports.Code(),
diagnostics.Another_export_default_is_here.Code(),
diagnostics.The_first_export_default_is_here.Code(),
diagnostics.Identifier_expected_0_is_a_reserved_word_at_the_top_level_of_a_module.Code(),
diagnostics.Identifier_expected_0_is_a_reserved_word_in_strict_mode_Modules_are_automatically_in_strict_mode.Code(),
diagnostics.Identifier_expected_0_is_a_reserved_word_that_cannot_be_used_here.Code(),
diagnostics.X_constructor_is_a_reserved_word.Code(),
diagnostics.X_delete_cannot_be_called_on_an_identifier_in_strict_mode.Code(),
diagnostics.Code_contained_in_a_class_is_evaluated_in_JavaScript_s_strict_mode_which_does_not_allow_this_use_of_0_For_more_information_see_https_Colon_Slash_Slashdeveloper_mozilla_org_Slashen_US_Slashdocs_SlashWeb_SlashJavaScript_SlashReference_SlashStrict_mode.Code(),
diagnostics.Invalid_use_of_0_Modules_are_automatically_in_strict_mode.Code(),
diagnostics.Invalid_use_of_0_in_strict_mode.Code(),
diagnostics.A_label_is_not_allowed_here.Code(),
diagnostics.X_with_statements_are_not_allowed_in_strict_mode.Code(),
// grammar errors
diagnostics.A_break_statement_can_only_be_used_within_an_enclosing_iteration_or_switch_statement.Code(),
diagnostics.A_break_statement_can_only_jump_to_a_label_of_an_enclosing_statement.Code(),
diagnostics.A_class_declaration_without_the_default_modifier_must_have_a_name.Code(),
diagnostics.A_class_member_cannot_have_the_0_keyword.Code(),
diagnostics.A_comma_expression_is_not_allowed_in_a_computed_property_name.Code(),
diagnostics.A_continue_statement_can_only_be_used_within_an_enclosing_iteration_statement.Code(),
diagnostics.A_continue_statement_can_only_jump_to_a_label_of_an_enclosing_iteration_statement.Code(),
diagnostics.A_default_clause_cannot_appear_more_than_once_in_a_switch_statement.Code(),
diagnostics.A_default_export_must_be_at_the_top_level_of_a_file_or_module_declaration.Code(),
diagnostics.A_definite_assignment_assertion_is_not_permitted_in_this_context.Code(),
diagnostics.A_destructuring_declaration_must_have_an_initializer.Code(),
diagnostics.A_get_accessor_cannot_have_parameters.Code(),
diagnostics.A_rest_element_cannot_contain_a_binding_pattern.Code(),
diagnostics.A_rest_element_cannot_have_a_property_name.Code(),
diagnostics.A_rest_element_cannot_have_an_initializer.Code(),
diagnostics.A_rest_element_must_be_last_in_a_destructuring_pattern.Code(),
diagnostics.A_rest_parameter_cannot_have_an_initializer.Code(),
diagnostics.A_rest_parameter_must_be_last_in_a_parameter_list.Code(),
diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma.Code(),
diagnostics.A_return_statement_cannot_be_used_inside_a_class_static_block.Code(),
diagnostics.A_set_accessor_cannot_have_rest_parameter.Code(),
diagnostics.A_set_accessor_must_have_exactly_one_parameter.Code(),
diagnostics.An_export_declaration_can_only_be_used_at_the_top_level_of_a_module.Code(),
diagnostics.An_export_declaration_cannot_have_modifiers.Code(),
diagnostics.An_import_declaration_can_only_be_used_at_the_top_level_of_a_module.Code(),
diagnostics.An_import_declaration_cannot_have_modifiers.Code(),
diagnostics.An_object_member_cannot_be_declared_optional.Code(),
diagnostics.Argument_of_dynamic_import_cannot_be_spread_element.Code(),
diagnostics.Cannot_assign_to_private_method_0_Private_methods_are_not_writable.Code(),
diagnostics.Cannot_redeclare_identifier_0_in_catch_clause.Code(),
diagnostics.Catch_clause_variable_cannot_have_an_initializer.Code(),
diagnostics.Class_decorators_can_t_be_used_with_static_private_identifier_Consider_removing_the_experimental_decorator.Code(),
diagnostics.Classes_can_only_extend_a_single_class.Code(),
diagnostics.Classes_may_not_have_a_field_named_constructor.Code(),
diagnostics.Did_you_mean_to_use_a_Colon_An_can_only_follow_a_property_name_when_the_containing_object_literal_is_part_of_a_destructuring_pattern.Code(),
diagnostics.Duplicate_label_0.Code(),
diagnostics.Dynamic_imports_can_only_accept_a_module_specifier_and_an_optional_set_of_attributes_as_arguments.Code(),
diagnostics.X_for_await_loops_cannot_be_used_inside_a_class_static_block.Code(),
diagnostics.JSX_attributes_must_only_be_assigned_a_non_empty_expression.Code(),
diagnostics.JSX_elements_cannot_have_multiple_attributes_with_the_same_name.Code(),
diagnostics.JSX_expressions_may_not_use_the_comma_operator_Did_you_mean_to_write_an_array.Code(),
diagnostics.JSX_property_access_expressions_cannot_include_JSX_namespace_names.Code(),
diagnostics.Jump_target_cannot_cross_function_boundary.Code(),
diagnostics.Line_terminator_not_permitted_before_arrow.Code(),
diagnostics.Modifiers_cannot_appear_here.Code(),
diagnostics.Only_a_single_variable_declaration_is_allowed_in_a_for_in_statement.Code(),
diagnostics.Only_a_single_variable_declaration_is_allowed_in_a_for_of_statement.Code(),
diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies.Code(),
diagnostics.Private_identifiers_are_only_allowed_in_class_bodies_and_may_only_be_used_as_part_of_a_class_member_declaration_property_access_or_on_the_left_hand_side_of_an_in_expression.Code(),
diagnostics.Property_0_is_not_accessible_outside_class_1_because_it_has_a_private_identifier.Code(),
diagnostics.Tagged_template_expressions_are_not_permitted_in_an_optional_chain.Code(),
diagnostics.The_left_hand_side_of_a_for_of_statement_may_not_be_async.Code(),
diagnostics.The_variable_declaration_of_a_for_in_statement_cannot_have_an_initializer.Code(),
diagnostics.The_variable_declaration_of_a_for_of_statement_cannot_have_an_initializer.Code(),
diagnostics.Trailing_comma_not_allowed.Code(),
diagnostics.Variable_declaration_list_cannot_be_empty.Code(),
diagnostics.X_0_and_1_operations_cannot_be_mixed_without_parentheses.Code(),
diagnostics.X_0_expected.Code(),
diagnostics.X_0_is_not_a_valid_meta_property_for_keyword_1_Did_you_mean_2.Code(),
diagnostics.X_0_list_cannot_be_empty.Code(),
diagnostics.X_0_modifier_already_seen.Code(),
diagnostics.X_0_modifier_cannot_appear_on_a_constructor_declaration.Code(),
diagnostics.X_0_modifier_cannot_appear_on_a_module_or_namespace_element.Code(),
diagnostics.X_0_modifier_cannot_appear_on_a_parameter.Code(),
diagnostics.X_0_modifier_cannot_appear_on_class_elements_of_this_kind.Code(),
diagnostics.X_0_modifier_cannot_be_used_here.Code(),
diagnostics.X_0_modifier_must_precede_1_modifier.Code(),
diagnostics.X_0_declarations_can_only_be_declared_inside_a_block.Code(),
diagnostics.X_0_declarations_must_be_initialized.Code(),
diagnostics.X_extends_clause_already_seen.Code(),
diagnostics.X_let_is_not_allowed_to_be_used_as_a_name_in_let_or_const_declarations.Code(),
diagnostics.Class_constructor_may_not_be_a_generator.Code(),
diagnostics.Class_constructor_may_not_be_an_accessor.Code(),
diagnostics.X_await_expressions_are_only_allowed_within_async_functions_and_at_the_top_levels_of_modules.Code(),
diagnostics.X_await_using_statements_are_only_allowed_within_async_functions_and_at_the_top_levels_of_modules.Code(),
diagnostics.Private_field_0_must_be_declared_in_an_enclosing_class.Code(),
// Type errors
diagnostics.This_condition_will_always_return_0_since_JavaScript_compares_objects_by_reference_not_value.Code(),
)