@@ -3,6 +3,8 @@ package org.opensearch.commons.alerting.model
3
3
import org.apache.commons.validator.routines.UrlValidator
4
4
import org.apache.hc.core5.net.URIBuilder
5
5
import org.opensearch.common.CheckedFunction
6
+ import org.opensearch.commons.alerting.util.CommonUtilsException
7
+ import org.opensearch.commons.utils.CLUSTER_NAME_REGEX
6
8
import org.opensearch.core.ParseField
7
9
import org.opensearch.core.common.io.stream.StreamInput
8
10
import org.opensearch.core.common.io.stream.StreamOutput
@@ -31,35 +33,46 @@ data class ClusterMetricsInput(
31
33
32
34
// Verify parameters are valid during creation
33
35
init {
34
- require(validateFields()) {
35
- " The uri.api_type field, uri.path field, or uri.uri field must be defined."
36
- }
36
+ // Wrap any validation exceptions in CommonUtilsException.
37
+ try {
38
+ require(validateFields()) {
39
+ " The uri.api_type field, uri.path field, or uri.uri field must be defined."
40
+ }
37
41
38
- // Create an UrlValidator that only accepts "http" and "https" as valid scheme and allows local URLs.
39
- val urlValidator = UrlValidator (arrayOf(" http" , " https" ), UrlValidator .ALLOW_LOCAL_URLS )
42
+ // Create an UrlValidator that only accepts "http" and "https" as valid scheme and allows local URLs.
43
+ val urlValidator = UrlValidator (arrayOf(" http" , " https" ), UrlValidator .ALLOW_LOCAL_URLS )
40
44
41
- // Build url field by field if not provided as whole.
42
- constructedUri = toConstructedUri()
45
+ // Build url field by field if not provided as whole.
46
+ constructedUri = toConstructedUri()
43
47
44
- require(urlValidator.isValid(constructedUri.toString())) {
45
- " Invalid URI constructed from the path and path_params inputs, or the url input."
46
- }
48
+ require(urlValidator.isValid(constructedUri.toString())) {
49
+ " Invalid URI constructed from the path and path_params inputs, or the url input."
50
+ }
47
51
48
- if (url.isNotEmpty() && validateFieldsNotEmpty()) {
49
- require(constructedUri == constructUrlFromInputs()) {
50
- " The provided URL and URI fields form different URLs."
52
+ if (url.isNotEmpty() && validateFieldsNotEmpty()) {
53
+ require(constructedUri == constructUrlFromInputs()) {
54
+ " The provided URL and URI fields form different URLs."
55
+ }
51
56
}
52
- }
53
57
54
- require(constructedUri.host.lowercase() == SUPPORTED_HOST ) {
55
- " Only host '$SUPPORTED_HOST ' is supported."
56
- }
57
- require(constructedUri.port == SUPPORTED_PORT ) {
58
- " Only port '$SUPPORTED_PORT ' is supported."
59
- }
58
+ require(constructedUri.host.lowercase() == SUPPORTED_HOST ) {
59
+ " Only host '$SUPPORTED_HOST ' is supported."
60
+ }
61
+ require(constructedUri.port == SUPPORTED_PORT ) {
62
+ " Only port '$SUPPORTED_PORT ' is supported."
63
+ }
60
64
61
- clusterMetricType = findApiType(constructedUri.path)
62
- this .parseEmptyFields()
65
+ if (clusters.isNotEmpty()) {
66
+ require(clusters.all { CLUSTER_NAME_REGEX .matches(it) }) {
67
+ " Cluster names are not valid."
68
+ }
69
+ }
70
+
71
+ clusterMetricType = findApiType(constructedUri.path)
72
+ this .parseEmptyFields()
73
+ } catch (exception: Exception ) {
74
+ throw CommonUtilsException .wrap(exception)
75
+ }
63
76
}
64
77
65
78
@Throws(IOException ::class )
@@ -158,7 +171,7 @@ data class ClusterMetricsInput(
158
171
/* *
159
172
* Isolates just the path parameters from the [ClusterMetricsInput] URI.
160
173
* @return The path parameters portion of the [ClusterMetricsInput] URI.
161
- * @throws IllegalArgumentException if the [ClusterMetricType] requires path parameters, but none are supplied;
174
+ * @throws CommonUtilsException if the [ClusterMetricType] requires path parameters, but none are supplied;
162
175
* or when path parameters are provided for an [ClusterMetricType] that does not use path parameters.
163
176
*/
164
177
fun parsePathParams (): String {
@@ -178,18 +191,22 @@ data class ClusterMetricsInput(
178
191
pathParams = pathParams.trim(' /' )
179
192
ILLEGAL_PATH_PARAMETER_CHARACTERS .forEach { character ->
180
193
if (pathParams.contains(character)) {
181
- throw IllegalArgumentException (
182
- " The provided path parameters contain invalid characters or spaces. Please omit: " + ILLEGAL_PATH_PARAMETER_CHARACTERS .joinToString(" " )
194
+ throw CommonUtilsException .wrap(
195
+ IllegalArgumentException (
196
+ " The provided path parameters contain invalid characters or spaces. Please omit: " + ILLEGAL_PATH_PARAMETER_CHARACTERS .joinToString(
197
+ " "
198
+ )
199
+ )
183
200
)
184
201
}
185
202
}
186
203
}
187
204
188
205
if (apiType.requiresPathParams && pathParams.isEmpty()) {
189
- throw IllegalArgumentException (" The API requires path parameters." )
206
+ throw CommonUtilsException .wrap( IllegalArgumentException (" The API requires path parameters." ) )
190
207
}
191
208
if (! apiType.supportsPathParams && pathParams.isNotEmpty()) {
192
- throw IllegalArgumentException (" The API does not use path parameters." )
209
+ throw CommonUtilsException .wrap( IllegalArgumentException (" The API does not use path parameters." ) )
193
210
}
194
211
195
212
return pathParams
@@ -199,7 +216,7 @@ data class ClusterMetricsInput(
199
216
* Examines the path of a [ClusterMetricsInput] to determine which API is being called.
200
217
* @param uriPath The path to examine.
201
218
* @return The [ClusterMetricType] associated with the [ClusterMetricsInput] monitor.
202
- * @throws IllegalArgumentException when the API to call cannot be determined from the URI.
219
+ * @throws CommonUtilsException when the API to call cannot be determined from the URI.
203
220
*/
204
221
private fun findApiType (uriPath : String ): ClusterMetricType {
205
222
var apiType = ClusterMetricType .BLANK
@@ -211,7 +228,7 @@ data class ClusterMetricsInput(
211
228
}
212
229
}
213
230
if (apiType.isBlank()) {
214
- throw IllegalArgumentException (" The API could not be determined from the provided URI." )
231
+ throw CommonUtilsException .wrap( IllegalArgumentException (" The API could not be determined from the provided URI." ) )
215
232
}
216
233
return apiType
217
234
}
0 commit comments