From 8f3492a42befadaa72e6ef314d7a1b52c8bee0ed Mon Sep 17 00:00:00 2001 From: yamal Date: Sat, 12 Nov 2022 18:32:11 +0100 Subject: [PATCH 1/7] calculate insights before rendering --- .../spotify/ruler/e2e/ReleaseReportTest.kt | 4 +- .../ruler/frontend/components/Common.kt | 6 +- .../ruler/frontend/components/Insights.kt | 57 ++++--- .../ruler/plugin/report/JsonReporter.kt | 157 +++++++++++++----- .../ruler/plugin/report/JsonReporterTest.kt | 83 ++++++--- .../com/spotify/ruler/models/AppReport.kt | 5 +- .../com/spotify/ruler/models/ComponentType.kt | 8 +- .../com/spotify/ruler/models/FileType.kt | 12 +- .../com/spotify/ruler/models/Insights.kt | 28 ++++ .../com/spotify/ruler/models/ResourceType.kt | 14 +- .../com/spotify/ruler/models/TypeInsights.kt | 32 ++++ 11 files changed, 295 insertions(+), 111 deletions(-) create mode 100644 ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Insights.kt create mode 100644 ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/TypeInsights.kt diff --git a/ruler-e2e-tests/src/test/kotlin/com/spotify/ruler/e2e/ReleaseReportTest.kt b/ruler-e2e-tests/src/test/kotlin/com/spotify/ruler/e2e/ReleaseReportTest.kt index efa458c4..039ed6a4 100644 --- a/ruler-e2e-tests/src/test/kotlin/com/spotify/ruler/e2e/ReleaseReportTest.kt +++ b/ruler-e2e-tests/src/test/kotlin/com/spotify/ruler/e2e/ReleaseReportTest.kt @@ -88,8 +88,8 @@ class ReleaseReportTest { downloadSize += component.downloadSize installSize += component.installSize } - assertThat(downloadSize).isEqualTo(report.downloadSize) - assertThat(installSize).isEqualTo(report.installSize) + assertThat(downloadSize).isEqualTo(report.insights.appDownloadSize) + assertThat(installSize).isEqualTo(report.insights.appInstallSize) } @Test diff --git a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt index 2659d173..b9cadf50 100644 --- a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt +++ b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt @@ -49,7 +49,7 @@ fun RBuilder.report(report: AppReport) { val tabs = listOf( Tab("/", "Breakdown") { breakdown(report.components, sizeType) }, - Tab("/insights", "Insights") { insights(report.components) }, + Tab("/insights", "Insights") { insights(report.insights) }, Tab("/ownership", "Ownership", hasOwnershipInfo) { ownership(report.components, sizeType) }, Tab("/dynamic", "Dynamic features", hasDynamicFeatures) { dynamicFeatures(report.dynamicFeatures, sizeType) }, ) @@ -74,8 +74,8 @@ fun RBuilder.header(report: AppReport) { h3 { +report.name } span(classes = "text-muted") { +"Version ${report.version} (${report.variant})" } } - headerSizeItem(report.downloadSize, "Download size") - headerSizeItem(report.installSize, "Install size") + headerSizeItem(report.insights.appDownloadSize, "Download size") + headerSizeItem(report.insights.appInstallSize, "Install size") } } diff --git a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Insights.kt b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Insights.kt index dd82e7f3..5df98eb0 100644 --- a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Insights.kt +++ b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Insights.kt @@ -22,8 +22,11 @@ import com.spotify.ruler.frontend.chart.ChartConfig import com.spotify.ruler.frontend.chart.BarChartConfig import com.spotify.ruler.frontend.formatSize import com.spotify.ruler.frontend.chart.seriesOf -import com.spotify.ruler.models.AppComponent -import com.spotify.ruler.models.Measurable +import com.spotify.ruler.models.ComponentType +import com.spotify.ruler.models.FileType +import com.spotify.ruler.models.Insights +import com.spotify.ruler.models.TypeInsights +import com.spotify.ruler.models.ResourceType import kotlinx.browser.document import kotlinx.html.id import react.RBuilder @@ -33,30 +36,30 @@ import react.dom.p import react.useEffect @RFunction -fun RBuilder.insights(components: List) { +fun RBuilder.insights(insights: Insights) { div(classes = "row mb-3") { - fileTypeGraphs(components) + fileTypeGraphs(insights.fileTypeInsights) } div(classes = "row") { - componentTypeGraphs(components) + componentTypeGraphs(insights.componentTypeInsights) } div(classes = "row") { - resourcesTypeGraphs(components) + resourcesTypeGraphs(insights.resourcesTypeInsights) } } @RFunction -fun RBuilder.fileTypeGraphs(components: List) { - val labels = arrayOf("Classes", "Resources", "Assets", "Native libraries", "Other") +fun RBuilder.fileTypeGraphs(fileTypeInsights: Map) { + val labels = FileType.values().map { it.label }.toTypedArray() val downloadSizes = LongArray(labels.size) val installSizes = LongArray(labels.size) val fileCounts = LongArray(labels.size) - components.flatMap(AppComponent::files).forEach { file -> - val index = file.type.ordinal - downloadSizes[index] += file.getSize(Measurable.SizeType.DOWNLOAD) - installSizes[index] += file.getSize(Measurable.SizeType.INSTALL) - fileCounts[index]++ + fileTypeInsights.forEach { (fileType, insights) -> + val index = fileType.ordinal + downloadSizes[index] = insights.downloadSize + installSizes[index] = insights.installSize + fileCounts[index] = insights.count } chart( @@ -83,17 +86,17 @@ fun RBuilder.fileTypeGraphs(components: List) { } @RFunction -fun RBuilder.componentTypeGraphs(components: List) { - val labels = arrayOf("Internal", "External") +fun RBuilder.componentTypeGraphs(componentTypeInsights: Map) { + val labels = ComponentType.values().map { it.label }.toTypedArray() val downloadSizes = LongArray(labels.size) val installSizes = LongArray(labels.size) val fileCounts = LongArray(labels.size) - components.forEach { component -> - val index = component.type.ordinal - downloadSizes[index] += component.getSize(Measurable.SizeType.DOWNLOAD) - installSizes[index] += component.getSize(Measurable.SizeType.INSTALL) - fileCounts[index]++ + componentTypeInsights.forEach { (componentType, insights) -> + val index = componentType.ordinal + downloadSizes[index] = insights.downloadSize + installSizes[index] = insights.installSize + fileCounts[index] = insights.count } chart( @@ -122,17 +125,17 @@ fun RBuilder.componentTypeGraphs(components: List) { } @RFunction -fun RBuilder.resourcesTypeGraphs(components: List) { - val labels = arrayOf("Drawable", "Layout", "Raw", "Values", "Font", "Other") +fun RBuilder.resourcesTypeGraphs(resourcesTypeInsights: Map) { + val labels = ResourceType.values().map { it.label }.toTypedArray() val downloadSizes = LongArray(labels.size) val installSizes = LongArray(labels.size) val fileCounts = LongArray(labels.size) - components.flatMap(AppComponent::files).filter { it.resourceType != null }.forEach { file -> - val index = file.resourceType!!.ordinal - downloadSizes[index] += file.getSize(Measurable.SizeType.DOWNLOAD) - installSizes[index] += file.getSize(Measurable.SizeType.INSTALL) - fileCounts[index]++ + resourcesTypeInsights.forEach { (resourceType, insights) -> + val index = resourceType.ordinal + downloadSizes[index] = insights.downloadSize + installSizes[index] = insights.installSize + fileCounts[index] = insights.count } chart( diff --git a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt index a487a6b5..93568dca 100644 --- a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt +++ b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt @@ -19,8 +19,14 @@ package com.spotify.ruler.plugin.report import com.spotify.ruler.models.AppComponent import com.spotify.ruler.models.AppFile import com.spotify.ruler.models.AppReport +import com.spotify.ruler.models.ComponentType import com.spotify.ruler.models.DynamicFeature +import com.spotify.ruler.models.FileType +import com.spotify.ruler.models.Insights import com.spotify.ruler.models.Measurable +import com.spotify.ruler.models.MutableTypeInsights +import com.spotify.ruler.models.ResourceType +import com.spotify.ruler.models.TypeInsights import com.spotify.ruler.plugin.dependency.DependencyComponent import com.spotify.ruler.plugin.models.AppInfo import com.spotify.ruler.plugin.ownership.OwnershipInfo @@ -48,49 +54,14 @@ class JsonReporter { ownershipInfo: OwnershipInfo?, targetDir: File ): File { + val appComponents = getAppComponents(components, ownershipInfo) val report = AppReport( name = appInfo.applicationId, version = appInfo.versionName, variant = appInfo.variantName, - downloadSize = components.values.flatten().sumOf(AppFile::downloadSize), - installSize = components.values.flatten().sumOf(AppFile::installSize), - components = components.map { (component, files) -> - AppComponent( - name = component.name, - type = component.type, - downloadSize = files.sumOf(AppFile::downloadSize), - installSize = files.sumOf(AppFile::installSize), - owner = ownershipInfo?.getOwner(component.name, component.type), - files = files.map { file -> - AppFile( - name = file.name, - type = file.type, - downloadSize = file.downloadSize, - installSize = file.installSize, - owner = ownershipInfo?.getOwner(file.name, component.name, component.type), - resourceType = file.resourceType, - ) - }.sortedWith(comparator.reversed()) - ) - }.sortedWith(comparator.reversed()), - dynamicFeatures = features.map { (feature, files) -> - DynamicFeature( - name = feature, - downloadSize = files.sumOf(AppFile::downloadSize), - installSize = files.sumOf(AppFile::installSize), - owner = ownershipInfo?.getOwner(feature), - files = files.map { file -> - AppFile( - name = file.name, - type = file.type, - downloadSize = file.downloadSize, - installSize = file.installSize, - owner = ownershipInfo?.getOwner(file.name, feature), - resourceType = file.resourceType, - ) - }.sortedWith(comparator.reversed()) - ) - }.sortedWith(comparator.reversed()), + components = appComponents, + dynamicFeatures = getDynamicFeatures(features, ownershipInfo), + insights = getInsights(components, appComponents) ) val format = Json { prettyPrint = true } @@ -98,4 +69,112 @@ class JsonReporter { reportFile.writeText(format.encodeToString(report)) return reportFile } + + private fun getAppComponents( + components: Map>, + ownershipInfo: OwnershipInfo? + ): List = + components.map { (component, files) -> + AppComponent( + name = component.name, + type = component.type, + downloadSize = files.sumOf(AppFile::downloadSize), + installSize = files.sumOf(AppFile::installSize), + owner = ownershipInfo?.getOwner(component.name, component.type), + files = files.map { file -> + AppFile( + name = file.name, + type = file.type, + downloadSize = file.downloadSize, + installSize = file.installSize, + owner = ownershipInfo?.getOwner(file.name, component.name, component.type), + resourceType = file.resourceType, + ) + }.sortedWith(comparator.reversed()) + ) + }.sortedWith(comparator.reversed()) + + private fun getDynamicFeatures( + features: Map>, + ownershipInfo: OwnershipInfo? + ): List = + features.map { (feature, files) -> + DynamicFeature( + name = feature, + downloadSize = files.sumOf(AppFile::downloadSize), + installSize = files.sumOf(AppFile::installSize), + owner = ownershipInfo?.getOwner(feature), + files = files.map { file -> + AppFile( + name = file.name, + type = file.type, + downloadSize = file.downloadSize, + installSize = file.installSize, + owner = ownershipInfo?.getOwner(file.name, feature), + resourceType = file.resourceType, + ) + }.sortedWith(comparator.reversed()) + ) + }.sortedWith(comparator.reversed()) + + private fun getInsights( + components: Map>, + appComponents: List, + ): Insights = + Insights( + appDownloadSize = components.values.flatten().sumOf(AppFile::downloadSize), + appInstallSize = components.values.flatten().sumOf(AppFile::installSize), + fileTypeInsights = getFileTypeInsights(appComponents), + componentTypeInsights = getComponentTypeInsights(appComponents), + resourcesTypeInsights = getResourcesTypeInsights(appComponents), + ) + + private fun getFileTypeInsights(components: List): Map = + getTypeInsights( + items = components.flatMap(AppComponent::files), + getKey = { type } + ) + + private fun getComponentTypeInsights(components: List): Map = + getTypeInsights( + items = components, + getKey = { type } + ) + + private fun getResourcesTypeInsights(components: List): Map = + getTypeInsights( + items = components.flatMap(AppComponent::files).filter { it.resourceType != null }, + getKey = { resourceType!! } + ) + + private inline fun getTypeInsights( + items: List, + getKey: T.() -> K + ): Map { + val typeInsights = mutableMapOf() + + items.forEach { item -> + val key = item.getKey() + val currentInsights = typeInsights.getOrPut(key) { + MutableTypeInsights( + downloadSize = 0L, + installSize = 0L, + count = 0L, + ) + } + currentInsights.downloadSize += item.getSize(Measurable.SizeType.DOWNLOAD) + currentInsights.installSize += item.getSize(Measurable.SizeType.INSTALL) + currentInsights.count++ + } + + return typeInsights.asSequence() + .map { + val (type, insights) = it + type to TypeInsights( + downloadSize = insights.downloadSize, + installSize = insights.installSize, + count = insights.count + ) + }.toMap() + } } diff --git a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt index 43d628e3..80c06168 100644 --- a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt +++ b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt @@ -23,7 +23,9 @@ import com.spotify.ruler.models.AppReport import com.spotify.ruler.models.ComponentType import com.spotify.ruler.models.DynamicFeature import com.spotify.ruler.models.FileType +import com.spotify.ruler.models.Insights import com.spotify.ruler.models.ResourceType +import com.spotify.ruler.models.TypeInsights import com.spotify.ruler.plugin.dependency.DependencyComponent import com.spotify.ruler.plugin.models.AppInfo import com.spotify.ruler.plugin.ownership.OwnershipEntry @@ -68,27 +70,68 @@ class JsonReporterTest { val reportFile = reporter.generateReport(appInfo, components, features, ownershipInfo, targetDir) val report = Json.decodeFromString(reportFile.readText()) - val expected = AppReport("com.spotify.music", "1.2.3", "release", 750, 1050, listOf( - AppComponent(":lib", ComponentType.INTERNAL, 500, 600, listOf( - AppFile("/assets/license.html", FileType.ASSET, 500, 600, "default-team"), - ), "default-team"), - AppComponent(":app", ComponentType.INTERNAL, 250, 450, listOf( - AppFile("/res/layout/activity_main.xml", FileType.RESOURCE, 150, 250, "app-team", ResourceType.LAYOUT), - AppFile("com.spotify.MainActivity", FileType.CLASS, 100, 200, "app-team"), - ), "app-team"), - ), listOf( - DynamicFeature("dynamic", 300, 550, listOf( - AppFile("com.spotify.DynamicActivity", FileType.CLASS, 200, 300, "dynamic-team"), - AppFile( - "/res/layout/activity_dynamic.xml", - FileType.RESOURCE, - 100, - 250, - "dynamic-team", - ResourceType.LAYOUT + val expected = AppReport( + name = "com.spotify.music", + version = "1.2.3", + variant = "release", + components = listOf( + AppComponent(":lib", ComponentType.INTERNAL, 500, 600, listOf( + AppFile("/assets/license.html", FileType.ASSET, 500, 600, "default-team"), + ), "default-team"), + AppComponent(":app", ComponentType.INTERNAL, 250, 450, listOf( + AppFile("/res/layout/activity_main.xml", FileType.RESOURCE, 150, 250, "app-team", ResourceType.LAYOUT), + AppFile("com.spotify.MainActivity", FileType.CLASS, 100, 200, "app-team"), + ), "app-team"), + ), + dynamicFeatures = listOf( + DynamicFeature("dynamic", 300, 550, listOf( + AppFile("com.spotify.DynamicActivity", FileType.CLASS, 200, 300, "dynamic-team"), + AppFile( + "/res/layout/activity_dynamic.xml", + FileType.RESOURCE, + 100, + 250, + "dynamic-team", + ResourceType.LAYOUT + ), + ), "dynamic-team"), + ), + insights = Insights( + appDownloadSize = 750, + appInstallSize = 1050, + fileTypeInsights = mapOf( + FileType.CLASS to TypeInsights( + downloadSize = 100, + installSize = 200, + count = 1 + ), + FileType.RESOURCE to TypeInsights( + downloadSize = 150, + installSize = 250, + count = 1 + ), + FileType.ASSET to TypeInsights( + downloadSize = 500, + installSize = 600, + count = 1 + ) + ), + componentTypeInsights = mapOf( + ComponentType.INTERNAL to TypeInsights( + downloadSize = 750, + installSize = 1050, + count = 2 + ) + ), + resourcesTypeInsights = mapOf( + ResourceType.LAYOUT to TypeInsights( + downloadSize = 150, + installSize = 250, + count = 1 + ) ), - ), "dynamic-team"), - )) + ) + ) assertThat(report).isEqualTo(expected) } diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppReport.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppReport.kt index d4603e77..45add3e1 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppReport.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppReport.kt @@ -24,8 +24,7 @@ data class AppReport( val name: String, val version: String, val variant: String, - override val downloadSize: Long, - override val installSize: Long, val components: List, val dynamicFeatures: List, -) : Measurable + val insights: Insights, +) diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ComponentType.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ComponentType.kt index 1ba9155f..cf8e6bca 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ComponentType.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ComponentType.kt @@ -17,7 +17,7 @@ package com.spotify.ruler.models /** Type of an [AppComponent]. */ -enum class ComponentType { - INTERNAL, - EXTERNAL, -} +enum class ComponentType(val label: String) { + INTERNAL("Internal"), + EXTERNAL("External"), +} \ No newline at end of file diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileType.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileType.kt index 55a7795d..39355632 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileType.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileType.kt @@ -17,10 +17,10 @@ package com.spotify.ruler.models /** Type of an [AppFile]. */ -enum class FileType { - CLASS, - RESOURCE, - ASSET, - NATIVE_LIB, - OTHER, +enum class FileType(val label: String) { + CLASS("Classes"), + RESOURCE("Resources"), + ASSET("Assets"), + NATIVE_LIB("Native libraries"), + OTHER("Other"), } diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Insights.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Insights.kt new file mode 100644 index 00000000..af961510 --- /dev/null +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Insights.kt @@ -0,0 +1,28 @@ +/* +* Copyright 2021 Spotify AB +* +* Licensed under the Apache License, Version 2.0 (the "License"); +* you may not use this file except in compliance with the License. +* You may obtain a copy of the License at +* +* http://www.apache.org/licenses/LICENSE-2.0 +* +* Unless required by applicable law or agreed to in writing, software +* distributed under the License is distributed on an "AS IS" BASIS, +* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +* See the License for the specific language governing permissions and +* limitations under the License. +*/ + +package com.spotify.ruler.models + +import kotlinx.serialization.Serializable + +@Serializable +data class Insights( + val appDownloadSize: Long, + val appInstallSize: Long, + val fileTypeInsights: Map, + val resourcesTypeInsights: Map, + val componentTypeInsights: Map, +) diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ResourceType.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ResourceType.kt index b97957de..3123ea0e 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ResourceType.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ResourceType.kt @@ -17,11 +17,11 @@ package com.spotify.ruler.models /** Resource type for [FileType.RESOURCE]. */ -enum class ResourceType { - DRAWABLE, - LAYOUT, - FONT, - RAW, - VALUES, - OTHER, +enum class ResourceType(val label: String) { + DRAWABLE("Drawable"), + LAYOUT("Layout"), + RAW("Raw"), + VALUES("Values"), + FONT("Font"), + OTHER("Other"), } diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/TypeInsights.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/TypeInsights.kt new file mode 100644 index 00000000..f3feb814 --- /dev/null +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/TypeInsights.kt @@ -0,0 +1,32 @@ +/* +* Copyright 2021 Spotify AB +* +* Licensed under the Apache License, Version 2.0 (the "License"); +* you may not use this file except in compliance with the License. +* You may obtain a copy of the License at +* +* http://www.apache.org/licenses/LICENSE-2.0 +* +* Unless required by applicable law or agreed to in writing, software +* distributed under the License is distributed on an "AS IS" BASIS, +* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +* See the License for the specific language governing permissions and +* limitations under the License. +*/ + +package com.spotify.ruler.models + +import kotlinx.serialization.Serializable + +@Serializable +data class TypeInsights( + val downloadSize: Long, + val installSize: Long, + val count: Long, +) + +data class MutableTypeInsights( + var downloadSize: Long, + var installSize: Long, + var count: Long, +) From 6ce88b3e641a2eeaaeffa72237599fd3f57f75c9 Mon Sep 17 00:00:00 2001 From: yamal Date: Mon, 14 Nov 2022 14:33:09 +0100 Subject: [PATCH 2/7] calculate ownership before rendering --- .../ruler/frontend/components/Breakdown.kt | 2 +- .../ruler/frontend/components/Common.kt | 4 +- .../ruler/frontend/components/Ownership.kt | 70 ++++++----- .../ruler/plugin/report/JsonReporter.kt | 103 +++++++++++---- .../ruler/plugin/report/JsonReporterTest.kt | 118 +++++++++++------- .../com/spotify/ruler/models/AppComponent.kt | 2 +- .../com/spotify/ruler/models/AppReport.kt | 1 + .../spotify/ruler/models/DynamicFeature.kt | 2 +- .../com/spotify/ruler/models/FileContainer.kt | 2 +- .../com/spotify/ruler/models/Measurable.kt | 3 - .../com/spotify/ruler/models/OwnedSize.kt | 25 ++++ .../kotlin/com/spotify/ruler/models/Owner.kt | 25 ++++ .../spotify/ruler/models/OwnershipOverview.kt | 28 +++++ 13 files changed, 279 insertions(+), 106 deletions(-) create mode 100644 ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedSize.kt create mode 100644 ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Owner.kt create mode 100644 ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt diff --git a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Breakdown.kt b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Breakdown.kt index 4b315323..cb15b5de 100644 --- a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Breakdown.kt +++ b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Breakdown.kt @@ -64,7 +64,7 @@ fun RBuilder.containerListItemHeader(id: Int, container: FileContainer, sizeType attrs["data-bs-toggle"] = "collapse" attrs["data-bs-target"] = "#module-$id-body" span(classes = "font-monospace text-truncate me-3") { +container.name } - container.owner?.let { owner -> span(classes = "badge bg-secondary me-3") { +owner } } + container.owner?.let { owner -> span(classes = "badge bg-secondary me-3") { +owner.name } } span(classes = "ms-auto me-3 text-nowrap") { +formatSize(container, sizeType) } diff --git a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt index b9cadf50..d17676e5 100644 --- a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt +++ b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt @@ -44,13 +44,13 @@ import react.useState fun RBuilder.report(report: AppReport) { val (sizeType, setSizeType) = useState(Measurable.SizeType.DOWNLOAD) - val hasOwnershipInfo = report.components.any { component -> component.owner != null } + val hasOwnershipInfo = report.ownershipOverview?.isNotEmpty() == true val hasDynamicFeatures = report.dynamicFeatures.isNotEmpty() val tabs = listOf( Tab("/", "Breakdown") { breakdown(report.components, sizeType) }, Tab("/insights", "Insights") { insights(report.insights) }, - Tab("/ownership", "Ownership", hasOwnershipInfo) { ownership(report.components, sizeType) }, + Tab("/ownership", "Ownership", hasOwnershipInfo) { ownership(report.components, sizeType, report.ownershipOverview!!) }, Tab("/dynamic", "Dynamic features", hasDynamicFeatures) { dynamicFeatures(report.dynamicFeatures, sizeType) }, ) diff --git a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Ownership.kt b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Ownership.kt index 5bab9a4a..1f182d00 100644 --- a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Ownership.kt +++ b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Ownership.kt @@ -22,9 +22,11 @@ import com.spotify.ruler.frontend.chart.BarChartConfig import com.spotify.ruler.frontend.chart.seriesOf import com.spotify.ruler.frontend.formatSize import com.spotify.ruler.models.AppComponent -import com.spotify.ruler.models.AppFile import com.spotify.ruler.models.ComponentType import com.spotify.ruler.models.Measurable +import com.spotify.ruler.models.OwnedSize +import com.spotify.ruler.models.Owner +import com.spotify.ruler.models.OwnershipOverview import react.RBuilder import react.dom.div import react.dom.h4 @@ -34,25 +36,21 @@ import react.useState const val PAGE_SIZE = 10 @RFunction -fun RBuilder.ownership(components: List, sizeType: Measurable.SizeType) { - componentOwnershipOverview(components) - componentOwnershipPerTeam(components, sizeType) +fun RBuilder.ownership( + components: List, + sizeType: Measurable.SizeType, + ownershipOverview: Map, +) { + componentOwnershipOverview(ownershipOverview) + componentOwnershipPerTeam(components, sizeType, ownershipOverview) } @RFunction -fun RBuilder.componentOwnershipOverview(components: List) { - val sizes = mutableMapOf() - components.flatMap(AppComponent::files).forEach { file -> - val owner = file.owner ?: return@forEach - val current = sizes.getOrPut(owner) { Measurable.Mutable(0, 0) } - current.downloadSize += file.downloadSize - current.installSize += file.installSize - } - - val sorted = sizes.entries.sortedByDescending { (_, measurable) -> measurable.downloadSize } +fun RBuilder.componentOwnershipOverview(ownershipOverview: Map) { + val sorted = ownershipOverview.entries.sortedByDescending { (_, ownership) -> ownership.totalInstallSize } val owners = sorted.map { (owner, _) -> owner } - val downloadSizes = sorted.map { (_, measurable) -> measurable.downloadSize } - val installSizes = sorted.map { (_, measurable) -> measurable.installSize } + val downloadSizes = sorted.map { (_, ownership) -> ownership.totalDownloadSize } + val installSizes = sorted.map { (_, ownership) -> ownership.totalInstallSize } pagedContent(owners.size, PAGE_SIZE) { pageStartIndex, pageEndIndex -> chart( @@ -74,34 +72,50 @@ fun RBuilder.componentOwnershipOverview(components: List) { } @RFunction -fun RBuilder.componentOwnershipPerTeam(components: List, sizeType: Measurable.SizeType) { +fun RBuilder.componentOwnershipPerTeam( + components: List, + sizeType: Measurable.SizeType, + ownershipOverview: Map, +) { val files = components.flatMap(AppComponent::files) - val owners = files.mapNotNull(AppFile::owner).distinct().sorted() + val owners = ownershipOverview.keys.sorted() var selectedOwner by useState(owners.first()) - val ownedComponents = components.filter { component -> component.owner == selectedOwner } + val ownedComponents = components.filter { component -> component.owner?.name == selectedOwner } val ownedFiles = files.filter { file -> file.owner == selectedOwner } + val ownedFilesCount = ownershipOverview[selectedOwner]?.filesCount ?: 0 + + val totalOwnedDownloadSize = ownershipOverview[selectedOwner]?.totalDownloadSize ?: 0 + val totalOwnedInstallSize = ownershipOverview[selectedOwner]?.totalInstallSize ?: 0 + val remainingOwnedDownloadSize = ownershipOverview[selectedOwner]?.filesFromNotOwnedComponentsDownloadSize ?: 0 + val remainingOwnedInstallSize = ownershipOverview[selectedOwner]?.filesFromNotOwnedComponentsInstallSize ?: 0 val remainingOwnedFiles = ownedFiles.toMutableSet() val processedComponents = ownedComponents.map { component -> val ownedFilesFromComponent = component.files.filter { file -> file.owner == selectedOwner } remainingOwnedFiles.removeAll(ownedFilesFromComponent) component.copy( - downloadSize = ownedFilesFromComponent.sumOf(AppFile::downloadSize), - installSize = ownedFilesFromComponent.sumOf(AppFile::installSize), + downloadSize = component.owner?.ownedSize?.downloadSize ?: 0, + installSize = component.owner?.ownedSize?.installSize ?: 0, files = ownedFilesFromComponent, ) }.toMutableList() // Group together all owned files which belong to components not owned by the currently selected owner - if (remainingOwnedFiles.isNotEmpty()) { + if (remainingOwnedDownloadSize > 0 || remainingOwnedInstallSize > 0) { processedComponents += AppComponent( name = "Other owned files", type = ComponentType.INTERNAL, - downloadSize = remainingOwnedFiles.sumOf(AppFile::downloadSize), - installSize = remainingOwnedFiles.sumOf(AppFile::installSize), + downloadSize = remainingOwnedDownloadSize, + installSize = remainingOwnedInstallSize, files = remainingOwnedFiles.toList(), - owner = selectedOwner, + owner = Owner( + name = selectedOwner, + ownedSize = OwnedSize( + downloadSize = remainingOwnedDownloadSize, + installSize = remainingOwnedInstallSize, + ), + ), ) } @@ -109,9 +123,9 @@ fun RBuilder.componentOwnershipPerTeam(components: List, sizeType: dropdown(owners, "owner-dropdown") { owner -> selectedOwner = owner } div(classes = "row mt-4 mb-4") { highlightedValue(ownedComponents.size, "Component(s)") - highlightedValue(ownedFiles.size, "File(s)") - highlightedValue(ownedFiles.sumOf(AppFile::downloadSize), "Download size", ::formatSize) - highlightedValue(ownedFiles.sumOf(AppFile::installSize), "Install size", ::formatSize) + highlightedValue(ownedFilesCount, "File(s)") + highlightedValue(totalOwnedDownloadSize, "Download size", ::formatSize) + highlightedValue(totalOwnedInstallSize, "Install size", ::formatSize) } containerList(processedComponents, sizeType) } diff --git a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt index 93568dca..79ca967e 100644 --- a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt +++ b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt @@ -25,6 +25,9 @@ import com.spotify.ruler.models.FileType import com.spotify.ruler.models.Insights import com.spotify.ruler.models.Measurable import com.spotify.ruler.models.MutableTypeInsights +import com.spotify.ruler.models.OwnedSize +import com.spotify.ruler.models.Owner +import com.spotify.ruler.models.OwnershipOverview import com.spotify.ruler.models.ResourceType import com.spotify.ruler.models.TypeInsights import com.spotify.ruler.plugin.dependency.DependencyComponent @@ -61,7 +64,8 @@ class JsonReporter { variant = appInfo.variantName, components = appComponents, dynamicFeatures = getDynamicFeatures(features, ownershipInfo), - insights = getInsights(components, appComponents) + insights = getInsights(components, appComponents), + ownershipOverview = ownershipInfo?.let { getOwnershipOverview(appComponents) }, ) val format = Json { prettyPrint = true } @@ -75,22 +79,25 @@ class JsonReporter { ownershipInfo: OwnershipInfo? ): List = components.map { (component, files) -> + val componentFiles = files.map { file -> + AppFile( + name = file.name, + type = file.type, + downloadSize = file.downloadSize, + installSize = file.installSize, + owner = ownershipInfo?.getOwner(file.name, component.name, component.type), + resourceType = file.resourceType, + ) + }.sortedWith(comparator.reversed()) AppComponent( name = component.name, type = component.type, downloadSize = files.sumOf(AppFile::downloadSize), installSize = files.sumOf(AppFile::installSize), - owner = ownershipInfo?.getOwner(component.name, component.type), - files = files.map { file -> - AppFile( - name = file.name, - type = file.type, - downloadSize = file.downloadSize, - installSize = file.installSize, - owner = ownershipInfo?.getOwner(file.name, component.name, component.type), - resourceType = file.resourceType, - ) - }.sortedWith(comparator.reversed()) + owner = ownershipInfo?.let { + getOwner(ownershipInfo.getOwner(component.name, component.type), componentFiles) + }, + files = componentFiles ) }.sortedWith(comparator.reversed()) @@ -99,24 +106,45 @@ class JsonReporter { ownershipInfo: OwnershipInfo? ): List = features.map { (feature, files) -> + val dynamicFeatureFiles = files.map { file -> + AppFile( + name = file.name, + type = file.type, + downloadSize = file.downloadSize, + installSize = file.installSize, + owner = ownershipInfo?.getOwner(file.name, feature), + resourceType = file.resourceType, + ) + }.sortedWith(comparator.reversed()) DynamicFeature( name = feature, downloadSize = files.sumOf(AppFile::downloadSize), installSize = files.sumOf(AppFile::installSize), - owner = ownershipInfo?.getOwner(feature), - files = files.map { file -> - AppFile( - name = file.name, - type = file.type, - downloadSize = file.downloadSize, - installSize = file.installSize, - owner = ownershipInfo?.getOwner(file.name, feature), - resourceType = file.resourceType, - ) - }.sortedWith(comparator.reversed()) + owner = ownershipInfo?.let { getOwner(ownershipInfo.getOwner(feature), dynamicFeatureFiles) }, + files = dynamicFeatureFiles ) }.sortedWith(comparator.reversed()) + private fun getOwner( + componentOwnerName: String, + files: List, + ): Owner { + var ownedDownloadSize = 0L + var ownedInstallSize = 0L + files.filter { it.owner == componentOwnerName }.forEach { ownedFile -> + ownedDownloadSize += ownedFile.downloadSize + ownedInstallSize += ownedFile.installSize + } + + return Owner( + name = componentOwnerName, + ownedSize = OwnedSize( + downloadSize = ownedDownloadSize, + installSize = ownedInstallSize, + ) + ) + } + private fun getInsights( components: Map>, appComponents: List, @@ -177,4 +205,33 @@ class JsonReporter { ) }.toMap() } + + private fun getOwnershipOverview(appComponents: List): Map { + val overview = mutableMapOf() + + appComponents.forEach { component -> + component.files.forEach { file -> + file.owner?.let { fileOwner -> + val current = overview.getOrPut(fileOwner) { + OwnershipOverview( + totalDownloadSize = 0, + totalInstallSize = 0, + filesCount = 0, + filesFromNotOwnedComponentsDownloadSize = 0, + filesFromNotOwnedComponentsInstallSize = 0, + ) + } + current.totalDownloadSize += file.downloadSize + current.totalInstallSize += file.installSize + current.filesCount++ + if (fileOwner != component.owner?.name) { + current.filesFromNotOwnedComponentsDownloadSize += file.downloadSize + current.filesFromNotOwnedComponentsInstallSize += file.installSize + } + } + } + } + + return overview + } } diff --git a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt index 80c06168..762e7ee1 100644 --- a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt +++ b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt @@ -24,6 +24,9 @@ import com.spotify.ruler.models.ComponentType import com.spotify.ruler.models.DynamicFeature import com.spotify.ruler.models.FileType import com.spotify.ruler.models.Insights +import com.spotify.ruler.models.OwnedSize +import com.spotify.ruler.models.Owner +import com.spotify.ruler.models.OwnershipOverview import com.spotify.ruler.models.ResourceType import com.spotify.ruler.models.TypeInsights import com.spotify.ruler.plugin.dependency.DependencyComponent @@ -44,6 +47,7 @@ class JsonReporterTest { DependencyComponent(":app", ComponentType.INTERNAL) to listOf( AppFile("com.spotify.MainActivity", FileType.CLASS, 100, 200), AppFile("/res/layout/activity_main.xml", FileType.RESOURCE, 150, 250, resourceType = ResourceType.LAYOUT), + AppFile("com.spotify.LoginActivity", FileType.CLASS, 50, 100, "login-team"), ), DependencyComponent(":lib", ComponentType.INTERNAL) to listOf( AppFile("/assets/license.html", FileType.ASSET, 500, 600), @@ -62,7 +66,11 @@ class JsonReporterTest { ), ) - private val ownershipEntries = listOf(OwnershipEntry(":app", "app-team"), OwnershipEntry("dynamic", "dynamic-team")) + private val ownershipEntries = listOf( + OwnershipEntry(":app", "app-team"), + OwnershipEntry("com.spotify.LoginActivity", "login-team"), + OwnershipEntry("dynamic", "dynamic-team") + ) private val ownershipInfo = OwnershipInfo(ownershipEntries, "default-team") @Test @@ -75,61 +83,78 @@ class JsonReporterTest { version = "1.2.3", variant = "release", components = listOf( - AppComponent(":lib", ComponentType.INTERNAL, 500, 600, listOf( - AppFile("/assets/license.html", FileType.ASSET, 500, 600, "default-team"), - ), "default-team"), - AppComponent(":app", ComponentType.INTERNAL, 250, 450, listOf( - AppFile("/res/layout/activity_main.xml", FileType.RESOURCE, 150, 250, "app-team", ResourceType.LAYOUT), - AppFile("com.spotify.MainActivity", FileType.CLASS, 100, 200, "app-team"), - ), "app-team"), + AppComponent( + name =":lib", + type = ComponentType.INTERNAL, + downloadSize = 500, + installSize = 600, + files = listOf( + AppFile("/assets/license.html", FileType.ASSET, 500, 600, "default-team"), + ), + owner = Owner(name = "default-team", ownedSize = OwnedSize(downloadSize = 500, installSize = 600)) + ), + AppComponent( + name = ":app", + type = ComponentType.INTERNAL, + downloadSize = 300, + installSize = 550, + files = listOf( + AppFile("/res/layout/activity_main.xml", FileType.RESOURCE, 150, 250, "app-team", ResourceType.LAYOUT), + AppFile("com.spotify.MainActivity", FileType.CLASS, 100, 200, "app-team"), + AppFile("com.spotify.LoginActivity", FileType.CLASS, 50, 100, "login-team"), + ), + owner = Owner(name = "app-team", ownedSize = OwnedSize(downloadSize = 250, installSize = 450)) + ), ), dynamicFeatures = listOf( - DynamicFeature("dynamic", 300, 550, listOf( - AppFile("com.spotify.DynamicActivity", FileType.CLASS, 200, 300, "dynamic-team"), - AppFile( - "/res/layout/activity_dynamic.xml", - FileType.RESOURCE, - 100, - 250, - "dynamic-team", - ResourceType.LAYOUT + DynamicFeature( + name = "dynamic", + downloadSize = 300, + installSize = 550, + files = listOf( + AppFile("com.spotify.DynamicActivity", FileType.CLASS, 200, 300, "dynamic-team"), + AppFile("/res/layout/activity_dynamic.xml", FileType.RESOURCE, 100, 250, "dynamic-team", ResourceType.LAYOUT), ), - ), "dynamic-team"), + owner = Owner(name = "dynamic-team", ownedSize = OwnedSize(downloadSize = 300, installSize = 550)) + ), ), insights = Insights( - appDownloadSize = 750, - appInstallSize = 1050, + appDownloadSize = 800, + appInstallSize = 1150, fileTypeInsights = mapOf( - FileType.CLASS to TypeInsights( - downloadSize = 100, - installSize = 200, - count = 1 - ), - FileType.RESOURCE to TypeInsights( - downloadSize = 150, - installSize = 250, - count = 1 - ), - FileType.ASSET to TypeInsights( - downloadSize = 500, - installSize = 600, - count = 1 - ) + FileType.CLASS to TypeInsights(downloadSize = 150, installSize = 300, count = 2), + FileType.RESOURCE to TypeInsights(downloadSize = 150, installSize = 250, count = 1), + FileType.ASSET to TypeInsights(downloadSize = 500, installSize = 600, count = 1) ), componentTypeInsights = mapOf( - ComponentType.INTERNAL to TypeInsights( - downloadSize = 750, - installSize = 1050, - count = 2 - ) + ComponentType.INTERNAL to TypeInsights(downloadSize = 800, installSize = 1150, count = 2) ), resourcesTypeInsights = mapOf( - ResourceType.LAYOUT to TypeInsights( - downloadSize = 150, - installSize = 250, - count = 1 - ) + ResourceType.LAYOUT to TypeInsights(downloadSize = 150, installSize = 250, count = 1) + ), + ), + ownershipOverview = mapOf( + "default-team" to OwnershipOverview( + totalDownloadSize = 500, + totalInstallSize = 600, + filesCount = 1, + filesFromNotOwnedComponentsDownloadSize = 0, + filesFromNotOwnedComponentsInstallSize = 0, + ), + "app-team" to OwnershipOverview( + totalDownloadSize = 250, + totalInstallSize = 450, + filesCount = 2, + filesFromNotOwnedComponentsDownloadSize = 0, + filesFromNotOwnedComponentsInstallSize = 0, ), + "login-team" to OwnershipOverview( + totalDownloadSize = 50, + totalInstallSize = 100, + filesCount = 1, + filesFromNotOwnedComponentsDownloadSize = 50, + filesFromNotOwnedComponentsInstallSize = 100, + ) ) ) assertThat(report).isEqualTo(expected) @@ -139,7 +164,8 @@ class JsonReporterTest { fun `Ownership info is omitted from report if it is null`(@TempDir targetDir: File) { val reportFile = reporter.generateReport(appInfo, components, features, null, targetDir) val reportContent = reportFile.readText() - assertThat(reportContent).doesNotContain("owner") + assertThat(reportContent).doesNotContain("\"owner\"") + assertThat(reportContent).contains("\"ownershipOverview\": null") } @Test diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppComponent.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppComponent.kt index 6feeb290..be430133 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppComponent.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppComponent.kt @@ -26,5 +26,5 @@ data class AppComponent( override val downloadSize: Long, override val installSize: Long, override val files: List, - override val owner: String? = null, + override val owner: Owner? = null, ) : FileContainer diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppReport.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppReport.kt index 45add3e1..0e0717cc 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppReport.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppReport.kt @@ -27,4 +27,5 @@ data class AppReport( val components: List, val dynamicFeatures: List, val insights: Insights, + val ownershipOverview: Map?, ) diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/DynamicFeature.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/DynamicFeature.kt index 4b27a730..ff5496cd 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/DynamicFeature.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/DynamicFeature.kt @@ -25,5 +25,5 @@ data class DynamicFeature( override val downloadSize: Long, override val installSize: Long, override val files: List, - override val owner: String? = null, + override val owner: Owner? = null, ) : FileContainer diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileContainer.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileContainer.kt index 6b66876c..85ee1599 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileContainer.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileContainer.kt @@ -19,6 +19,6 @@ package com.spotify.ruler.models /** Piece of an app that can contain files. */ interface FileContainer : Measurable { val name: String - val owner: String? + val owner: Owner? val files: List } diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Measurable.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Measurable.kt index 60c6769c..7d45884b 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Measurable.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Measurable.kt @@ -33,7 +33,4 @@ interface Measurable { SizeType.DOWNLOAD -> downloadSize SizeType.INSTALL -> installSize } - - /** A mutable [Measurable] implementation. */ - data class Mutable(override var downloadSize: Long, override var installSize: Long) : Measurable } diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedSize.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedSize.kt new file mode 100644 index 00000000..e9978eb8 --- /dev/null +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedSize.kt @@ -0,0 +1,25 @@ +/* +* Copyright 2021 Spotify AB +* +* Licensed under the Apache License, Version 2.0 (the "License"); +* you may not use this file except in compliance with the License. +* You may obtain a copy of the License at +* +* http://www.apache.org/licenses/LICENSE-2.0 +* +* Unless required by applicable law or agreed to in writing, software +* distributed under the License is distributed on an "AS IS" BASIS, +* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +* See the License for the specific language governing permissions and +* limitations under the License. +*/ + +package com.spotify.ruler.models + +import kotlinx.serialization.Serializable + +@Serializable +data class OwnedSize( + override val downloadSize: Long, + override val installSize: Long +) : Measurable diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Owner.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Owner.kt new file mode 100644 index 00000000..fde12161 --- /dev/null +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Owner.kt @@ -0,0 +1,25 @@ +/* +* Copyright 2021 Spotify AB +* +* Licensed under the Apache License, Version 2.0 (the "License"); +* you may not use this file except in compliance with the License. +* You may obtain a copy of the License at +* +* http://www.apache.org/licenses/LICENSE-2.0 +* +* Unless required by applicable law or agreed to in writing, software +* distributed under the License is distributed on an "AS IS" BASIS, +* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +* See the License for the specific language governing permissions and +* limitations under the License. +*/ + +package com.spotify.ruler.models + +import kotlinx.serialization.Serializable + +@Serializable +data class Owner( + val name: String, + val ownedSize: OwnedSize, +) diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt new file mode 100644 index 00000000..25a1e1af --- /dev/null +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt @@ -0,0 +1,28 @@ +/* +* Copyright 2021 Spotify AB +* +* Licensed under the Apache License, Version 2.0 (the "License"); +* you may not use this file except in compliance with the License. +* You may obtain a copy of the License at +* +* http://www.apache.org/licenses/LICENSE-2.0 +* +* Unless required by applicable law or agreed to in writing, software +* distributed under the License is distributed on an "AS IS" BASIS, +* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +* See the License for the specific language governing permissions and +* limitations under the License. +*/ + +package com.spotify.ruler.models + +import kotlinx.serialization.Serializable + +@Serializable +data class OwnershipOverview( + var totalDownloadSize: Long, + var totalInstallSize: Long, + var filesCount: Long, + var filesFromNotOwnedComponentsDownloadSize: Long, + var filesFromNotOwnedComponentsInstallSize: Long, +) From fbe92f7b4d6210476a299a35c9551b9d9e447dcd Mon Sep 17 00:00:00 2001 From: yamal Date: Tue, 15 Nov 2022 13:26:04 +0100 Subject: [PATCH 3/7] optional config for excluding file listing --- .../spotify/ruler/plugin/RulerExtension.kt | 3 + .../com/spotify/ruler/plugin/RulerPlugin.kt | 4 +- .../com/spotify/ruler/plugin/RulerTask.kt | 12 +++- .../ruler/plugin/report/JsonReporter.kt | 24 +++++++- .../ruler/plugin/report/JsonReporterTest.kt | 61 ++++++++++++++++++- 5 files changed, 97 insertions(+), 7 deletions(-) diff --git a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerExtension.kt b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerExtension.kt index 9b45f940..82bc95a8 100644 --- a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerExtension.kt +++ b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerExtension.kt @@ -29,8 +29,11 @@ open class RulerExtension(objects: ObjectFactory) { val ownershipFile: RegularFileProperty = objects.fileProperty() val defaultOwner: Property = objects.property(String::class.java) + val shouldExcludeFileListing: Property = objects.property(Boolean::class.java) + // Set up default values init { defaultOwner.convention("unknown") + shouldExcludeFileListing.convention(false) } } diff --git a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerPlugin.kt b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerPlugin.kt index 05d7b27e..0201b3e3 100644 --- a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerPlugin.kt +++ b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerPlugin.kt @@ -53,6 +53,8 @@ class RulerPlugin : Plugin { task.workingDir.set(project.layout.buildDirectory.dir("intermediates/ruler/${variant.name}")) task.reportDir.set(project.layout.buildDirectory.dir("reports/ruler/${variant.name}")) + task.shouldExcludeFileListing.set(rulerExtension.shouldExcludeFileListing) + // Add explicit dependency to support DexGuard task.dependsOn("bundle$variantName") } @@ -64,7 +66,7 @@ class RulerPlugin : Plugin { private fun getAppInfo(project: Project, variant: ApplicationVariant) = project.provider { AppInfo( applicationId = variant.applicationId.get(), - versionName = variant.outputs.first().versionName.get() ?: "-", + versionName = variant.outputs.first().versionName.orNull ?: "-", variantName = variant.name, ) } diff --git a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerTask.kt b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerTask.kt index 161565f2..e4f40668 100644 --- a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerTask.kt +++ b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/RulerTask.kt @@ -69,6 +69,9 @@ abstract class RulerTask : DefaultTask() { @get:Input abstract val defaultOwner: Property + @get:Input + abstract val shouldExcludeFileListing: Property + @get:OutputDirectory abstract val workingDir: DirectoryProperty @@ -132,7 +135,14 @@ abstract class RulerTask : DefaultTask() { val reportDir = reportDir.asFile.get() val jsonReporter = JsonReporter() - val jsonReport = jsonReporter.generateReport(appInfo.get(), components, features, ownershipInfo, reportDir) + val jsonReport = jsonReporter.generateReport( + appInfo.get(), + components, + features, + ownershipInfo, + shouldExcludeFileListing.get(), + reportDir, + ) project.logger.lifecycle("Wrote JSON report to ${jsonReport.toPath().toUri()}") val htmlReporter = HtmlReporter() diff --git a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt index 79ca967e..48fd136b 100644 --- a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt +++ b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt @@ -55,6 +55,7 @@ class JsonReporter { components: Map>, features: Map>, ownershipInfo: OwnershipInfo?, + shouldExcludeFileListing: Boolean, targetDir: File ): File { val appComponents = getAppComponents(components, ownershipInfo) @@ -62,8 +63,9 @@ class JsonReporter { name = appInfo.applicationId, version = appInfo.versionName, variant = appInfo.variantName, - components = appComponents, - dynamicFeatures = getDynamicFeatures(features, ownershipInfo), + components = appComponents.excludeComponentFilesIfNeeded(shouldExcludeFileListing), + dynamicFeatures = getDynamicFeatures(features, ownershipInfo) + .excludeDyamicFeatureFilesIfNeeded(shouldExcludeFileListing), insights = getInsights(components, appComponents), ownershipOverview = ownershipInfo?.let { getOwnershipOverview(appComponents) }, ) @@ -234,4 +236,22 @@ class JsonReporter { return overview } + + private fun List.excludeComponentFilesIfNeeded( + shouldExcludeFileListing: Boolean + ): List = + if (shouldExcludeFileListing) { + map { it.copy(files = emptyList()) } + } else { + this + } + + private fun List.excludeDyamicFeatureFilesIfNeeded( + shouldExcludeFileListing: Boolean + ): List = + if (shouldExcludeFileListing) { + map { it.copy(files = emptyList()) } + } else { + this + } } diff --git a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt index 762e7ee1..17f87505 100644 --- a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt +++ b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt @@ -75,7 +75,7 @@ class JsonReporterTest { @Test fun `JSON report is generated`(@TempDir targetDir: File) { - val reportFile = reporter.generateReport(appInfo, components, features, ownershipInfo, targetDir) + val reportFile = reporter.generateReport(appInfo, components, features, ownershipInfo, false, targetDir) val report = Json.decodeFromString(reportFile.readText()) val expected = AppReport( @@ -160,9 +160,64 @@ class JsonReporterTest { assertThat(report).isEqualTo(expected) } + @Test + fun `JSON report is generated without file listing`(@TempDir targetDir: File) { + val components = mapOf( + DependencyComponent(":app", ComponentType.INTERNAL) to listOf( + AppFile("com.spotify.MainActivity", FileType.CLASS, 100, 200), + ), + ) + val features = mapOf( + "dynamic" to listOf( + AppFile("com.spotify.DynamicActivity", FileType.CLASS, 200, 300) + ) + ) + val reportFile = reporter.generateReport(appInfo, components, features, null, true, targetDir) + val report = Json.decodeFromString(reportFile.readText()) + + val expected = AppReport( + name = "com.spotify.music", + version = "1.2.3", + variant = "release", + components = listOf( + AppComponent( + name =":app", + type = ComponentType.INTERNAL, + downloadSize = 100, + installSize = 200, + files = emptyList(), + owner = null + ), + ), + dynamicFeatures = listOf( + DynamicFeature( + name = "dynamic", + downloadSize = 200, + installSize = 300, + files = emptyList(), + owner = null + ), + ), + insights = Insights( + appDownloadSize = 100, + appInstallSize = 200, + fileTypeInsights = mapOf( + FileType.CLASS to TypeInsights(downloadSize = 100, installSize = 200, count = 1), + ), + componentTypeInsights = mapOf( + ComponentType.INTERNAL to TypeInsights(downloadSize = 100, installSize = 200, count = 1) + ), + resourcesTypeInsights = emptyMap(), + ), + ownershipOverview = null + ) + + assertThat(expected).isEqualTo(report) + } + @Test fun `Ownership info is omitted from report if it is null`(@TempDir targetDir: File) { - val reportFile = reporter.generateReport(appInfo, components, features, null, targetDir) + val reportFile = reporter.generateReport(appInfo, components, features, null, false, targetDir) val reportContent = reportFile.readText() assertThat(reportContent).doesNotContain("\"owner\"") assertThat(reportContent).contains("\"ownershipOverview\": null") @@ -171,7 +226,7 @@ class JsonReporterTest { @Test fun `Existing reports are overwritten`(@TempDir targetDir: File) { repeat(2) { - reporter.generateReport(appInfo, components, features, ownershipInfo, targetDir) + reporter.generateReport(appInfo, components, features, ownershipInfo, false, targetDir) } } } From 4d275e431a981a1d5b96d19b68e78257fbf50bef Mon Sep 17 00:00:00 2001 From: yamal Date: Wed, 16 Nov 2022 16:28:01 +0100 Subject: [PATCH 4/7] add missing javadoc --- .../spotify/ruler/frontend/components/Ownership.kt | 8 ++++---- .../com/spotify/ruler/plugin/report/JsonReporter.kt | 11 ++++++----- .../spotify/ruler/plugin/report/JsonReporterTest.kt | 10 +++++----- .../kotlin/com/spotify/ruler/models/AppComponent.kt | 2 +- .../ruler/models/{Owner.kt => ComponentOwner.kt} | 7 +++++-- .../kotlin/com/spotify/ruler/models/DynamicFeature.kt | 2 +- .../kotlin/com/spotify/ruler/models/FileContainer.kt | 2 +- .../kotlin/com/spotify/ruler/models/Insights.kt | 1 + .../models/{OwnedSize.kt => OwnedComponentSize.kt} | 3 ++- .../com/spotify/ruler/models/OwnershipOverview.kt | 1 + .../kotlin/com/spotify/ruler/models/TypeInsights.kt | 2 ++ 11 files changed, 29 insertions(+), 20 deletions(-) rename ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/{Owner.kt => ComponentOwner.kt} (79%) rename ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/{OwnedSize.kt => OwnedComponentSize.kt} (88%) diff --git a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Ownership.kt b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Ownership.kt index 1f182d00..59723e7b 100644 --- a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Ownership.kt +++ b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Ownership.kt @@ -24,8 +24,8 @@ import com.spotify.ruler.frontend.formatSize import com.spotify.ruler.models.AppComponent import com.spotify.ruler.models.ComponentType import com.spotify.ruler.models.Measurable -import com.spotify.ruler.models.OwnedSize -import com.spotify.ruler.models.Owner +import com.spotify.ruler.models.OwnedComponentSize +import com.spotify.ruler.models.ComponentOwner import com.spotify.ruler.models.OwnershipOverview import react.RBuilder import react.dom.div @@ -109,9 +109,9 @@ fun RBuilder.componentOwnershipPerTeam( downloadSize = remainingOwnedDownloadSize, installSize = remainingOwnedInstallSize, files = remainingOwnedFiles.toList(), - owner = Owner( + owner = ComponentOwner( name = selectedOwner, - ownedSize = OwnedSize( + ownedSize = OwnedComponentSize( downloadSize = remainingOwnedDownloadSize, installSize = remainingOwnedInstallSize, ), diff --git a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt index 48fd136b..5619fa63 100644 --- a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt +++ b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt @@ -25,8 +25,8 @@ import com.spotify.ruler.models.FileType import com.spotify.ruler.models.Insights import com.spotify.ruler.models.Measurable import com.spotify.ruler.models.MutableTypeInsights -import com.spotify.ruler.models.OwnedSize -import com.spotify.ruler.models.Owner +import com.spotify.ruler.models.OwnedComponentSize +import com.spotify.ruler.models.ComponentOwner import com.spotify.ruler.models.OwnershipOverview import com.spotify.ruler.models.ResourceType import com.spotify.ruler.models.TypeInsights @@ -47,6 +47,7 @@ class JsonReporter { * @param appInfo General info about the analyzed app. * @param components Map of app component names to their respective files * @param ownershipInfo Optional info about the owners of components. + * @param shouldExcludeFileListing Flag to determine if file listing should be included in the report * @param targetDir Directory where the generated report will be located * @return Generated JSON report file */ @@ -130,7 +131,7 @@ class JsonReporter { private fun getOwner( componentOwnerName: String, files: List, - ): Owner { + ): ComponentOwner { var ownedDownloadSize = 0L var ownedInstallSize = 0L files.filter { it.owner == componentOwnerName }.forEach { ownedFile -> @@ -138,9 +139,9 @@ class JsonReporter { ownedInstallSize += ownedFile.installSize } - return Owner( + return ComponentOwner( name = componentOwnerName, - ownedSize = OwnedSize( + ownedSize = OwnedComponentSize( downloadSize = ownedDownloadSize, installSize = ownedInstallSize, ) diff --git a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt index 17f87505..e4a70078 100644 --- a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt +++ b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/report/JsonReporterTest.kt @@ -24,8 +24,8 @@ import com.spotify.ruler.models.ComponentType import com.spotify.ruler.models.DynamicFeature import com.spotify.ruler.models.FileType import com.spotify.ruler.models.Insights -import com.spotify.ruler.models.OwnedSize -import com.spotify.ruler.models.Owner +import com.spotify.ruler.models.OwnedComponentSize +import com.spotify.ruler.models.ComponentOwner import com.spotify.ruler.models.OwnershipOverview import com.spotify.ruler.models.ResourceType import com.spotify.ruler.models.TypeInsights @@ -91,7 +91,7 @@ class JsonReporterTest { files = listOf( AppFile("/assets/license.html", FileType.ASSET, 500, 600, "default-team"), ), - owner = Owner(name = "default-team", ownedSize = OwnedSize(downloadSize = 500, installSize = 600)) + owner = ComponentOwner(name = "default-team", ownedSize = OwnedComponentSize(downloadSize = 500, installSize = 600)) ), AppComponent( name = ":app", @@ -103,7 +103,7 @@ class JsonReporterTest { AppFile("com.spotify.MainActivity", FileType.CLASS, 100, 200, "app-team"), AppFile("com.spotify.LoginActivity", FileType.CLASS, 50, 100, "login-team"), ), - owner = Owner(name = "app-team", ownedSize = OwnedSize(downloadSize = 250, installSize = 450)) + owner = ComponentOwner(name = "app-team", ownedSize = OwnedComponentSize(downloadSize = 250, installSize = 450)) ), ), dynamicFeatures = listOf( @@ -115,7 +115,7 @@ class JsonReporterTest { AppFile("com.spotify.DynamicActivity", FileType.CLASS, 200, 300, "dynamic-team"), AppFile("/res/layout/activity_dynamic.xml", FileType.RESOURCE, 100, 250, "dynamic-team", ResourceType.LAYOUT), ), - owner = Owner(name = "dynamic-team", ownedSize = OwnedSize(downloadSize = 300, installSize = 550)) + owner = ComponentOwner(name = "dynamic-team", ownedSize = OwnedComponentSize(downloadSize = 300, installSize = 550)) ), ), insights = Insights( diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppComponent.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppComponent.kt index be430133..7cfd64e4 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppComponent.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/AppComponent.kt @@ -26,5 +26,5 @@ data class AppComponent( override val downloadSize: Long, override val installSize: Long, override val files: List, - override val owner: Owner? = null, + override val owner: ComponentOwner? = null, ) : FileContainer diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Owner.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ComponentOwner.kt similarity index 79% rename from ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Owner.kt rename to ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ComponentOwner.kt index fde12161..be4bfb33 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Owner.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/ComponentOwner.kt @@ -18,8 +18,11 @@ package com.spotify.ruler.models import kotlinx.serialization.Serializable +/** Representation of the relation between the owner of a component + * and the corresponding OwnedComponentSize + * */ @Serializable -data class Owner( +data class ComponentOwner( val name: String, - val ownedSize: OwnedSize, + val ownedSize: OwnedComponentSize, ) diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/DynamicFeature.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/DynamicFeature.kt index ff5496cd..2760bb59 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/DynamicFeature.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/DynamicFeature.kt @@ -25,5 +25,5 @@ data class DynamicFeature( override val downloadSize: Long, override val installSize: Long, override val files: List, - override val owner: Owner? = null, + override val owner: ComponentOwner? = null, ) : FileContainer diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileContainer.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileContainer.kt index 85ee1599..cbd50164 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileContainer.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/FileContainer.kt @@ -19,6 +19,6 @@ package com.spotify.ruler.models /** Piece of an app that can contain files. */ interface FileContainer : Measurable { val name: String - val owner: Owner? + val owner: ComponentOwner? val files: List } diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Insights.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Insights.kt index af961510..bdbe0f3f 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Insights.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/Insights.kt @@ -18,6 +18,7 @@ package com.spotify.ruler.models import kotlinx.serialization.Serializable +/** Global Insights about the app size */ @Serializable data class Insights( val appDownloadSize: Long, diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedSize.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedComponentSize.kt similarity index 88% rename from ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedSize.kt rename to ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedComponentSize.kt index e9978eb8..72a98b50 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedSize.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnedComponentSize.kt @@ -18,8 +18,9 @@ package com.spotify.ruler.models import kotlinx.serialization.Serializable +/** Total size of a component that belongs to a specific owner */ @Serializable -data class OwnedSize( +data class OwnedComponentSize( override val downloadSize: Long, override val installSize: Long ) : Measurable diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt index 25a1e1af..96418fb5 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt @@ -18,6 +18,7 @@ package com.spotify.ruler.models import kotlinx.serialization.Serializable +/** Overview with global size details obout the components/files owned by some owner */ @Serializable data class OwnershipOverview( var totalDownloadSize: Long, diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/TypeInsights.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/TypeInsights.kt index f3feb814..78942a8f 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/TypeInsights.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/TypeInsights.kt @@ -18,6 +18,7 @@ package com.spotify.ruler.models import kotlinx.serialization.Serializable +/** Insights about some type of measurable item (e.g: a file) */ @Serializable data class TypeInsights( val downloadSize: Long, @@ -25,6 +26,7 @@ data class TypeInsights( val count: Long, ) +/** Helper and mutable class used in intermediate calculation of TypeInsights */ data class MutableTypeInsights( var downloadSize: Long, var installSize: Long, From d08998c8ee8d9a277c28cb3d1c17f9ce2407f271 Mon Sep 17 00:00:00 2001 From: yamal Date: Wed, 16 Nov 2022 16:58:53 +0100 Subject: [PATCH 5/7] display component items as non collapsable if no files --- .../ruler/frontend/components/Breakdown.kt | 30 ++++++++++++++----- .../ruler/frontend/components/Common.kt | 2 +- 2 files changed, 24 insertions(+), 8 deletions(-) diff --git a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Breakdown.kt b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Breakdown.kt index cb15b5de..6b388113 100644 --- a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Breakdown.kt +++ b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Breakdown.kt @@ -52,8 +52,12 @@ fun RBuilder.containerList(containers: List, sizeType: Measurable @Suppress("UNUSED_PARAMETER") fun RBuilder.containerListItem(id: Int, container: FileContainer, sizeType: Measurable.SizeType, @RKey key: String) { div(classes = "accordion-item") { - containerListItemHeader(id, container, sizeType) - containerListItemBody(id, container, sizeType) + if (container.files.isEmpty()) { + containerWithoutFilesListItemHeader(container, sizeType) + } else { + containerListItemHeader(id, container, sizeType) + containerListItemBody(id, container, sizeType) + } } } @@ -63,15 +67,27 @@ fun RBuilder.containerListItemHeader(id: Int, container: FileContainer, sizeType button(classes = "accordion-button collapsed") { attrs["data-bs-toggle"] = "collapse" attrs["data-bs-target"] = "#module-$id-body" - span(classes = "font-monospace text-truncate me-3") { +container.name } - container.owner?.let { owner -> span(classes = "badge bg-secondary me-3") { +owner.name } } - span(classes = "ms-auto me-3 text-nowrap") { - +formatSize(container, sizeType) - } + containerHeader(container, sizeType) } } } +@RFunction +fun RBuilder.containerWithoutFilesListItemHeader(container: FileContainer, sizeType: Measurable.SizeType) { + div(classes = "list-group-item d-flex border-0") { + containerHeader(container, sizeType) + } +} + +@RFunction +fun RBuilder.containerHeader(container: FileContainer, sizeType: Measurable.SizeType) { + span(classes = "font-monospace text-truncate me-3") { +container.name } + container.owner?.let { owner -> span(classes = "badge bg-secondary me-3") { +owner.name } } + span(classes = "ms-auto me-3 text-nowrap") { + +formatSize(container, sizeType) + } +} + @RFunction fun RBuilder.containerListItemBody(id: Int, container: FileContainer, sizeType: Measurable.SizeType) { div(classes = "accordion-collapse collapse") { diff --git a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt index d17676e5..fd201301 100644 --- a/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt +++ b/ruler-frontend/src/main/kotlin/com/spotify/ruler/frontend/components/Common.kt @@ -50,7 +50,7 @@ fun RBuilder.report(report: AppReport) { val tabs = listOf( Tab("/", "Breakdown") { breakdown(report.components, sizeType) }, Tab("/insights", "Insights") { insights(report.insights) }, - Tab("/ownership", "Ownership", hasOwnershipInfo) { ownership(report.components, sizeType, report.ownershipOverview!!) }, + Tab("/ownership", "Ownership", hasOwnershipInfo) { ownership(report.components, sizeType, report.ownershipOverview ?: emptyMap()) }, Tab("/dynamic", "Dynamic features", hasDynamicFeatures) { dynamicFeatures(report.dynamicFeatures, sizeType) }, ) From 7c575529051b8fbe50529345c328e6280400de67 Mon Sep 17 00:00:00 2001 From: yamal Date: Wed, 16 Nov 2022 17:09:25 +0100 Subject: [PATCH 6/7] do not expose mutable class --- .../spotify/ruler/plugin/report/JsonReporter.kt | 16 +++++++++++++--- .../spotify/ruler/models/OwnershipOverview.kt | 11 ++++++++++- 2 files changed, 23 insertions(+), 4 deletions(-) diff --git a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt index 5619fa63..4c6d993f 100644 --- a/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt +++ b/ruler-gradle-plugin/src/main/kotlin/com/spotify/ruler/plugin/report/JsonReporter.kt @@ -27,6 +27,7 @@ import com.spotify.ruler.models.Measurable import com.spotify.ruler.models.MutableTypeInsights import com.spotify.ruler.models.OwnedComponentSize import com.spotify.ruler.models.ComponentOwner +import com.spotify.ruler.models.MutableOwnershipOverview import com.spotify.ruler.models.OwnershipOverview import com.spotify.ruler.models.ResourceType import com.spotify.ruler.models.TypeInsights @@ -210,13 +211,13 @@ class JsonReporter { } private fun getOwnershipOverview(appComponents: List): Map { - val overview = mutableMapOf() + val overview = mutableMapOf() appComponents.forEach { component -> component.files.forEach { file -> file.owner?.let { fileOwner -> val current = overview.getOrPut(fileOwner) { - OwnershipOverview( + MutableOwnershipOverview( totalDownloadSize = 0, totalInstallSize = 0, filesCount = 0, @@ -235,7 +236,16 @@ class JsonReporter { } } - return overview + return overview.map { + val (owner, ownershipOverview) = it + owner to OwnershipOverview( + totalDownloadSize = ownershipOverview.totalDownloadSize, + totalInstallSize = ownershipOverview.totalInstallSize, + filesCount = ownershipOverview.filesCount, + filesFromNotOwnedComponentsDownloadSize = ownershipOverview.filesFromNotOwnedComponentsDownloadSize, + filesFromNotOwnedComponentsInstallSize = ownershipOverview.filesFromNotOwnedComponentsInstallSize, + ) + }.toMap() } private fun List.excludeComponentFilesIfNeeded( diff --git a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt index 96418fb5..0dabadf1 100644 --- a/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt +++ b/ruler-models/src/commonMain/kotlin/com/spotify/ruler/models/OwnershipOverview.kt @@ -21,9 +21,18 @@ import kotlinx.serialization.Serializable /** Overview with global size details obout the components/files owned by some owner */ @Serializable data class OwnershipOverview( + val totalDownloadSize: Long, + val totalInstallSize: Long, + val filesCount: Long, + val filesFromNotOwnedComponentsDownloadSize: Long, + val filesFromNotOwnedComponentsInstallSize: Long, +) + +/** Helper and mutable class used in intermediate calculation of OwnershipOverview */ +data class MutableOwnershipOverview( var totalDownloadSize: Long, var totalInstallSize: Long, var filesCount: Long, var filesFromNotOwnedComponentsDownloadSize: Long, var filesFromNotOwnedComponentsInstallSize: Long, -) +) \ No newline at end of file From f8b2989f68a6e8382d8abe514b1a14084dddccb3 Mon Sep 17 00:00:00 2001 From: yamal Date: Wed, 16 Nov 2022 19:00:35 +0100 Subject: [PATCH 7/7] fix tests --- .../kotlin/com/spotify/ruler/plugin/RulerIntegrationTest.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/RulerIntegrationTest.kt b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/RulerIntegrationTest.kt index faa2ff16..04658fcd 100644 --- a/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/RulerIntegrationTest.kt +++ b/ruler-gradle-plugin/src/test/kotlin/com/spotify/ruler/plugin/RulerIntegrationTest.kt @@ -77,7 +77,7 @@ class RulerIntegrationTest { gradlew(task) val reportDir = projectDir.resolve("app/build/reports/ruler/$variant") - assertThat(reportDir.resolve("report.json").readText()).doesNotContain("owner") + assertThat(reportDir.resolve("report.json").readText()).contains("\"ownershipOverview\": null") } @ParameterizedTest