-
Notifications
You must be signed in to change notification settings - Fork 1.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add tests for all supported query types for derived fields
Signed-off-by: Rishabh Maurya <[email protected]>
- Loading branch information
1 parent
4f081d9
commit 1fdb6d7
Showing
6 changed files
with
326 additions
and
44 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
279 changes: 279 additions & 0 deletions
279
server/src/test/java/org/opensearch/index/mapper/DerivedFieldMapperQueryTests.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,279 @@ | ||
/* | ||
* SPDX-License-Identifier: Apache-2.0 | ||
* | ||
* The OpenSearch Contributors require contributions made to | ||
* this file be licensed under the Apache-2.0 license or a | ||
* compatible open source license. | ||
*/ | ||
|
||
package org.opensearch.index.mapper; | ||
|
||
import org.apache.lucene.document.Document; | ||
import org.apache.lucene.document.Field; | ||
import org.apache.lucene.document.TextField; | ||
import org.apache.lucene.index.DirectoryReader; | ||
import org.apache.lucene.index.IndexReader; | ||
import org.apache.lucene.index.IndexWriter; | ||
import org.apache.lucene.index.IndexWriterConfig; | ||
import org.apache.lucene.search.IndexSearcher; | ||
import org.apache.lucene.search.Query; | ||
import org.apache.lucene.search.TopDocs; | ||
import org.apache.lucene.store.Directory; | ||
import org.opensearch.common.lucene.Lucene; | ||
import org.opensearch.core.index.Index; | ||
import org.opensearch.index.query.QueryBuilders; | ||
import org.opensearch.index.query.QueryShardContext; | ||
import org.opensearch.script.DerivedFieldScript; | ||
|
||
import java.io.IOException; | ||
import java.util.ArrayList; | ||
import java.util.List; | ||
import java.util.Set; | ||
|
||
import org.mockito.Mockito; | ||
|
||
import static org.mockito.Mockito.when; | ||
|
||
public class DerivedFieldMapperQueryTests extends MapperServiceTestCase { | ||
|
||
// First element is the document ingested, other elements corresponds to value returned against a given derived field script | ||
// Raw Message, Request Succeeded (boolean), Timestamp (long), Client IP, Method, Request Size (double), Duration (long) | ||
private static final Object[][] raw_requests = new Object[][] { | ||
{ | ||
"40.135.0.0 GET /images/hm_bg.jpg?size=1.5KB HTTP/1.0 200 2024-03-20T08:30:45 1500", | ||
true, | ||
1710923445000L, | ||
"40.135.0.0", | ||
"GET", | ||
1.5, | ||
1500L }, | ||
{ | ||
"232.0.0.0 GET /images/hm_bg.jpg?size=2.3KB HTTP/1.0 400 2024-03-20T09:15:20 2300", | ||
false, | ||
1710926120000L, | ||
"232.0.0.0", | ||
"GET", | ||
2.3, | ||
2300L }, | ||
{ | ||
"26.1.0.0 DELETE /images/hm_bg.jpg?size=3.7KB HTTP/1.0 200 2024-03-20T10:05:55 3700", | ||
true, | ||
1710929155000L, | ||
"26.1.0.0", | ||
"DELETE", | ||
3.7, | ||
3700L }, | ||
{ | ||
"247.37.0.0 GET /french/splash_inet.html?size=4.1KB HTTP/1.0 400 2024-03-20T11:20:10 4100", | ||
false, | ||
1710933610000L, | ||
"247.37.0.0", | ||
"GET", | ||
4.1, | ||
4100L }, | ||
{ | ||
"247.37.0.0 DELETE /french/splash_inet.html?size=5.8KB HTTP/1.0 400 2024-03-20T12:45:30 5800", | ||
false, | ||
1710938730000L, | ||
"247.37.0.0", | ||
"DELETE", | ||
5.8, | ||
5800L }, | ||
{ | ||
"10.20.30.40 GET /path/to/resource?size=6.3KB HTTP/1.0 200 2024-03-20T13:10:15 6300", | ||
true, | ||
1710940215000L, | ||
"10.20.30.40", | ||
"GET", | ||
6.3, | ||
6300L }, | ||
{ | ||
"50.60.70.80 GET /path/to/resource?size=7.2KB HTTP/1.0 404 2024-03-20T14:20:50 7200", | ||
false, | ||
1710944450000L, | ||
"50.60.70.80", | ||
"GET", | ||
7.2, | ||
7200L }, | ||
{ | ||
"127.0.0.1 PUT /path/to/resource?size=8.9KB HTTP/1.0 500 2024-03-20T15:30:25 8900", | ||
false, | ||
1710948625000L, | ||
"127.0.0.1", | ||
"PUT", | ||
8.9, | ||
8900L }, | ||
{ | ||
"127.0.0.1 GET /path/to/resource?size=9.4KB HTTP/1.0 200 2024-03-20T16:40:15 9400", | ||
true, | ||
1710952815000L, | ||
"127.0.0.1", | ||
"GET", | ||
9.4, | ||
9400L }, | ||
{ | ||
"192.168.1.1 GET /path/to/resource?size=10.7KB HTTP/1.0 400 2024-03-20T17:50:40 10700", | ||
false, | ||
1710957040000L, | ||
"192.168.1.1", | ||
"GET", | ||
10.7, | ||
10700L } }; | ||
|
||
public void testAllPossibleQueriesOnDerivedFields() throws IOException { | ||
MapperService mapperService = createMapperService(topMapping(b -> { | ||
b.startObject("properties"); | ||
{ | ||
b.startObject("raw_message"); | ||
{ | ||
b.field("type", "text"); | ||
} | ||
b.endObject(); | ||
} | ||
b.endObject(); | ||
b.startObject("derived"); | ||
{ | ||
b.startObject("request_succeeded"); | ||
{ | ||
b.field("type", "boolean"); | ||
b.field("script", ""); | ||
} | ||
b.endObject(); | ||
b.startObject("@timestamp"); | ||
{ | ||
b.field("type", "date"); | ||
b.field("script", ""); | ||
} | ||
b.endObject(); | ||
b.startObject("client_ip"); | ||
{ | ||
b.field("type", "ip"); | ||
b.field("script", ""); | ||
} | ||
b.endObject(); | ||
b.startObject("method"); | ||
{ | ||
b.field("type", "keyword"); | ||
b.field("script", ""); | ||
} | ||
b.endObject(); | ||
b.startObject("request_size"); | ||
{ | ||
b.field("type", "double"); | ||
b.field("script", ""); | ||
} | ||
b.endObject(); | ||
b.startObject("duration"); | ||
{ | ||
b.field("type", "long"); | ||
b.field("script", ""); | ||
} | ||
b.endObject(); | ||
} | ||
b.endObject(); | ||
})); | ||
|
||
List<Document> docs = new ArrayList<>(); | ||
for (Object[] request : raw_requests) { | ||
Document document = new Document(); | ||
document.add(new TextField("raw_message", (String) request[0], Field.Store.YES)); | ||
docs.add(document); | ||
} | ||
|
||
int[] scriptIndex = { 1 }; | ||
|
||
// Mock DerivedFieldScript.Factory | ||
DerivedFieldScript.Factory factory = (params, lookup) -> (DerivedFieldScript.LeafFactory) ctx -> new DerivedFieldScript( | ||
params, | ||
lookup, | ||
ctx | ||
) { | ||
int docId = 0; | ||
|
||
@Override | ||
public void setDocument(int docId) { | ||
super.setDocument(docId); | ||
this.docId = docId; | ||
} | ||
|
||
@Override | ||
public Object execute() { | ||
return raw_requests[docId][scriptIndex[0]]; | ||
} | ||
}; | ||
|
||
QueryShardContext queryShardContext = createQueryShardContext(mapperService); | ||
when(queryShardContext.compile(Mockito.any(), Mockito.any())).thenReturn(factory); | ||
when(queryShardContext.sourcePath("raw_message")).thenReturn(Set.of("raw_message")); | ||
when(queryShardContext.index()).thenReturn(new Index("test_index", "uuid")); | ||
|
||
// Index and Search | ||
try (Directory dir = newDirectory()) { | ||
IndexWriter iw = new IndexWriter(dir, new IndexWriterConfig(Lucene.STANDARD_ANALYZER)); | ||
for (Document d : docs) { | ||
iw.addDocument(d); | ||
} | ||
try (IndexReader reader = DirectoryReader.open(iw)) { | ||
iw.close(); | ||
|
||
IndexSearcher searcher = new IndexSearcher(reader); | ||
Query query = QueryBuilders.termQuery("request_succeeded", "true").toQuery(queryShardContext); | ||
TopDocs topDocs = searcher.search(query, 10); | ||
assertEquals(4, topDocs.totalHits.value); | ||
|
||
// IP Field Term Query | ||
scriptIndex[0] = 3; | ||
query = QueryBuilders.termQuery("client_ip", "192.168.0.0/16").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(1, topDocs.totalHits.value); | ||
|
||
scriptIndex[0] = 4; | ||
query = QueryBuilders.termsQuery("method", "DELETE", "PUT").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(3, topDocs.totalHits.value); | ||
|
||
query = QueryBuilders.termsQuery("method", "delete").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(0, topDocs.totalHits.value); | ||
|
||
query = QueryBuilders.termQuery("method", "delete").caseInsensitive(true).toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(2, topDocs.totalHits.value); | ||
|
||
// Range queries of types - date, long and double | ||
scriptIndex[0] = 2; | ||
query = QueryBuilders.rangeQuery("@timestamp").from("2024-03-20T14:20:50").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(4, topDocs.totalHits.value); | ||
|
||
scriptIndex[0] = 5; | ||
query = QueryBuilders.rangeQuery("request_size").from("4.1").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(7, topDocs.totalHits.value); | ||
|
||
scriptIndex[0] = 6; | ||
query = QueryBuilders.rangeQuery("duration").from("5800").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(6, topDocs.totalHits.value); | ||
|
||
scriptIndex[0] = 4; | ||
|
||
// Prefix Query | ||
query = QueryBuilders.prefixQuery("method", "DE").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(2, topDocs.totalHits.value); | ||
|
||
scriptIndex[0] = 4; | ||
query = QueryBuilders.wildcardQuery("method", "G*").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(7, topDocs.totalHits.value); | ||
|
||
// Regexp Query | ||
scriptIndex[0] = 4; | ||
query = QueryBuilders.regexpQuery("method", ".*LET.*").toQuery(queryShardContext); | ||
topDocs = searcher.search(query, 10); | ||
assertEquals(2, topDocs.totalHits.value); | ||
} | ||
} | ||
} | ||
} |
Oops, something went wrong.