|
12 | 12 | import org.elasticsearch.common.settings.Settings;
|
13 | 13 | import org.elasticsearch.common.time.DateFormatter;
|
14 | 14 | import org.elasticsearch.common.time.FormatNames;
|
| 15 | +import org.elasticsearch.common.xcontent.support.XContentMapValues; |
15 | 16 | import org.elasticsearch.test.cluster.ElasticsearchCluster;
|
16 | 17 | import org.elasticsearch.test.cluster.local.distribution.DistributionType;
|
17 | 18 | import org.elasticsearch.test.rest.ESRestTestCase;
|
@@ -221,6 +222,117 @@ public void testEsqlRuntimeFields() throws IOException {
|
221 | 222 | assertThat(sumLength, equalTo(20 * numDocs));
|
222 | 223 | }
|
223 | 224 |
|
| 225 | + public void testSyntheticSourceRuntimeFieldQueries() throws IOException { |
| 226 | + String mappings = """ |
| 227 | + { |
| 228 | + "runtime": { |
| 229 | + "message_length": { |
| 230 | + "type": "long" |
| 231 | + } |
| 232 | + }, |
| 233 | + "dynamic": false, |
| 234 | + "properties": { |
| 235 | + "@timestamp": { |
| 236 | + "type": "date" |
| 237 | + }, |
| 238 | + "log" : { |
| 239 | + "properties": { |
| 240 | + "level": { |
| 241 | + "type": "keyword" |
| 242 | + } |
| 243 | + } |
| 244 | + } |
| 245 | + } |
| 246 | + } |
| 247 | + """; |
| 248 | + String indexName = "test-foo"; |
| 249 | + createIndex(indexName, Settings.builder().put("index.mode", "logsdb").build(), mappings); |
| 250 | + |
| 251 | + int numDocs = 1000; |
| 252 | + var sb = new StringBuilder(); |
| 253 | + var now = Instant.now(); |
| 254 | + for (int i = 0; i < numDocs; i++) { |
| 255 | + String level = randomBoolean() ? "info" : randomBoolean() ? "warning" : randomBoolean() ? "error" : "fatal"; |
| 256 | + String msg = randomAlphaOfLength(20); |
| 257 | + String messageLength = Integer.toString(msg.length()); |
| 258 | + sb.append("{ \"create\": {} }").append('\n'); |
| 259 | + if (randomBoolean()) { |
| 260 | + sb.append(""" |
| 261 | + {"@timestamp":"$now","message":"$msg","message_length":$l,"log":{"level":"$level"}} |
| 262 | + """.replace("$now", formatInstant(now)).replace("$level", level).replace("$msg", msg).replace("$l", messageLength)); |
| 263 | + } else { |
| 264 | + sb.append(""" |
| 265 | + {"@timestamp": "$now", "message": "$msg", "message_length": $l} |
| 266 | + """.replace("$now", formatInstant(now)).replace("$msg", msg).replace("$l", messageLength)); |
| 267 | + } |
| 268 | + sb.append('\n'); |
| 269 | + if (i != numDocs - 1) { |
| 270 | + now = now.plusSeconds(1); |
| 271 | + } |
| 272 | + } |
| 273 | + |
| 274 | + var bulkRequest = new Request("POST", "/" + indexName + "/_bulk"); |
| 275 | + bulkRequest.setJsonEntity(sb.toString()); |
| 276 | + bulkRequest.addParameter("refresh", "true"); |
| 277 | + var bulkResponse = client().performRequest(bulkRequest); |
| 278 | + var bulkResponseBody = responseAsMap(bulkResponse); |
| 279 | + assertThat(bulkResponseBody, Matchers.hasEntry("errors", false)); |
| 280 | + |
| 281 | + var forceMergeRequest = new Request("POST", "/" + indexName + "/_forcemerge"); |
| 282 | + var forceMergeResponse = client().performRequest(forceMergeRequest); |
| 283 | + assertOK(forceMergeResponse); |
| 284 | + |
| 285 | + var searchRequest = new Request("POST", "/" + indexName + "/_search"); |
| 286 | + |
| 287 | + searchRequest.setJsonEntity(""" |
| 288 | + { |
| 289 | + "size": 1, |
| 290 | + "query": { |
| 291 | + "bool": { |
| 292 | + "should": [ |
| 293 | + { |
| 294 | + "range": { |
| 295 | + "message_length": { |
| 296 | + "gte": 1, |
| 297 | + "lt": 900000 |
| 298 | + } |
| 299 | + } |
| 300 | + }, |
| 301 | + { |
| 302 | + "range": { |
| 303 | + "message_length": { |
| 304 | + "gte": 900000, |
| 305 | + "lt": 1000000 |
| 306 | + } |
| 307 | + } |
| 308 | + } |
| 309 | + ], |
| 310 | + "minimum_should_match": "1", |
| 311 | + "must_not": [ |
| 312 | + { |
| 313 | + "range": { |
| 314 | + "message_length": { |
| 315 | + "lt": 0 |
| 316 | + } |
| 317 | + } |
| 318 | + } |
| 319 | + ] |
| 320 | + } |
| 321 | + } |
| 322 | + } |
| 323 | + """); |
| 324 | + var searchResponse = client().performRequest(searchRequest); |
| 325 | + assertOK(searchResponse); |
| 326 | + var searchResponseBody = responseAsMap(searchResponse); |
| 327 | + int totalHits = (int) XContentMapValues.extractValue("hits.total.value", searchResponseBody); |
| 328 | + assertThat(totalHits, equalTo(numDocs)); |
| 329 | + |
| 330 | + var shardsHeader = (Map<?, ?>) searchResponseBody.get("_shards"); |
| 331 | + assertThat(shardsHeader.get("failed"), equalTo(0)); |
| 332 | + assertThat(shardsHeader.get("successful"), equalTo(1)); |
| 333 | + assertThat(shardsHeader.get("skipped"), equalTo(0)); |
| 334 | + } |
| 335 | + |
224 | 336 | static String formatInstant(Instant instant) {
|
225 | 337 | return DateFormatter.forPattern(FormatNames.STRICT_DATE_OPTIONAL_TIME.getName()).format(instant);
|
226 | 338 | }
|
|
0 commit comments