Skip to content

Commit 353b0d7

Browse files
authored
Datasource disable feature (opensearch-project#2539)
Signed-off-by: Vamsi Manohar <[email protected]>
1 parent 1a09f96 commit 353b0d7

File tree

34 files changed

+1105
-646
lines changed

34 files changed

+1105
-646
lines changed

core/build.gradle

+3-1
Original file line numberDiff line numberDiff line change
@@ -108,7 +108,9 @@ jacocoTestCoverageVerification {
108108
excludes = [
109109
'org.opensearch.sql.utils.MLCommonsConstants',
110110
'org.opensearch.sql.utils.Constants',
111-
'org.opensearch.sql.datasource.model.*'
111+
'org.opensearch.sql.datasource.model.DataSource',
112+
'org.opensearch.sql.datasource.model.DataSourceStatus',
113+
'org.opensearch.sql.datasource.model.DataSourceType'
112114
]
113115
limit {
114116
counter = 'LINE'

core/src/main/java/org/opensearch/sql/datasource/DataSourceService.java

+10-10
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,8 @@
1414
public interface DataSourceService {
1515

1616
/**
17-
* Returns {@link DataSource} corresponding to the DataSource name.
17+
* Returns {@link DataSource} corresponding to the DataSource name only if the datasource is
18+
* active and authorized.
1819
*
1920
* @param dataSourceName Name of the {@link DataSource}.
2021
* @return {@link DataSource}.
@@ -40,15 +41,6 @@ public interface DataSourceService {
4041
*/
4142
DataSourceMetadata getDataSourceMetadata(String name);
4243

43-
/**
44-
* Returns dataSourceMetadata object with specific name. The returned objects contain all the
45-
* metadata information without any filtering.
46-
*
47-
* @param name name of the {@link DataSource}.
48-
* @return set of {@link DataSourceMetadata}.
49-
*/
50-
DataSourceMetadata getRawDataSourceMetadata(String name);
51-
5244
/**
5345
* Register {@link DataSource} defined by {@link DataSourceMetadata}.
5446
*
@@ -84,4 +76,12 @@ public interface DataSourceService {
8476
* @param dataSourceName name of the {@link DataSource}.
8577
*/
8678
Boolean dataSourceExists(String dataSourceName);
79+
80+
/**
81+
* Performs authorization and datasource status check and then returns RawDataSourceMetadata.
82+
* Specifically for addressing use cases in SparkQueryDispatcher.
83+
*
84+
* @param dataSourceName of the {@link DataSource}
85+
*/
86+
DataSourceMetadata verifyDataSourceAccessAndGetRawMetadata(String dataSourceName);
8787
}

core/src/main/java/org/opensearch/sql/datasource/model/DataSourceMetadata.java

+167-75
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
import com.fasterxml.jackson.annotation.JsonFormat;
1111
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
1212
import com.fasterxml.jackson.annotation.JsonProperty;
13+
import com.google.common.collect.ImmutableList;
1314
import com.google.common.collect.ImmutableMap;
1415
import java.util.ArrayList;
1516
import java.util.Collections;
@@ -19,27 +20,26 @@
1920
import java.util.function.Function;
2021
import lombok.EqualsAndHashCode;
2122
import lombok.Getter;
22-
import lombok.Setter;
2323
import org.apache.commons.lang3.StringUtils;
2424
import org.opensearch.sql.datasource.DataSourceService;
2525

2626
@Getter
27-
@Setter
2827
@EqualsAndHashCode
2928
@JsonIgnoreProperties(ignoreUnknown = true)
3029
public class DataSourceMetadata {
3130

3231
public static final String DEFAULT_RESULT_INDEX = "query_execution_result";
3332
public static final int MAX_RESULT_INDEX_NAME_SIZE = 255;
33+
private static String DATASOURCE_NAME_REGEX = "[@*A-Za-z]+?[*a-zA-Z_\\-0-9]*";
3434
// OS doesn’t allow uppercase: https://tinyurl.com/yse2xdbx
3535
public static final String RESULT_INDEX_NAME_PATTERN = "[a-z0-9_-]+";
3636
public static String INVALID_RESULT_INDEX_NAME_SIZE =
3737
"Result index name size must contains less than "
3838
+ MAX_RESULT_INDEX_NAME_SIZE
39-
+ " characters";
39+
+ " characters.";
4040
public static String INVALID_CHAR_IN_RESULT_INDEX_NAME =
4141
"Result index name has invalid character. Valid characters are a-z, 0-9, -(hyphen) and"
42-
+ " _(underscore)";
42+
+ " _(underscore).";
4343
public static String INVALID_RESULT_INDEX_PREFIX =
4444
"Result index must start with " + DEFAULT_RESULT_INDEX;
4545

@@ -57,96 +57,188 @@ public class DataSourceMetadata {
5757

5858
@JsonProperty private String resultIndex;
5959

60+
@JsonProperty private DataSourceStatus status;
61+
6062
public static Function<String, String> DATASOURCE_TO_RESULT_INDEX =
6163
datasourceName -> String.format("%s_%s", DEFAULT_RESULT_INDEX, datasourceName);
6264

63-
public DataSourceMetadata(
64-
String name,
65-
String description,
66-
DataSourceType connector,
67-
List<String> allowedRoles,
68-
Map<String, String> properties,
69-
String resultIndex) {
70-
this.name = name;
71-
String errorMessage = validateCustomResultIndex(resultIndex);
72-
if (errorMessage != null) {
73-
throw new IllegalArgumentException(errorMessage);
65+
private DataSourceMetadata(Builder builder) {
66+
this.name = builder.name;
67+
this.description = builder.description;
68+
this.connector = builder.connector;
69+
this.allowedRoles = builder.allowedRoles;
70+
this.properties = builder.properties;
71+
this.resultIndex = builder.resultIndex;
72+
this.status = builder.status;
73+
}
74+
75+
public static class Builder {
76+
private String name;
77+
private String description;
78+
private DataSourceType connector;
79+
private List<String> allowedRoles;
80+
private Map<String, String> properties;
81+
private String resultIndex; // Optional
82+
private DataSourceStatus status;
83+
84+
public Builder() {}
85+
86+
public Builder(DataSourceMetadata dataSourceMetadata) {
87+
this.name = dataSourceMetadata.getName();
88+
this.description = dataSourceMetadata.getDescription();
89+
this.connector = dataSourceMetadata.getConnector();
90+
this.resultIndex = dataSourceMetadata.getResultIndex();
91+
this.status = dataSourceMetadata.getStatus();
92+
this.allowedRoles = new ArrayList<>(dataSourceMetadata.getAllowedRoles());
93+
this.properties = new HashMap<>(dataSourceMetadata.getProperties());
7494
}
75-
if (resultIndex == null) {
76-
this.resultIndex = fromNameToCustomResultIndex();
77-
} else {
78-
this.resultIndex = resultIndex;
95+
96+
public Builder setName(String name) {
97+
this.name = name;
98+
return this;
7999
}
80100

81-
this.connector = connector;
82-
this.description = description;
83-
this.properties = properties;
84-
this.allowedRoles = allowedRoles;
85-
}
101+
public Builder setDescription(String description) {
102+
this.description = description;
103+
return this;
104+
}
86105

87-
public DataSourceMetadata() {
88-
this.description = StringUtils.EMPTY;
89-
this.allowedRoles = new ArrayList<>();
90-
this.properties = new HashMap<>();
91-
}
106+
public Builder setConnector(DataSourceType connector) {
107+
this.connector = connector;
108+
return this;
109+
}
92110

93-
/**
94-
* Default OpenSearch {@link DataSourceMetadata}. Which is used to register default OpenSearch
95-
* {@link DataSource} to {@link DataSourceService}.
96-
*/
97-
public static DataSourceMetadata defaultOpenSearchDataSourceMetadata() {
98-
return new DataSourceMetadata(
99-
DEFAULT_DATASOURCE_NAME,
100-
StringUtils.EMPTY,
101-
DataSourceType.OPENSEARCH,
102-
Collections.emptyList(),
103-
ImmutableMap.of(),
104-
null);
105-
}
111+
public Builder setAllowedRoles(List<String> allowedRoles) {
112+
this.allowedRoles = allowedRoles;
113+
return this;
114+
}
106115

107-
public String validateCustomResultIndex(String resultIndex) {
108-
if (resultIndex == null) {
109-
return null;
116+
public Builder setProperties(Map<String, String> properties) {
117+
this.properties = properties;
118+
return this;
110119
}
111-
if (resultIndex.length() > MAX_RESULT_INDEX_NAME_SIZE) {
112-
return INVALID_RESULT_INDEX_NAME_SIZE;
120+
121+
public Builder setResultIndex(String resultIndex) {
122+
this.resultIndex = resultIndex;
123+
return this;
113124
}
114-
if (!resultIndex.matches(RESULT_INDEX_NAME_PATTERN)) {
115-
return INVALID_CHAR_IN_RESULT_INDEX_NAME;
125+
126+
public Builder setDataSourceStatus(DataSourceStatus status) {
127+
this.status = status;
128+
return this;
116129
}
117-
if (resultIndex != null && !resultIndex.startsWith(DEFAULT_RESULT_INDEX)) {
118-
return INVALID_RESULT_INDEX_PREFIX;
130+
131+
public DataSourceMetadata build() {
132+
validateMissingAttributes();
133+
validateName();
134+
validateCustomResultIndex();
135+
fillNullAttributes();
136+
return new DataSourceMetadata(this);
119137
}
120-
return null;
121-
}
122138

123-
/**
124-
* Since we are using datasource name to create result index, we need to make sure that the final
125-
* name is valid
126-
*
127-
* @param resultIndex result index name
128-
* @return valid result index name
129-
*/
130-
private String convertToValidResultIndex(String resultIndex) {
131-
// Limit Length
132-
if (resultIndex.length() > MAX_RESULT_INDEX_NAME_SIZE) {
133-
resultIndex = resultIndex.substring(0, MAX_RESULT_INDEX_NAME_SIZE);
139+
private void fillNullAttributes() {
140+
if (resultIndex == null) {
141+
this.resultIndex = fromNameToCustomResultIndex();
142+
}
143+
if (status == null) {
144+
this.status = DataSourceStatus.ACTIVE;
145+
}
146+
if (description == null) {
147+
this.description = StringUtils.EMPTY;
148+
}
149+
if (properties == null) {
150+
this.properties = ImmutableMap.of();
151+
}
152+
if (allowedRoles == null) {
153+
this.allowedRoles = ImmutableList.of();
154+
}
134155
}
135156

136-
// Pattern Matching: Remove characters that don't match the pattern
137-
StringBuilder validChars = new StringBuilder();
138-
for (char c : resultIndex.toCharArray()) {
139-
if (String.valueOf(c).matches(RESULT_INDEX_NAME_PATTERN)) {
140-
validChars.append(c);
157+
private void validateMissingAttributes() {
158+
List<String> missingAttributes = new ArrayList<>();
159+
if (name == null) {
160+
missingAttributes.add("name");
161+
}
162+
if (connector == null) {
163+
missingAttributes.add("connector");
164+
}
165+
if (!missingAttributes.isEmpty()) {
166+
String errorMessage =
167+
"Datasource configuration error: "
168+
+ String.join(", ", missingAttributes)
169+
+ " cannot be null or empty.";
170+
throw new IllegalArgumentException(errorMessage);
141171
}
142172
}
143-
return validChars.toString();
144-
}
145173

146-
public String fromNameToCustomResultIndex() {
147-
if (name == null) {
148-
throw new IllegalArgumentException("Datasource name cannot be null");
174+
private void validateName() {
175+
if (!name.matches(DATASOURCE_NAME_REGEX)) {
176+
throw new IllegalArgumentException(
177+
String.format(
178+
"DataSource Name: %s contains illegal characters. Allowed characters:"
179+
+ " a-zA-Z0-9_-*@.",
180+
name));
181+
}
182+
}
183+
184+
private void validateCustomResultIndex() {
185+
if (resultIndex == null) {
186+
return;
187+
}
188+
StringBuilder errorMessage = new StringBuilder();
189+
if (resultIndex.length() > MAX_RESULT_INDEX_NAME_SIZE) {
190+
errorMessage.append(INVALID_RESULT_INDEX_NAME_SIZE);
191+
}
192+
if (!resultIndex.matches(RESULT_INDEX_NAME_PATTERN)) {
193+
errorMessage.append(INVALID_CHAR_IN_RESULT_INDEX_NAME);
194+
}
195+
if (!resultIndex.startsWith(DEFAULT_RESULT_INDEX)) {
196+
errorMessage.append(INVALID_RESULT_INDEX_PREFIX);
197+
}
198+
if (errorMessage.length() > 0) {
199+
throw new IllegalArgumentException(errorMessage.toString());
200+
}
201+
}
202+
203+
/**
204+
* Since we are using datasource name to create result index, we need to make sure that the
205+
* final name is valid
206+
*
207+
* @param resultIndex result index name
208+
* @return valid result index name
209+
*/
210+
private String convertToValidResultIndex(String resultIndex) {
211+
// Limit Length
212+
if (resultIndex.length() > MAX_RESULT_INDEX_NAME_SIZE) {
213+
resultIndex = resultIndex.substring(0, MAX_RESULT_INDEX_NAME_SIZE);
214+
}
215+
216+
// Pattern Matching: Remove characters that don't match the pattern
217+
StringBuilder validChars = new StringBuilder();
218+
for (char c : resultIndex.toCharArray()) {
219+
if (String.valueOf(c).matches(RESULT_INDEX_NAME_PATTERN)) {
220+
validChars.append(c);
221+
}
222+
}
223+
return validChars.toString();
149224
}
150-
return convertToValidResultIndex(DATASOURCE_TO_RESULT_INDEX.apply(name.toLowerCase()));
225+
226+
private String fromNameToCustomResultIndex() {
227+
return convertToValidResultIndex(DATASOURCE_TO_RESULT_INDEX.apply(name.toLowerCase()));
228+
}
229+
}
230+
231+
/**
232+
* Default OpenSearch {@link DataSourceMetadata}. Which is used to register default OpenSearch
233+
* {@link DataSource} to {@link DataSourceService}.
234+
*/
235+
public static DataSourceMetadata defaultOpenSearchDataSourceMetadata() {
236+
return new DataSourceMetadata.Builder()
237+
.setName(DEFAULT_DATASOURCE_NAME)
238+
.setDescription(StringUtils.EMPTY)
239+
.setConnector(DataSourceType.OPENSEARCH)
240+
.setAllowedRoles(Collections.emptyList())
241+
.setProperties(ImmutableMap.of())
242+
.build();
151243
}
152244
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
/*
2+
* Copyright OpenSearch Contributors
3+
* SPDX-License-Identifier: Apache-2.0
4+
*/
5+
6+
package org.opensearch.sql.datasource.model;
7+
8+
/** Enum for capturing the current datasource status. */
9+
public enum DataSourceStatus {
10+
ACTIVE("active"),
11+
DISABLED("disabled");
12+
13+
private String text;
14+
15+
DataSourceStatus(String text) {
16+
this.text = text;
17+
}
18+
19+
public String getText() {
20+
return this.text;
21+
}
22+
23+
/**
24+
* Get DataSourceStatus from text.
25+
*
26+
* @param text text.
27+
* @return DataSourceStatus {@link DataSourceStatus}.
28+
*/
29+
public static DataSourceStatus fromString(String text) {
30+
for (DataSourceStatus dataSourceStatus : DataSourceStatus.values()) {
31+
if (dataSourceStatus.text.equalsIgnoreCase(text)) {
32+
return dataSourceStatus;
33+
}
34+
}
35+
throw new IllegalArgumentException("No DataSourceStatus with text " + text + " found");
36+
}
37+
}

0 commit comments

Comments
 (0)