-
Notifications
You must be signed in to change notification settings - Fork 381
Expand file tree
/
Copy pathOrtProject.kt
More file actions
129 lines (114 loc) · 4.25 KB
/
OrtProject.kt
File metadata and controls
129 lines (114 loc) · 4.25 KB
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
/*
* Copyright (C) 2025 The ORT Project Copyright Holders <https://github.com/oss-review-toolkit/ort/blob/main/NOTICE>
*
* 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.plugins.packagemanagers.ortprojectfile
import com.charleskorn.kaml.Yaml
import java.io.File
import java.io.IOException
import kotlinx.serialization.KSerializer
import kotlinx.serialization.Serializable
import kotlinx.serialization.decodeFromString
import kotlinx.serialization.descriptors.PrimitiveKind
import kotlinx.serialization.descriptors.PrimitiveSerialDescriptor
import kotlinx.serialization.descriptors.SerialDescriptor
import kotlinx.serialization.encoding.Decoder
import kotlinx.serialization.encoding.Encoder
import kotlinx.serialization.json.Json
import org.ossreviewtoolkit.model.Identifier
import org.ossreviewtoolkit.model.PackageLinkage
import org.ossreviewtoolkit.model.utils.toPackageUrl
@Serializable
internal data class OrtProject(
val projectName: String? = null,
val declaredLicenses: Set<String> = emptySet(),
val description: String? = null,
val homepageUrl: String? = null,
val authors: Set<String> = emptySet(),
val dependencies: List<Dependency> = emptyList()
) {
@Serializable
data class Dependency(
@Serializable(with = IdentifierSerializer::class)
val id: Identifier? = null,
val purl: String? = null,
val description: String? = null,
val vcs: Vcs? = null,
val sourceArtifact: SourceArtifact? = null,
val declaredLicenses: Set<String> = emptySet(),
val homepageUrl: String? = null,
val labels: Map<String, String> = emptyMap(),
val authors: Set<String> = emptySet(),
val scopes: Set<String>? = null,
val linkage: PackageLinkage? = null,
val isModified: Boolean? = null,
val isMetadataOnly: Boolean? = null
) {
init {
require(listOfNotNull(id, purl).isNotEmpty()) {
"There is no id or purl defined for the package."
}
if (id != null) {
require(id.type.isNotBlank() && id.name.isNotBlank() && id.version.isNotBlank()) {
"The id '${id.toCoordinates()}' is not a valid Identifier."
}
}
if (purl != null) {
requireNotNull(purl.toPackageUrl()) {
"The purl '$purl' is not a valid PackageURL."
}
}
}
}
@Serializable
data class Vcs(
val type: String,
val url: String,
val revision: String,
val path: String = ""
)
@Serializable
data class SourceArtifact(
val url: String,
val hash: Hash?
)
@Serializable
data class Hash(
val value: String,
val algorithm: String
)
}
internal fun File.parseOrtProject(): OrtProject =
runCatching {
when (extension) {
"json" -> Json.decodeFromString<OrtProject>(readText())
"yml", "yaml" -> Yaml.default.decodeFromString<OrtProject>(readText())
else -> error("Unknown file extension: '$extension'.")
}
}.getOrElse { cause ->
throw IOException("Could not parse ORT project file at '$absolutePath'.", cause)
}
private object IdentifierSerializer : KSerializer<Identifier> {
override val descriptor: SerialDescriptor = PrimitiveSerialDescriptor(
serialName = checkNotNull(Identifier::class.qualifiedName),
kind = PrimitiveKind.STRING
)
override fun serialize(encoder: Encoder, value: Identifier) {
encoder.encodeString(value.toCoordinates())
}
override fun deserialize(decoder: Decoder) = Identifier(decoder.decodeString())
}