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

org.apache.druid.server.http.ClusterResource Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */

package org.apache.druid.server.http;

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.google.common.collect.Collections2;
import com.google.common.collect.ImmutableMap;
import com.google.inject.Inject;
import com.sun.jersey.spi.container.ResourceFilters;
import org.apache.druid.discovery.DiscoveryDruidNode;
import org.apache.druid.discovery.DruidNodeDiscoveryProvider;
import org.apache.druid.discovery.NodeRole;
import org.apache.druid.guice.LazySingleton;
import org.apache.druid.server.DruidNode;
import org.apache.druid.server.http.security.StateResourceFilter;

import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;
import javax.ws.rs.QueryParam;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import java.util.Arrays;
import java.util.Collection;

/**
 */
@Path("/druid/coordinator/v1/cluster")
@LazySingleton
@ResourceFilters(StateResourceFilter.class)
public class ClusterResource
{
  private final DruidNodeDiscoveryProvider druidNodeDiscoveryProvider;

  @Inject
  public ClusterResource(DruidNodeDiscoveryProvider discoveryProvider)
  {
    this.druidNodeDiscoveryProvider = discoveryProvider;
  }

  @GET
  @Produces(MediaType.APPLICATION_JSON)
  public Response getClusterServers(@QueryParam("full") boolean full)
  {
    ImmutableMap.Builder entityBuilder = new ImmutableMap.Builder<>();

    entityBuilder.put(NodeRole.COORDINATOR, getNodes(NodeRole.COORDINATOR, full));
    entityBuilder.put(NodeRole.OVERLORD, getNodes(NodeRole.OVERLORD, full));
    entityBuilder.put(NodeRole.BROKER, getNodes(NodeRole.BROKER, full));
    entityBuilder.put(NodeRole.HISTORICAL, getNodes(NodeRole.HISTORICAL, full));

    Collection mmNodes = getNodes(NodeRole.MIDDLE_MANAGER, full);
    if (!mmNodes.isEmpty()) {
      entityBuilder.put(NodeRole.MIDDLE_MANAGER, mmNodes);
    }

    Collection indexerNodes = getNodes(NodeRole.INDEXER, full);
    if (!indexerNodes.isEmpty()) {
      entityBuilder.put(NodeRole.INDEXER, indexerNodes);
    }

    Collection routerNodes = getNodes(NodeRole.ROUTER, full);
    if (!routerNodes.isEmpty()) {
      entityBuilder.put(NodeRole.ROUTER, routerNodes);
    }

    return Response.status(Response.Status.OK).entity(entityBuilder.build()).build();
  }

  @GET
  @Produces({MediaType.APPLICATION_JSON})
  @Path("/{nodeRole}")
  public Response getClusterServers(@PathParam("nodeRole") NodeRole nodeRole, @QueryParam("full") boolean full)
  {
    if (nodeRole == null) {
      return Response.serverError()
                     .status(Response.Status.BAD_REQUEST)
                     .entity("Invalid nodeRole of null. Valid node roles are " + Arrays.toString(NodeRole.values()))
                     .build();
    } else {
      return Response.status(Response.Status.OK).entity(getNodes(nodeRole, full)).build();
    }
  }

  private Collection getNodes(NodeRole nodeRole, boolean full)
  {
    Collection discoveryDruidNodes = druidNodeDiscoveryProvider.getForNodeRole(nodeRole)
                                                                                   .getAllNodes();
    if (full) {
      return (Collection) discoveryDruidNodes;
    } else {
      return Collections2.transform(
          discoveryDruidNodes,
          (discoveryDruidNode) -> Node.from(discoveryDruidNode.getDruidNode())
      );
    }
  }

  @JsonInclude(JsonInclude.Include.NON_NULL)
  private static class Node
  {
    private final String host;
    private final String service;
    private final Integer plaintextPort;
    private final Integer tlsPort;

    @JsonCreator
    public Node(String host, String service, Integer plaintextPort, Integer tlsPort)
    {
      this.host = host;
      this.service = service;
      this.plaintextPort = plaintextPort;
      this.tlsPort = tlsPort;
    }

    @JsonProperty
    public String getHost()
    {
      return host;
    }

    @JsonProperty
    public String getService()
    {
      return service;
    }

    @JsonProperty
    public Integer getPlaintextPort()
    {
      return plaintextPort;
    }

    @JsonProperty
    public Integer getTlsPort()
    {
      return tlsPort;
    }

    public static Node from(DruidNode druidNode)
    {
      return new Node(
          druidNode.getHost(),
          druidNode.getServiceName(),
          druidNode.getPlaintextPort() > 0 ? druidNode.getPlaintextPort() : null,
          druidNode.getTlsPort() > 0 ? druidNode.getTlsPort() : null
      );
    }
  }
}