|
20 | 20 |
|
21 | 21 | package com.dropbox.affectedmoduledetector |
22 | 22 |
|
23 | | -import org.gradle.BuildAdapter |
24 | | -import org.gradle.BuildResult |
25 | | -import org.gradle.api.invocation.Gradle |
| 23 | +import org.gradle.api.Project |
| 24 | +import org.gradle.api.file.RegularFileProperty |
26 | 25 | import org.gradle.api.logging.LogLevel |
27 | 26 | import org.gradle.api.logging.Logger |
| 27 | +import org.gradle.api.provider.Property |
| 28 | +import org.gradle.api.provider.Provider |
| 29 | +import org.gradle.api.services.BuildService |
| 30 | +import org.gradle.api.services.BuildServiceParameters |
| 31 | +import org.gradle.internal.build.event.BuildEventListenerRegistryInternal |
28 | 32 | import org.gradle.internal.logging.slf4j.OutputEventListenerBackedLogger |
29 | 33 | import org.gradle.internal.logging.slf4j.OutputEventListenerBackedLoggerContext |
| 34 | +import org.gradle.internal.operations.* |
30 | 35 | import org.gradle.internal.time.Clock |
| 36 | +import org.gradle.invocation.DefaultGradle |
| 37 | +import java.io.File |
31 | 38 |
|
32 | 39 | /** |
33 | 40 | * Gradle logger that logs to a string. |
34 | 41 | */ |
35 | | -class ToStringLogger( |
36 | | - private val stringBuilder: StringBuilder = StringBuilder() |
| 42 | +internal open class ToStringLogger( |
| 43 | + private val loggerProvider: Provider<ToStringLoggerBuildService>? |
37 | 44 | ) : OutputEventListenerBackedLogger( |
38 | 45 | "amd", |
39 | | - OutputEventListenerBackedLoggerContext( |
40 | | - Clock { |
41 | | - System.currentTimeMillis() |
42 | | - } |
43 | | - ).also { |
| 46 | + OutputEventListenerBackedLoggerContext { |
| 47 | + System.currentTimeMillis() |
| 48 | + }.also { |
44 | 49 | it.level = LogLevel.DEBUG |
45 | | - it.setOutputEventListener { |
46 | | - stringBuilder.appendln(it.toString()) |
| 50 | + it.setOutputEventListener { outputEvent -> |
| 51 | + loggerProvider?.get()?.parameters?.getStringBuilderProperty()?.get()?.appendLine(outputEvent.toString()) |
47 | 52 | } |
48 | 53 | }, |
49 | 54 | Clock { |
50 | 55 | System.currentTimeMillis() |
51 | 56 | } |
52 | 57 | ) { |
| 58 | + |
53 | 59 | /** |
54 | 60 | * Returns the current log. |
55 | 61 | */ |
56 | | - fun buildString() = stringBuilder.toString() |
| 62 | + fun buildString() = loggerProvider?.get()?.parameters?.getStringBuilderProperty()?.get()?.toString() |
57 | 63 |
|
| 64 | + @Suppress("UnstableApiUsage") // BuildService is not yet stable |
58 | 65 | companion object { |
| 66 | + internal abstract class ToStringLoggerBuildService : BuildService<ToStringLoggerBuildService.ToStringLoggerBuildServiceParameters>, BuildOperationListener, AutoCloseable { |
| 67 | + interface ToStringLoggerBuildServiceParameters : BuildServiceParameters { |
| 68 | + fun getStringBuilderProperty(): Property<StringBuilder> |
| 69 | + fun getOutputFileProperty(): RegularFileProperty |
| 70 | + } |
| 71 | + |
| 72 | + override fun started(p0: BuildOperationDescriptor, p1: OperationStartEvent) { } |
| 73 | + |
| 74 | + override fun progress(p0: OperationIdentifier, p1: OperationProgressEvent) { } |
| 75 | + |
| 76 | + override fun finished(buildOperationDescriptor: BuildOperationDescriptor, operationFinishEvent: OperationFinishEvent) { } |
| 77 | + |
| 78 | + override fun close() { |
| 79 | + val outputFile = parameters.getOutputFileProperty().orNull?.asFile ?: return |
| 80 | + outputFile.appendText(parameters.getStringBuilderProperty().get().toString()) |
| 81 | + println("Wrote dependency log to ${outputFile.absolutePath}") |
| 82 | + } |
| 83 | + } |
| 84 | + |
| 85 | + /** |
| 86 | + * Creates the [ToStringLogger] |
| 87 | + * |
| 88 | + * @param project the current project to apply to |
| 89 | + * @param logFilename the filename for the logs to go |
| 90 | + * @param logFolder the path to where the log should output. if null doesn't output |
| 91 | + */ |
59 | 92 | fun createWithLifecycle( |
60 | | - gradle: Gradle, |
61 | | - onComplete: (String) -> Unit |
| 93 | + project: Project, |
| 94 | + logFilename: String, |
| 95 | + logFolder: String? = null |
62 | 96 | ): Logger { |
63 | | - val logger = ToStringLogger() |
64 | | - gradle.addBuildListener(object : BuildAdapter() { |
65 | | - override fun buildFinished(result: BuildResult) { |
66 | | - onComplete(logger.buildString()) |
| 97 | + val gradle = project.gradle |
| 98 | + val stringBuilder = StringBuilder() |
| 99 | + val toStringLoggerBuildService = gradle.sharedServices.registerIfAbsent("to-string-logger-build-listener", ToStringLoggerBuildService::class.java) { service -> |
| 100 | + service.parameters.getStringBuilderProperty().set(stringBuilder) |
| 101 | + if (logFolder != null) { |
| 102 | + val distDir = File(logFolder) |
| 103 | + if (!distDir.exists()) { |
| 104 | + distDir.mkdirs() |
| 105 | + } |
| 106 | + val outputFile = distDir.resolve(logFilename) |
| 107 | + service.parameters.getOutputFileProperty().set(outputFile) |
67 | 108 | } |
68 | | - }) |
| 109 | + } |
| 110 | + val logger = ToStringLogger(toStringLoggerBuildService) |
| 111 | + (gradle as DefaultGradle).services[BuildEventListenerRegistryInternal::class.java].onOperationCompletion(toStringLoggerBuildService) |
69 | 112 | return logger |
70 | 113 | } |
71 | 114 | } |
|
0 commit comments