Scripting: improve reporting on cyclic dependency

#KT-48177 fixed
the issue is in fact fixed by the previous commit (report error
on duplicated import) by normalizing import path before processing,
but here we're making error reporting nicer for the case.
This commit is contained in:
Ilya Chernikov
2021-08-17 17:25:47 +02:00
committed by TeamCityServer
parent c204d7a86f
commit e60e80f19a
7 changed files with 86 additions and 51 deletions
@@ -150,25 +150,25 @@ open class KJvmReplCompilerBase<AnalyzerT : ReplCodeAnalyzerBase> protected cons
history.push(LineId(snippetNo, 0, snippet.hashCode()), scriptDescriptor)
val dependenciesProvider = ScriptDependenciesProvider.getInstance(context.environment.project)
val compiledScript =
makeCompiledScript(
generationState,
snippet,
sourceFiles.first(),
sourceDependencies
) { ktFile ->
dependenciesProvider?.getScriptConfiguration(ktFile)?.configuration
?: context.baseScriptCompilationConfiguration
}
makeCompiledScript(
generationState,
snippet,
sourceFiles.first(),
sourceDependencies
) { ktFile ->
dependenciesProvider?.getScriptConfiguration(ktFile)?.configuration
?: context.baseScriptCompilationConfiguration
}.onSuccess { compiledScript ->
lastCompiledSnippet = lastCompiledSnippet.add(compiledScript)
lastCompiledSnippet = lastCompiledSnippet.add(compiledScript)
lastCompiledSnippet?.asSuccess(messageCollector.diagnostics)
?: failure(
snippet,
messageCollector,
"last compiled snippet should not be null"
)
lastCompiledSnippet?.asSuccess(messageCollector.diagnostics)
?: failure(
snippet,
messageCollector,
"last compiled snippet should not be null"
)
}
}
}.last()
@@ -196,16 +196,15 @@ private fun doCompile(
val generationState =
generate(analysisResult, sourceFiles, context.environment.configuration)
val compiledScript =
makeCompiledScript(
generationState,
script,
sourceFiles.first(),
sourceDependencies,
getScriptConfiguration
)
return ResultWithDiagnostics.Success(compiledScript, messageCollector.diagnostics)
return makeCompiledScript(
generationState,
script,
sourceFiles.first(),
sourceDependencies,
getScriptConfiguration
).onSuccess { compiledScript ->
ResultWithDiagnostics.Success(compiledScript, messageCollector.diagnostics)
}
}
private fun analyze(sourceFiles: Collection<KtFile>, environment: KotlinCoreEnvironment): AnalysisResult {
@@ -116,32 +116,41 @@ internal fun makeCompiledScript(
ktFile: KtFile,
sourceDependencies: List<ScriptsCompilationDependencies.SourceDependencies>,
getScriptConfiguration: (KtFile) -> ScriptCompilationConfiguration
): KJvmCompiledScript {
): ResultWithDiagnostics<KJvmCompiledScript> {
val scriptDependenciesStack = ArrayDeque<KtScript>()
val ktScript = ktFile.declarations.firstIsInstanceOrNull<KtScript>()
?: throw IllegalStateException("Expecting script file: KtScript is not found in ${ktFile.name}")
fun makeOtherScripts(script: KtScript): List<KJvmCompiledScript> {
fun makeOtherScripts(script: KtScript): ResultWithDiagnostics<List<KJvmCompiledScript>> {
// TODO: ensure that it is caught earlier (as well) since it would be more economical
if (scriptDependenciesStack.contains(script))
throw IllegalArgumentException("Unable to handle recursive script dependencies")
return ResultWithDiagnostics.Failure(
ScriptDiagnostic(
ScriptDiagnostic.unspecifiedError,
"Unable to handle recursive script dependencies",
sourcePath = script.containingFile.virtualFile?.path
)
)
scriptDependenciesStack.push(script)
val containingKtFile = script.containingKtFile
val otherScripts: List<KJvmCompiledScript> =
sourceDependencies.find { it.scriptFile == containingKtFile }?.sourceDependencies?.valueOrThrow()?.mapNotNull { sourceFile ->
sourceFile.declarations.firstIsInstanceOrNull<KtScript>()?.let {
KJvmCompiledScript(
containingKtFile.virtualFilePath,
getScriptConfiguration(sourceFile),
it.fqName.asString(),
null,
makeOtherScripts(it),
null
)
}
} ?: emptyList()
val otherScripts =
sourceDependencies.find { it.scriptFile == containingKtFile }?.sourceDependencies?.valueOrThrow()
?.mapNotNullSuccess { sourceFile ->
sourceFile.declarations.firstIsInstanceOrNull<KtScript>()?.let { ktScript ->
makeOtherScripts(ktScript).onSuccess { otherScripts ->
KJvmCompiledScript(
containingKtFile.virtualFilePath,
getScriptConfiguration(sourceFile),
ktScript.fqName.asString(),
null,
otherScripts,
null
).asSuccess()
}
} ?: null.asSuccess()
} ?: emptyList<KJvmCompiledScript>().asSuccess()
scriptDependenciesStack.pop()
return otherScripts
@@ -154,13 +163,15 @@ internal fun makeCompiledScript(
else resultFieldName!! to KotlinType(resultTypeString ?: DescriptorRenderer.FQ_NAMES_IN_TYPES.renderType(resultType!!))
}
return KJvmCompiledScript(
script.locationId,
getScriptConfiguration(ktScript.containingKtFile),
ktScript.fqName.asString(),
resultField,
makeOtherScripts(ktScript),
module
)
return makeOtherScripts(ktScript).onSuccess { otherScripts ->
KJvmCompiledScript(
script.locationId,
getScriptConfiguration(ktScript.containingKtFile),
ktScript.fqName.asString(),
resultField,
otherScripts,
module
).asSuccess()
}
}