Skip to content

Commit

Permalink
Add tests for all supported query types for derived fields
Browse files Browse the repository at this point in the history
Signed-off-by: Rishabh Maurya <rishabhmaurya05@gmail.com>
  • Loading branch information
rishabhmaurya authored and qreshi committed Mar 21, 2024
1 parent 4c6aaa3 commit 3e3b6b4
Show file tree
Hide file tree
Showing 6 changed files with 326 additions and 44 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -56,12 +56,7 @@ private static DerivedFieldMapper toType(FieldMapper in) {
*/
public static class Builder extends ParametrizedFieldMapper.Builder {
// TODO: The type of parameter may change here if the actual underlying FieldType object is needed
private final Parameter<String> type = Parameter.stringParam(
"type",
false,
m -> toType(m).type,
"text"
);
private final Parameter<String> type = Parameter.stringParam("type", false, m -> toType(m).type, "text");

private final Parameter<Script> script = new Parameter<>(
"script",
Expand All @@ -83,9 +78,17 @@ protected List<Parameter<?>> getParameters() {
@Override
public DerivedFieldMapper build(BuilderContext context) {
FieldMapper fieldMapper = DerivedFieldSupportedTypes.getFieldMapperFromType(type.getValue(), name, context);
Function<Object, IndexableField> fieldFunction =
DerivedFieldSupportedTypes.getIndexableFieldGeneratorType(type.getValue(), name);
DerivedFieldType ft = new DerivedFieldType(buildFullName(context), type.getValue(), script.getValue(), fieldMapper, fieldFunction);
Function<Object, IndexableField> fieldFunction = DerivedFieldSupportedTypes.getIndexableFieldGeneratorType(
type.getValue(),
name
);
DerivedFieldType ft = new DerivedFieldType(
buildFullName(context),
type.getValue(),
script.getValue(),
fieldMapper,
fieldFunction
);
return new DerivedFieldMapper(name, ft, multiFieldsBuilder.build(this, context), copyTo.build(), this);
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -721,7 +721,8 @@ public void toXContent(XContentBuilder builder, Params params, ToXContent custom
doXContent(builder, params);

// sort the mappers so we get consistent serialization format
Mapper[] derivedSortedMappers = mappers.values().stream()
Mapper[] derivedSortedMappers = mappers.values()
.stream()
.filter(m -> m instanceof DerivedFieldMapper)
.toArray(size -> new Mapper[size]);
Arrays.sort(derivedSortedMappers, new Comparator<Mapper>() {
Expand All @@ -731,7 +732,8 @@ public int compare(Mapper o1, Mapper o2) {
}
});

Mapper[] sortedMappers = mappers.values().stream()
Mapper[] sortedMappers = mappers.values()
.stream()
.filter(m -> !(m instanceof DerivedFieldMapper))
.toArray(size -> new Mapper[size]);
Arrays.sort(sortedMappers, new Comparator<Mapper>() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -671,7 +671,9 @@ public final void parse(String name, ParserContext parserContext, Map<String, Ob
}
}
String type = (String) fieldNode.get("type");
if (paramsMap.get("type") == null) { fieldNode.remove("type"); }
if (paramsMap.get("type") == null) {
fieldNode.remove("type");
}

for (Iterator<Map.Entry<String, Object>> iterator = fieldNode.entrySet().iterator(); iterator.hasNext();) {
Map.Entry<String, Object> entry = iterator.next();
Expand Down
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);
}
}
}
}
Loading

0 comments on commit 3e3b6b4

Please sign in to comment.