/*
 * 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; you may not use this file except in compliance with the Elastic License
 * 2.0.
 */

package org.elasticsearch.xpack.application.rules;

import org.elasticsearch.common.bytes.BytesArray;
import org.elasticsearch.common.bytes.BytesReference;
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.xcontent.XContentHelper;
import org.elasticsearch.search.SearchModule;
import org.elasticsearch.test.ESTestCase;
import org.elasticsearch.xcontent.ToXContent;
import org.elasticsearch.xcontent.XContentParser;
import org.elasticsearch.xcontent.XContentType;
import org.elasticsearch.xpack.application.EnterpriseSearchModuleTestUtils;
import org.junit.Before;

import java.io.IOException;
import java.util.List;

import static java.util.Collections.emptyList;
import static org.elasticsearch.common.xcontent.XContentHelper.toXContent;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertToXContentEquivalent;
import static org.hamcrest.CoreMatchers.equalTo;

public class QueryRulesetTests extends ESTestCase {
    private NamedWriteableRegistry namedWriteableRegistry;

    @Before
    public void registerNamedObjects() {
        SearchModule searchModule = new SearchModule(Settings.EMPTY, emptyList());

        List<NamedWriteableRegistry.Entry> namedWriteables = searchModule.getNamedWriteables();
        namedWriteableRegistry = new NamedWriteableRegistry(namedWriteables);
    }

    public final void testRandomSerialization() throws IOException {
        for (int runs = 0; runs < 10; runs++) {
            QueryRuleset testInstance = EnterpriseSearchModuleTestUtils.randomQueryRuleset();
            assertTransportSerialization(testInstance);
            assertXContent(testInstance, randomBoolean());
        }
    }

    public void testToXContent() throws IOException {
        String content = XContentHelper.stripWhitespace("""
            {
              "ruleset_id": "my_ruleset_id",
              "rules": [
                {
                  "rule_id": "my_query_rule1",
                  "type": "pinned",
                  "criteria": [ {"type": "exact", "metadata": "query_string", "values": ["foo", "bar"]} ],
                  "actions": {
                    "ids": ["id1", "id2"]
                  }
                },
                {
                  "rule_id": "my_query_rule2",
                  "type": "pinned",
                  "criteria": [ {"type": "exact", "metadata": "query_string", "values": ["baz"]} ],
                  "actions": {
                    "ids": ["id3", "id4"]
                  }
                }
              ]
            }""");

        QueryRuleset queryRuleset = QueryRuleset.fromXContentBytes("my_ruleset_id", new BytesArray(content), XContentType.JSON);
        boolean humanReadable = true;
        BytesReference originalBytes = toShuffledXContent(queryRuleset, XContentType.JSON, ToXContent.EMPTY_PARAMS, humanReadable);
        QueryRuleset parsed;
        try (XContentParser parser = createParser(XContentType.JSON.xContent(), originalBytes)) {
            parsed = QueryRuleset.fromXContent("my_ruleset_id", parser);
        }
        assertToXContentEquivalent(originalBytes, toXContent(parsed, XContentType.JSON, humanReadable), XContentType.JSON);
    }

    public void testToXContentInvalidQueryRulesetId() throws IOException {
        String content = XContentHelper.stripWhitespace("""
            {
              "ruleset_id": "my_ruleset_id",
              "rules": [
                {
                  "rule_id": "my_query_rule1",
                  "type": "pinned",
                  "criteria": [ {"type": "exact", "metadata": "query_string", "values": ["foo", "bar"]} ],
                  "actions": {
                    "ids": ["id1", "id2"]
                  }
                },
                {
                  "rule_id": "my_query_rule2",
                  "type": "pinned",
                  "criteria": [ {"type": "exact", "metadata": "query_string", "values": ["baz"]} ],
                  "actions": {
                    "ids": ["id3", "id4"]
                  }
                }
              ]
            }""");
        expectThrows(
            IllegalArgumentException.class,
            () -> QueryRuleset.fromXContentBytes("not_my_ruleset_id", new BytesArray(content), XContentType.JSON)
        );
    }

    private void assertXContent(QueryRuleset queryRuleset, boolean humanReadable) throws IOException {
        BytesReference originalBytes = toShuffledXContent(queryRuleset, XContentType.JSON, ToXContent.EMPTY_PARAMS, humanReadable);
        QueryRuleset parsed;
        try (XContentParser parser = createParser(XContentType.JSON.xContent(), originalBytes)) {
            parsed = QueryRuleset.fromXContent(queryRuleset.id(), parser);
        }
        assertToXContentEquivalent(originalBytes, toXContent(parsed, XContentType.JSON, humanReadable), XContentType.JSON);
    }

    private void assertTransportSerialization(QueryRuleset testInstance) throws IOException {
        QueryRuleset deserializedInstance = copyInstance(testInstance);
        assertNotSame(testInstance, deserializedInstance);
        assertThat(testInstance, equalTo(deserializedInstance));
    }

    private QueryRuleset copyInstance(QueryRuleset instance) throws IOException {
        return copyWriteable(instance, namedWriteableRegistry, QueryRuleset::new);
    }
}
