-
Notifications
You must be signed in to change notification settings - Fork 313
/
Identifier.kt
126 lines (109 loc) · 4.48 KB
/
Identifier.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
/*
* Copyright (C) 2017 The ORT Project Authors (see <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.model
import com.fasterxml.jackson.annotation.JsonCreator
import com.fasterxml.jackson.annotation.JsonValue
import org.ossreviewtoolkit.utils.common.AlphaNumericComparator
import org.ossreviewtoolkit.utils.common.encodeOr
/**
* A unique identifier for a software package.
*/
data class Identifier(
/**
* The type of package. When used in the context of a [Project], the type is the name of the package manager that
* manages the project (e.g. "Gradle" for a Gradle project). When used in the context of a [Package], the type is
* the name of the package type or protocol (e.g. "Maven" for a file from a Maven repository).
*/
val type: String,
/**
* The namespace of the package, for example the group for "Maven" or the scope for "NPM".
*/
val namespace: String,
/**
* The name of the package.
*/
val name: String,
/**
* The version of the package.
*/
val version: String
) : Comparable<Identifier> {
companion object {
/**
* A constant for an [Identifier] where all properties are empty strings.
*/
@JvmField
val EMPTY = Identifier(
type = "",
namespace = "",
name = "",
version = ""
)
private val COMPARATOR = compareBy<Identifier>({ it.type }, { it.namespace }, { it.name })
.thenComparing({ it.version }, AlphaNumericComparator)
}
private constructor(components: List<String>) : this(
type = components.getOrElse(0) { "" },
namespace = components.getOrElse(1) { "" },
name = components.getOrElse(2) { "" },
version = components.getOrElse(3) { "" }
)
/**
* Create an [Identifier] from a string with the format "type:namespace:name:version". If the string has less than
* three colon separators the missing values are assigned empty strings.
*/
@JsonCreator
constructor(identifier: String) : this(identifier.split(':', limit = 4))
private val sanitizedComponents = listOf(type, namespace, name, version).map { component ->
component.trim().filterNot { it < ' ' }
}
init {
require(sanitizedComponents.none { ":" in it }) {
"An identifier's properties must not contain ':' because that character is used as a separator in the " +
"string representation: type='$type', namespace='$namespace', name='$name', version='$version'."
}
}
override fun compareTo(other: Identifier) = COMPARATOR.compare(this, other)
/**
* Return whether this [Identifier] is likely to belong any of the organizations mentioned in [names] by looking at
* the [namespace].
*/
fun isFromOrg(vararg names: String) =
names.any { name ->
val lowerName = name.lowercase()
val vendorNamespace = when (type) {
"NPM" -> "@$lowerName"
"Gradle", "Maven", "SBT" -> "((com|io|net|org)\\.)?$lowerName(\\..+)?"
else -> ""
}
// TODO: Think about how to handle package managers that do not have the concept of namespaces, like Cargo.
vendorNamespace.isNotEmpty() && namespace.matches(vendorNamespace.toRegex())
}
/**
* Create Maven-like coordinates based on the properties of the [Identifier].
*/
@JsonValue
fun toCoordinates() = sanitizedComponents.joinToString(":")
/**
* Create a file system path based on the properties of the [Identifier]. All properties are encoded using
* [encodeOr] with [emptyValue] as parameter.
*/
fun toPath(separator: String = "/", emptyValue: String = "unknown"): String =
sanitizedComponents.joinToString(separator) { it.encodeOr(emptyValue) }
}