10
10
import com .fasterxml .jackson .annotation .JsonFormat ;
11
11
import com .fasterxml .jackson .annotation .JsonIgnoreProperties ;
12
12
import com .fasterxml .jackson .annotation .JsonProperty ;
13
+ import com .google .common .collect .ImmutableList ;
13
14
import com .google .common .collect .ImmutableMap ;
14
15
import java .util .ArrayList ;
15
16
import java .util .Collections ;
19
20
import java .util .function .Function ;
20
21
import lombok .EqualsAndHashCode ;
21
22
import lombok .Getter ;
22
- import lombok .Setter ;
23
23
import org .apache .commons .lang3 .StringUtils ;
24
24
import org .opensearch .sql .datasource .DataSourceService ;
25
25
26
26
@ Getter
27
- @ Setter
28
27
@ EqualsAndHashCode
29
28
@ JsonIgnoreProperties (ignoreUnknown = true )
30
29
public class DataSourceMetadata {
31
30
32
31
public static final String DEFAULT_RESULT_INDEX = "query_execution_result" ;
33
32
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]*" ;
34
34
// OS doesn’t allow uppercase: https://tinyurl.com/yse2xdbx
35
35
public static final String RESULT_INDEX_NAME_PATTERN = "[a-z0-9_-]+" ;
36
36
public static String INVALID_RESULT_INDEX_NAME_SIZE =
37
37
"Result index name size must contains less than "
38
38
+ MAX_RESULT_INDEX_NAME_SIZE
39
- + " characters" ;
39
+ + " characters. " ;
40
40
public static String INVALID_CHAR_IN_RESULT_INDEX_NAME =
41
41
"Result index name has invalid character. Valid characters are a-z, 0-9, -(hyphen) and"
42
- + " _(underscore)" ;
42
+ + " _(underscore). " ;
43
43
public static String INVALID_RESULT_INDEX_PREFIX =
44
44
"Result index must start with " + DEFAULT_RESULT_INDEX ;
45
45
@@ -57,96 +57,188 @@ public class DataSourceMetadata {
57
57
58
58
@ JsonProperty private String resultIndex ;
59
59
60
+ @ JsonProperty private DataSourceStatus status ;
61
+
60
62
public static Function <String , String > DATASOURCE_TO_RESULT_INDEX =
61
63
datasourceName -> String .format ("%s_%s" , DEFAULT_RESULT_INDEX , datasourceName );
62
64
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 ());
74
94
}
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 ;
79
99
}
80
100
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
+ }
86
105
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
+ }
92
110
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
+ }
106
115
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 ;
110
119
}
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 ;
113
124
}
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 ;
116
129
}
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 );
119
137
}
120
- return null ;
121
- }
122
138
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
+ }
134
155
}
135
156
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 );
141
171
}
142
172
}
143
- return validChars .toString ();
144
- }
145
173
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 ();
149
224
}
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 ();
151
243
}
152
244
}
0 commit comments