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

org.flyte.jflyte.utils.FlyteAdminClient Maven / Gradle / Ivy

/*
 * Copyright 2020-2023 Flyte Authors.
 *
 * Licensed 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.flyte.jflyte.utils;

import static com.google.common.base.Verify.verifyNotNull;

import com.google.common.annotations.VisibleForTesting;
import flyteidl.admin.Common;
import flyteidl.admin.Common.ResourceListRequest;
import flyteidl.admin.ExecutionOuterClass;
import flyteidl.admin.LaunchPlanOuterClass;
import flyteidl.admin.TaskOuterClass;
import flyteidl.admin.WorkflowOuterClass;
import flyteidl.service.AdminServiceGrpc;
import io.grpc.Channel;
import io.grpc.ClientInterceptor;
import io.grpc.ClientInterceptors;
import io.grpc.ManagedChannel;
import io.grpc.ManagedChannelBuilder;
import io.grpc.Status;
import io.grpc.StatusRuntimeException;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import javax.annotation.Nullable;
import org.flyte.api.v1.LaunchPlan;
import org.flyte.api.v1.LaunchPlanIdentifier;
import org.flyte.api.v1.NamedEntityIdentifier;
import org.flyte.api.v1.TaskIdentifier;
import org.flyte.api.v1.TaskTemplate;
import org.flyte.api.v1.WorkflowIdentifier;
import org.flyte.api.v1.WorkflowTemplate;
import org.flyte.jflyte.api.TokenSource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * This is a thin synchronous wrapper around the auto-generated gRPC stubs for communicating with
 * the admin service.
 */
public class FlyteAdminClient implements AutoCloseable {

  private static final Logger LOG = LoggerFactory.getLogger(FlyteAdminClient.class);
  static final String TRIGGERING_PRINCIPAL = "sdk";
  static final int USER_TRIGGERED_EXECUTION_NESTING = 0;

  private final AdminServiceGrpc.AdminServiceBlockingStub stub;
  private final ManagedChannel channel;
  private final GrpcRetries retries;

  @VisibleForTesting
  FlyteAdminClient(
      AdminServiceGrpc.AdminServiceBlockingStub stub, ManagedChannel channel, GrpcRetries retries) {
    this.stub = stub;
    this.channel = channel;
    this.retries = retries;
  }

  public static FlyteAdminClient create(
      String target, boolean insecure, @Nullable TokenSource tokenSource) {
    ManagedChannelBuilder builder = ManagedChannelBuilder.forTarget(target);

    if (insecure) {
      builder.usePlaintext();
    }

    ManagedChannel originChannel = builder.build();
    GrpcRetries retries = GrpcRetries.create();
    if (tokenSource == null) {
      // In case of no tokenSource, no need to intercept the grpc call.
      return new FlyteAdminClient(
          AdminServiceGrpc.newBlockingStub(originChannel), originChannel, retries);
    } else {
      ClientInterceptor interceptor = new AuthorizationHeaderInterceptor(tokenSource);
      Channel channel = ClientInterceptors.intercept(originChannel, interceptor);
      return new FlyteAdminClient(
          AdminServiceGrpc.newBlockingStub(channel), originChannel, retries);
    }
  }

  public void createTask(TaskIdentifier id, TaskTemplate template) {
    LOG.debug("createTask {}", id);

    TaskOuterClass.TaskCreateRequest request =
        TaskOuterClass.TaskCreateRequest.newBuilder()
            .setId(ProtoUtil.serialize(id))
            .setSpec(
                TaskOuterClass.TaskSpec.newBuilder()
                    .setTemplate(ProtoUtil.serialize(template))
                    .build())
            .build();

    idempotentCreate("createTask", id, () -> stub.createTask(request));
  }

  public void createWorkflow(
      WorkflowIdentifier id,
      WorkflowTemplate template,
      Map subWorkflows) {
    LOG.debug("createWorkflow {}", id);

    WorkflowOuterClass.WorkflowSpec.Builder specBuilder =
        WorkflowOuterClass.WorkflowSpec.newBuilder().setTemplate(ProtoUtil.serialize(template));

    subWorkflows.forEach(
        (subWorkflowId, subWorkflow) ->
            specBuilder.addSubWorkflows(ProtoUtil.serialize(subWorkflowId, subWorkflow)));

    WorkflowOuterClass.WorkflowCreateRequest request =
        WorkflowOuterClass.WorkflowCreateRequest.newBuilder()
            .setId(ProtoUtil.serialize(id))
            .setSpec(specBuilder.build())
            .build();

    idempotentCreate("createWorkflow", id, () -> stub.createWorkflow(request));
  }

