org.jetbrains.kotlin.gradle.tasks.tasksUtils.kt Maven / Gradle / Ivy
The newest version!
package org.jetbrains.kotlin.gradle.tasks
import org.jetbrains.kotlin.build.report.metrics.*
import org.jetbrains.kotlin.cli.common.ExitCode
import org.jetbrains.kotlin.buildtools.api.KotlinLogger
import org.jetbrains.kotlin.gradle.internal.tasks.TaskWithLocalState
import org.jetbrains.kotlin.gradle.internal.tasks.allOutputFiles
import org.jetbrains.kotlin.gradle.logging.GradleKotlinLogger
import org.jetbrains.kotlin.gradle.logging.kotlinDebug
import org.jetbrains.kotlin.incremental.deleteDirectoryContents
import org.jetbrains.kotlin.incremental.deleteRecursivelyOrThrow
import java.io.File
import java.rmi.RemoteException
/** Throws [FailedCompilationException] if compilation completed with [exitCode] != [ExitCode.OK]. */
fun throwExceptionIfCompilationFailed(
exitCode: ExitCode,
executionStrategy: KotlinCompilerExecutionStrategy
) {
when (exitCode) {
ExitCode.COMPILATION_ERROR -> throw CompilationErrorException("Compilation error. See log for more details")
ExitCode.INTERNAL_ERROR -> throw FailedCompilationException("Internal compiler error. See log for more details")
ExitCode.SCRIPT_EXECUTION_ERROR -> throw FailedCompilationException("Script execution error. See log for more details")
ExitCode.OOM_ERROR -> throw OOMErrorException(executionStrategy)
ExitCode.OK -> Unit
else -> throw IllegalStateException("Unexpected exit code: $exitCode")
}
}
internal const val kotlinDaemonOOMHelperMessage = "Not enough memory to run compilation. " +
"Try to increase it via 'gradle.properties':\nkotlin.daemon.jvmargs=-Xmx"
internal const val kotlinInProcessOOMHelperMessage = "Not enough memory to run compilation. " +
" Try to increase it via 'gradle.properties':\norg.gradle.jvmargs=-Xmx"
internal const val kotlinOutOfProcessOOMHelperMessage = "Not enough memory to run compilation."
private const val kotlinDaemonCrashedMessage =
"Connection to the Kotlin daemon has been unexpectedly lost. This might be caused by the daemon being killed by another process or the operating system, or by JVM crash."
internal fun Throwable.hasOOMCause(): Boolean = when (cause) {
is OutOfMemoryError -> true
else -> cause?.hasOOMCause() ?: false
}
/**
* Wraps an exception occurred during compiler execution.
* Covers the case when compiler invocation failed before returning any [ExitCode].
* Always throws some kind of exception.
*/
internal fun wrapAndRethrowCompilationException(executionStrategy: KotlinCompilerExecutionStrategy, e: Throwable): Nothing {
if (e is OutOfMemoryError || e.hasOOMCause()) {
throw OOMErrorException(executionStrategy)
} else if (e is RemoteException) {
throw DaemonCrashedException(e)
} else {
throw e
}
}
/** Exception thrown when [ExitCode] != [ExitCode.OK]. */
internal open class FailedCompilationException(message: String, cause: Throwable? = null) : RuntimeException(message, cause)
/** Exception thrown when [ExitCode] == [ExitCode.COMPILATION_ERROR]. */
internal class CompilationErrorException(message: String) : FailedCompilationException(message)
/** Exception thrown when [ExitCode] == [ExitCode.OOM_ERROR]. */
internal class OOMErrorException(message: String) : FailedCompilationException(message)
private fun OOMErrorException(executionStrategy: KotlinCompilerExecutionStrategy): OOMErrorException {
val exceptionMessage = when (executionStrategy) {
KotlinCompilerExecutionStrategy.DAEMON -> kotlinDaemonOOMHelperMessage
KotlinCompilerExecutionStrategy.IN_PROCESS -> kotlinInProcessOOMHelperMessage
KotlinCompilerExecutionStrategy.OUT_OF_PROCESS -> kotlinOutOfProcessOOMHelperMessage
}
return OOMErrorException(exceptionMessage)
}
/** Exception thrown when during the compilation [java.rmi.RemoteException] is caught */
internal class DaemonCrashedException(cause: Throwable) : FailedCompilationException(kotlinDaemonCrashedMessage, cause)
internal fun TaskWithLocalState.cleanOutputsAndLocalState(reason: String? = null) {
val log = GradleKotlinLogger(logger)
cleanOutputsAndLocalState(allOutputFiles(), log, metrics.get(), reason)
}
internal fun cleanOutputsAndLocalState(
outputFiles: Iterable,
log: KotlinLogger,
metrics: BuildMetricsReporter,
reason: String? = null
) {
log.kotlinDebug {
val suffix = reason?.let { " ($it)" }.orEmpty()
"Cleaning output$suffix:"
}
metrics.measure(GradleBuildTime.CLEAR_OUTPUT) {
for (file in outputFiles) {
when {
file.isDirectory -> {
log.debug("Deleting contents of output directory: $file")
file.deleteDirectoryContents()
}
file.isFile -> {
log.debug("Deleting output file: $file")
file.deleteRecursivelyOrThrow()
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy