/*
 * 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", the "GNU Affero General Public License v3.0 only", 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", the "GNU Affero General Public
 * License v3.0 only", or the "Server Side Public License, v 1".
 */

package org.elasticsearch.health.node.selection;

import org.elasticsearch.action.support.replication.ClusterStateCreationUtils;
import org.elasticsearch.cluster.ClusterState;
import org.elasticsearch.cluster.node.DiscoveryNode;
import org.elasticsearch.cluster.node.DiscoveryNodeRole;
import org.elasticsearch.cluster.node.DiscoveryNodeUtils;
import org.elasticsearch.persistent.ClusterPersistentTasksCustomMetadata;
import org.elasticsearch.test.ESTestCase;

import java.util.Set;

import static org.elasticsearch.persistent.PersistentTasksExecutor.NO_NODE_FOUND;
import static org.hamcrest.Matchers.equalTo;
import static org.hamcrest.Matchers.notNullValue;
import static org.hamcrest.Matchers.nullValue;

public class HealthNodeTests extends ESTestCase {

    private final DiscoveryNode node1 = DiscoveryNodeUtils.builder("node_1")
        .roles(Set.of(DiscoveryNodeRole.MASTER_ROLE, DiscoveryNodeRole.DATA_ROLE))
        .build();
    private final DiscoveryNode node2 = DiscoveryNodeUtils.builder("node_2")
        .roles(Set.of(DiscoveryNodeRole.MASTER_ROLE, DiscoveryNodeRole.DATA_ROLE))
        .build();
    private final DiscoveryNode[] allNodes = new DiscoveryNode[] { node1, node2 };

    public void testFindTask() {
        ClusterState state = ClusterStateCreationUtils.state(node1, node1, node1, allNodes);
        assertThat(HealthNode.findTask(state), notNullValue());
    }

    public void testFindNoTask() {
        ClusterState state = ClusterStateCreationUtils.state(node1, node1, allNodes);
        assertThat(HealthNode.findTask(state), nullValue());
    }

    public void testFindHealthNode() {
        ClusterState state = ClusterStateCreationUtils.state(node1, node1, node1, allNodes);
        assertThat(HealthNode.findHealthNode(state), equalTo(node1));
    }

    public void testFindHealthNodeNoTask() {
        ClusterState state = ClusterStateCreationUtils.state(node1, node1, allNodes);
        assertThat(HealthNode.findHealthNode(state), nullValue());
    }

    public void testfindHealthNodeNoAssignment() {
        ClusterPersistentTasksCustomMetadata.Builder tasks = ClusterPersistentTasksCustomMetadata.builder();
        tasks.addTask(HealthNode.TASK_NAME, HealthNode.TASK_NAME, HealthNodeTaskParams.INSTANCE, NO_NODE_FOUND);
        ClusterState state = ClusterStateCreationUtils.state(node1, node1, allNodes)
            .copyAndUpdateMetadata(b -> b.putCustom(ClusterPersistentTasksCustomMetadata.TYPE, tasks.build()));
        assertThat(HealthNode.findHealthNode(state), nullValue());
    }

    public void testFindHealthNodeMissingNode() {
        ClusterState state = ClusterStateCreationUtils.state(node1, node1);
        assertThat(HealthNode.findHealthNode(state), nullValue());
    }
}
