All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.elasticsearch.test.XContentTestUtils Maven / Gradle / Ivy

There is a newer version: 8.16.0
Show newest version
/*
 * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
 * or more contributor license agreements. Licensed under the Elastic License
 * 2.0 and the Server Side Public License, v 1; you may not use this file except
 * in compliance with, at your election, the Elastic License 2.0 or the Server
 * Side Public License, v 1.
 */

package org.elasticsearch.test;

import org.elasticsearch.common.bytes.BytesReference;
import org.elasticsearch.common.xcontent.ChunkedToXContent;
import org.elasticsearch.common.xcontent.XContentHelper;
import org.elasticsearch.test.rest.ObjectPath;
import org.elasticsearch.xcontent.DeprecationHandler;
import org.elasticsearch.xcontent.NamedXContentRegistry;
import org.elasticsearch.xcontent.ToXContent;
import org.elasticsearch.xcontent.XContent;
import org.elasticsearch.xcontent.XContentBuilder;
import org.elasticsearch.xcontent.XContentFactory;
import org.elasticsearch.xcontent.XContentParser;
import org.elasticsearch.xcontent.XContentType;
import org.elasticsearch.xcontent.json.JsonXContent;

import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Deque;
import java.util.List;
import java.util.Map;
import java.util.Random;
import java.util.Stack;
import java.util.function.Predicate;
import java.util.function.Supplier;
import java.util.stream.Collectors;

import static com.carrotsearch.randomizedtesting.generators.RandomStrings.randomAsciiOfLength;
import static org.elasticsearch.common.xcontent.XContentHelper.createParser;
import static org.elasticsearch.xcontent.ToXContent.EMPTY_PARAMS;

public final class XContentTestUtils {
    private XContentTestUtils() {

    }

    public static Map convertToMap(ChunkedToXContent chunkedToXContent) throws IOException {
        return convertToMap(ChunkedToXContent.wrapAsToXContent(chunkedToXContent));
    }

    public static Map convertToMap(ToXContent part) throws IOException {
        XContentBuilder builder = XContentFactory.jsonBuilder();
        if (part.isFragment()) {
            builder.startObject();
            part.toXContent(builder, EMPTY_PARAMS);
            builder.endObject();
        } else {
            part.toXContent(builder, EMPTY_PARAMS);
        }
        return XContentHelper.convertToMap(BytesReference.bytes(builder), false, builder.contentType()).v2();
    }

    public static BytesReference convertToXContent(Map map, XContentType xContentType) throws IOException {
        try (XContentBuilder builder = XContentFactory.contentBuilder(xContentType)) {
            builder.map(map);
            return BytesReference.bytes(builder);
        }
    }

    /**
     * Compares two maps generated from XContentObjects. The order of elements in arrays is ignored.
     *
     * @return null if maps are equal or path to the element where the difference was found
     */
    public static String differenceBetweenMapsIgnoringArrayOrder(Map first, Map second) {
        return differenceBetweenMapsIgnoringArrayOrder("", first, second);
    }

    private static String differenceBetweenMapsIgnoringArrayOrder(String path, Map first, Map second) {
        if (first.size() != second.size()) {
            return path + ": sizes of the maps don't match: " + first.size() + " != " + second.size();
        }

        for (String key : first.keySet()) {
            String reason = differenceBetweenObjectsIgnoringArrayOrder(path + "/" + key, first.get(key), second.get(key));
            if (reason != null) {
                return reason;
            }
        }
        return null;
    }

    @SuppressWarnings("unchecked")
    private static String differenceBetweenObjectsIgnoringArrayOrder(String path, Object first, Object second) {
        if (first == null) {
            if (second == null) {
                return null;
            } else {
                return path + ": first element is null, the second element is not null";
            }
        } else if (first instanceof List) {
            if (second instanceof List) {
                List secondList = new ArrayList<>((List) second);
                List firstList = (List) first;
                if (firstList.size() == secondList.size()) {
                    String reason = path + ": no matches found";
                    for (Object firstObj : firstList) {
                        boolean found = false;
                        for (Object secondObj : secondList) {
                            reason = differenceBetweenObjectsIgnoringArrayOrder(path + "/*", firstObj, secondObj);
                            if (reason == null) {
                                secondList.remove(secondObj);
                                found = true;
                                break;
                            }
                        }
                        if (found == false) {
                            return reason;
                        }
                    }
                    if (secondList.isEmpty()) {
                        return null;
                    } else {
                        return path + ": the second list is not empty";
                    }
                } else {
                    return path + ": sizes of the arrays don't match: " + firstList.size() + " != " + secondList.size();
                }
            } else {
                return path + ": the second element is not an array";
            }
        } else if (first instanceof Map) {
            if (second instanceof Map) {
                return differenceBetweenMapsIgnoringArrayOrder(path, (Map) first, (Map) second);
            } else {
                return path + ": the second element is not a map (got " + second + ")";
            }
        } else {
            if (first.equals(second)) {
                return null;
            } else {
                return path + ": the elements don't match: [" + first + "] != [" + second + "]";
            }

        }
    }

    /**
     * This method takes the input xContent data and adds a random field value, inner object or array into each
     * json object. This can e.g. be used to test if parsers that handle the resulting xContent can handle the
     * augmented xContent correctly, for example when testing lenient parsing.
     *
     * If the xContent output contains objects that should be skipped of such treatment, an optional filtering
     * {@link Predicate} can be supplied that checks xContent paths that should be excluded from this treatment.
     *
     * This predicate should check the xContent path that we want to insert to and return {@code true} if the
     * path should be excluded. Paths are string concatenating field names and array indices, so e.g. in:
     *
     * 
     * {
     *      "foo1 : {
     *          "bar" : [
     *              { ... },
     *              { ... },
     *              {
     *                  "baz" : {
     *                      // insert here
     *                  }
     *              }
     *          ]
     *      }
     * }
     * 
* * "foo1.bar.2.baz" would point to the desired insert location. * * To exclude inserting into the "foo1" object we would user a {@link Predicate} like *
     * {@code
     *      (path) -> path.endsWith("foo1")
     * }
     * 
* * or if we don't want any random insertions in the "foo1" tree we could use *
     * {@code
     *      (path) -> path.contains("foo1")
     * }
     * 
*/ public static BytesReference insertRandomFields( XContentType contentType, BytesReference xContent, Predicate excludeFilter, Random random ) throws IOException { List insertPaths; // we can use NamedXContentRegistry.EMPTY here because we only traverse the xContent once and don't use it try ( XContentParser parser = createParser( NamedXContentRegistry.EMPTY, DeprecationHandler.THROW_UNSUPPORTED_OPERATION, xContent, contentType ) ) { parser.nextToken(); List possiblePaths = XContentTestUtils.getInsertPaths(parser, new ArrayDeque<>()); if (excludeFilter == null) { insertPaths = possiblePaths; } else { insertPaths = new ArrayList<>(); possiblePaths.stream().filter(excludeFilter.negate()).forEach(insertPaths::add); } } Supplier value = () -> { List randomValues = RandomObjects.randomStoredFieldValues(random, contentType).v1(); if (random.nextBoolean()) { return randomValues.get(0); } else { if (random.nextBoolean()) { return randomValues.stream().collect(Collectors.toMap(obj -> randomAsciiOfLength(random, 10), obj -> obj)); } else { return randomValues; } } }; return BytesReference.bytes( XContentTestUtils.insertIntoXContent( contentType.xContent(), xContent, insertPaths, () -> randomAsciiOfLength(random, 10), value ) ); } /** * This utility method takes an XContentParser and walks the xContent structure to find all * possible paths to where a new object or array starts. This can be used in tests that add random * xContent values to test parsing code for errors or to check their robustness against new fields. * * The path uses dot separated fieldnames and numbers for array indices, similar to what we do in * {@link ObjectPath}. * * The {@link Stack} passed in should initially be empty, it gets pushed to by recursive calls * * As an example, the following json xContent: *
     *     {
     *         "foo" : "bar",
     *         "foo1" : [ 1, { "foo2" : "baz" }, 3, 4]
     *         "foo3" : {
     *             "foo4" : {
     *                  "foo5": "buzz"
     *             }
     *         }
     *     }
     * 
* * Would return the following list: * *
    *
  • "" (the empty string is the path to the root object)
  • *
  • "foo1.1"
  • *
  • "foo3
  • *
  • "foo3.foo4
  • *
*/ static List getInsertPaths(XContentParser parser, Deque currentPath) throws IOException { assert parser.currentToken() == XContentParser.Token.START_OBJECT || parser.currentToken() == XContentParser.Token.START_ARRAY : "should only be called when new objects or arrays start"; List validPaths = new ArrayList<>(); // parser.currentName() can be null for root object and unnamed objects in arrays if (parser.currentName() != null) { // dots in randomized field names need to be escaped, we use that character as the path separator currentPath.addLast(parser.currentName().replace(".", "\\.")); } if (parser.currentToken() == XContentParser.Token.START_OBJECT) { validPaths.add(String.join(".", currentPath)); while (parser.nextToken() != XContentParser.Token.END_OBJECT) { if (parser.currentToken() == XContentParser.Token.START_OBJECT || parser.currentToken() == XContentParser.Token.START_ARRAY) { validPaths.addAll(getInsertPaths(parser, currentPath)); } } } else if (parser.currentToken() == XContentParser.Token.START_ARRAY) { int itemCount = 0; while (parser.nextToken() != XContentParser.Token.END_ARRAY) { if (parser.currentToken() == XContentParser.Token.START_OBJECT || parser.currentToken() == XContentParser.Token.START_ARRAY) { currentPath.addLast(Integer.toString(itemCount)); validPaths.addAll(getInsertPaths(parser, currentPath)); currentPath.removeLast(); } itemCount++; } } if (parser.currentName() != null) { currentPath.removeLast(); } return validPaths; } /** * Inserts key/value pairs into xContent passed in as {@link BytesReference} and returns a new {@link XContentBuilder} * The paths argument uses dot separated fieldnames and numbers for array indices, similar to what we do in * {@link ObjectPath}. * The key/value arguments can suppliers that either return fixed or random values. */ public static XContentBuilder insertIntoXContent( XContent xContent, BytesReference original, List paths, Supplier key, Supplier value ) throws IOException { ObjectPath object = ObjectPath.createFromXContent(xContent, original); for (String path : paths) { Map insertMap = object.evaluate(path); insertMap.put(key.get(), value.get()); } return object.toXContentBuilder(xContent); } public static JsonMapView createJsonMapView(InputStream inputStream) { final Map responseMap = XContentHelper.convertToMap(JsonXContent.jsonXContent, inputStream, true); return new JsonMapView(responseMap); } public static class JsonMapView { private final Map map; public JsonMapView(Map map) { this.map = map; } @SuppressWarnings("unchecked") public T get(String path) { String[] keys = path.split("\\."); Object context = map; for (String key : keys) { if (context instanceof Map) { context = ((Map) context).get(key); } else if (context instanceof List) { context = ((List) context).get(Integer.parseInt(key)); } else { return null; // node does not exist } } return (T) context; } @Override public String toString() { return "JsonMapView{map=" + map + '}'; } } }