forked from oss-review-toolkit/ort
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCreateCommand.kt
139 lines (119 loc) · 5.51 KB
/
CreateCommand.kt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
/*
* Copyright (C) 2020-2021 HERE Europe B.V.
*
* 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
*
* https://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.
*
* SPDX-License-Identifier: Apache-2.0
* License-Filename: LICENSE
*/
package org.ossreviewtoolkit.helper.commands.packageconfig
import com.github.ajalt.clikt.core.CliktCommand
import com.github.ajalt.clikt.core.UsageError
import com.github.ajalt.clikt.parameters.options.convert
import com.github.ajalt.clikt.parameters.options.flag
import com.github.ajalt.clikt.parameters.options.option
import com.github.ajalt.clikt.parameters.options.required
import com.github.ajalt.clikt.parameters.types.file
import java.io.File
import org.ossreviewtoolkit.helper.common.write
import org.ossreviewtoolkit.model.ArtifactProvenance
import org.ossreviewtoolkit.model.Identifier
import org.ossreviewtoolkit.model.Provenance
import org.ossreviewtoolkit.model.RepositoryProvenance
import org.ossreviewtoolkit.model.VcsType
import org.ossreviewtoolkit.model.config.PackageConfiguration
import org.ossreviewtoolkit.model.config.VcsMatcher
import org.ossreviewtoolkit.scanner.storages.FileBasedStorage
import org.ossreviewtoolkit.utils.common.expandTilde
import org.ossreviewtoolkit.utils.common.safeMkdirs
import org.ossreviewtoolkit.utils.core.storage.LocalFileStorage
internal class CreateCommand : CliktCommand(
help = "Creates one package configuration for the source artifact scan and one for the VCS scan, if " +
"a corresponding scan result exists in the given ORT result for the respective provenance. The output " +
"package configuration YAML files are written to the given output directory."
) {
private val scanResultsStorageDir by option(
"--scan-results-storage-dir",
help = "The scan results storage to extract the scan results to."
).convert { it.expandTilde() }
.file(mustExist = false, canBeFile = false, canBeDir = true, mustBeWritable = false, mustBeReadable = false)
.convert { it.absoluteFile.normalize() }
.required()
private val packageId by option(
"--package-id",
help = "The target package for which the package configuration shall be generated."
).convert { Identifier(it) }
.required()
private val outputDir by option(
"--output-dir",
help = "The output directory to write the package configurations to."
).convert { it.expandTilde() }
.file(mustExist = false, canBeFile = false, canBeDir = true, mustBeWritable = false, mustBeReadable = false)
.convert { it.absoluteFile.normalize() }
.required()
private val createHierarchicalDirs by option(
"--create-hierarchical-dirs",
help = "Place the output YAML files in the directory '\$outputdir/\$type/\$namespace/\$name/\$version'."
).flag()
private val forceOverwrite by option(
"--force-overwrite",
help = "Overwrite any output files if they already exist."
).flag()
override fun run() {
outputDir.safeMkdirs()
val scanResultsStorage = FileBasedStorage(LocalFileStorage(scanResultsStorageDir))
val scanResults = scanResultsStorage.read(packageId).getOrDefault(emptyList())
scanResults.find { it.provenance is RepositoryProvenance }?.provenance
?.writePackageConfigurationFile("vcs.yml")
scanResults.find { it.provenance is ArtifactProvenance }?.provenance
?.writePackageConfigurationFile("source-artifact.yml")
}
private fun Provenance.writePackageConfigurationFile(filename: String) {
val packageConfiguration = createPackageConfiguration(packageId, this)
val outputFile = getOutputFile(filename)
if (!forceOverwrite && outputFile.exists()) {
throw UsageError("The output file '${outputFile.absolutePath}' must not exist yet.", statusCode = 2)
}
packageConfiguration.write(outputFile)
println("Wrote a package configuration to '${outputFile.absolutePath}'.")
}
private fun getOutputFile(filename: String): File {
val relativeOutputFilePath = if (createHierarchicalDirs) {
"${packageId.toPath(emptyValue = "_")}/$filename"
} else {
filename
}
return outputDir.resolve(relativeOutputFilePath)
}
}
private fun createPackageConfiguration(id: Identifier, provenance: Provenance): PackageConfiguration =
when (provenance) {
is ArtifactProvenance -> {
PackageConfiguration(
id = id,
sourceArtifactUrl = provenance.sourceArtifact.url
)
}
is RepositoryProvenance -> {
PackageConfiguration(
id = id,
vcs = VcsMatcher(
type = provenance.vcsInfo.type,
url = provenance.vcsInfo.url,
revision = provenance.resolvedRevision,
path = provenance.vcsInfo.path.takeIf { provenance.vcsInfo.type == VcsType.GIT_REPO }
)
)
}
else -> throw IllegalArgumentException("Unsupported provenance $provenance.")
}