  public void createLaunchPlan(LaunchPlanIdentifier id, LaunchPlan launchPlan) {
    LOG.debug("createLaunchPlan {}", id);

    LaunchPlanOuterClass.LaunchPlanCreateRequest request =
        LaunchPlanOuterClass.LaunchPlanCreateRequest.newBuilder()
            .setId(ProtoUtil.serialize(id))
            .setSpec(ProtoUtil.serialize(launchPlan))
            .build();

    idempotentCreate("createLaunchPlan", id, () -> stub.createLaunchPlan(request));
  }

  void createExecution(String domain, String project, LaunchPlanIdentifier launchPlanId) {
    LOG.debug("createExecution {} {} {}", domain, project, launchPlanId);

    ExecutionOuterClass.ExecutionMetadata metadata =
        ExecutionOuterClass.ExecutionMetadata.newBuilder()
            .setMode(ExecutionOuterClass.ExecutionMetadata.ExecutionMode.MANUAL)
            .setPrincipal(TRIGGERING_PRINCIPAL)
            .setNesting(USER_TRIGGERED_EXECUTION_NESTING)
            .build();

    ExecutionOuterClass.ExecutionSpec spec =
        ExecutionOuterClass.ExecutionSpec.newBuilder()
            .setLaunchPlan(ProtoUtil.serialize(launchPlanId))
            .setMetadata(metadata)
            .build();

    ExecutionOuterClass.ExecutionCreateRequest request =
        ExecutionOuterClass.ExecutionCreateRequest.newBuilder()
            .setDomain(domain)
            .setProject(project)
            .setSpec(spec)
            .build();

    // NB: createExecution doesn't compare payloads when throwing ALREADY_EXISTS, so only using
    // retries

    // create operation is idempotent, so it's fine to retry
    ExecutionOuterClass.ExecutionCreateResponse response =
        retries.retry(() -> stub.createExecution(request));

    verifyNotNull(
        response,
        "Unexpected null response when creating execution %s on project %s domain %s",
        launchPlanId,
        project,
        domain);
  }

  @Nullable
  public TaskIdentifier fetchLatestTaskId(NamedEntityIdentifier taskId) {
    return fetchLatestResource(
        taskId,
        request -> stub.listTasks(request).getTasksList(),
        task -> ProtoUtil.deserializeTaskId(task.getId()));
  }

  @Nullable
  public TaskTemplate fetchLatestTaskTemplate(NamedEntityIdentifier taskId) {
    return fetchLatestResource(
        taskId,
        request -> stub.listTasks(request).getTasksList(),
        task -> ProtoUtil.deserialize(task.getClosure().getCompiledTask().getTemplate()));
  }

  @Nullable
  public WorkflowIdentifier fetchLatestWorkflowId(NamedEntityIdentifier workflowId) {
    return fetchLatestResource(
        workflowId,
        request -> stub.listWorkflows(request).getWorkflowsList(),
        workflow -> ProtoUtil.deserializeWorkflowId(workflow.getId()));
  }

  @Nullable
  public LaunchPlanIdentifier fetchLatestLaunchPlanId(NamedEntityIdentifier launchPlanId) {
    return fetchLatestResource(
        launchPlanId,
        request -> stub.listLaunchPlans(request).getLaunchPlansList(),
        launchPlan -> ProtoUtil.deserializeLaunchPlanId(launchPlan.getId()));
  }

  @Nullable
  private  T fetchLatestResource(
      NamedEntityIdentifier nameId,
      Function> performRequestFn,
      Function deserializeFn) {
    ResourceListRequest request =
        ResourceListRequest.newBuilder()
            .setLimit(1)
            .setId(ProtoUtil.serialize(nameId))
            .setSortBy(
                Common.Sort.newBuilder()
                    .setKey("created_at")
                    .setDirection(Common.Sort.Direction.DESCENDING)
                    .build())
            .build();

    List list = retries.retry(() -> performRequestFn.apply(request));

    if (list.isEmpty()) {
      return null;
    }

    return deserializeFn.apply(list.get(0));
  }

  private  void idempotentCreate(String label, Object id, GrpcRetries.Retryable retryable) {
    try {
      // create operation is idempotent, so it's fine to retry
      T response = retries.retry(retryable);

      verifyNotNull(response, "%s %s: Unexpected null response", label, id);
    } catch (StatusRuntimeException e) {
      // flyteadmin uses ALREADY_EXISTS only if payload is identical, except for executions
      if (e.getStatus().getCode() == Status.Code.ALREADY_EXISTS) {
        LOG.debug("{} {}: ALREADY_EXISTS with identical payload", label, id);
      } else {
        throw e;
      }
    }
  }

  @Override
  public void close() {
    if (channel != null) {
      channel.shutdown();
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy