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

io.cdap.cdap.internal.app.runtime.service.http.BasicHttpServiceContext Maven / Gradle / Ivy

The newest version!
/*
 * Copyright © 2014-2018 Cask Data, Inc.
 *
 * 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 io.cdap.cdap.internal.app.runtime.service.http;

import io.cdap.cdap.api.artifact.ArtifactInfo;
import io.cdap.cdap.api.artifact.ArtifactManager;
import io.cdap.cdap.api.artifact.CloseableClassLoader;
import io.cdap.cdap.api.metadata.MetadataReader;
import io.cdap.cdap.api.metrics.MetricsCollectionService;
import io.cdap.cdap.api.plugin.PluginConfigurer;
import io.cdap.cdap.api.security.AccessException;
import io.cdap.cdap.api.security.store.SecureStore;
import io.cdap.cdap.api.security.store.SecureStoreManager;
import io.cdap.cdap.api.service.http.HttpServiceContext;
import io.cdap.cdap.api.service.http.HttpServiceHandlerSpecification;
import io.cdap.cdap.api.service.http.ServicePluginConfigurer;
import io.cdap.cdap.app.program.Program;
import io.cdap.cdap.app.runtime.ProgramOptions;
import io.cdap.cdap.common.conf.CConfiguration;
import io.cdap.cdap.common.conf.Constants;
import io.cdap.cdap.common.internal.remote.RemoteClientFactory;
import io.cdap.cdap.common.namespace.NamespaceQueryAdmin;
import io.cdap.cdap.common.utils.DirUtils;
import io.cdap.cdap.data2.dataset2.DatasetFramework;
import io.cdap.cdap.data2.metadata.writer.FieldLineageWriter;
import io.cdap.cdap.data2.metadata.writer.MetadataPublisher;
import io.cdap.cdap.internal.app.DefaultServicePluginConfigurer;
import io.cdap.cdap.internal.app.runtime.AbstractContext;
import io.cdap.cdap.internal.app.runtime.AppStateStoreProvider;
import io.cdap.cdap.internal.app.runtime.ProgramRunners;
import io.cdap.cdap.internal.app.runtime.artifact.PluginFinder;
import io.cdap.cdap.internal.app.runtime.plugin.PluginInstantiator;
import io.cdap.cdap.messaging.spi.MessagingService;
import io.cdap.cdap.proto.id.ArtifactId;
import io.cdap.cdap.proto.id.NamespaceId;
import java.io.Closeable;
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicInteger;
import javax.annotation.Nullable;
import org.apache.tephra.TransactionSystemClient;
import org.apache.twill.discovery.DiscoveryServiceClient;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Default implementation of HttpServiceContext which simply stores and retrieves the spec provided
 * when this class is instantiated
 */
public class BasicHttpServiceContext extends AbstractContext implements HttpServiceContext {

  private static final Logger LOG = LoggerFactory.getLogger(BasicHttpServiceContext.class);

  private final CConfiguration cConf;
  private final ArtifactId artifactId;
  private final HttpServiceHandlerSpecification spec;
  private final int instanceId;
  private final AtomicInteger instanceCount;
  private final ArtifactManager artifactManager;
  private final PluginFinder pluginFinder;
  private final Collection closeables;

  /**
   * Creates a BasicHttpServiceContext for the given HttpServiceHandlerSpecification.
   *
   * @param program program of the context.
   * @param programOptions program options for the program execution context
   * @param cConf the CDAP configuration
   * @param spec spec of the service handler of this context. If {@code null} is provided, this
   *     context is not associated with any service handler (e.g. for the http server itself).
   * @param instanceId instanceId of the component.
   * @param instanceCount total number of instances of the component.
   * @param metricsCollectionService metricsCollectionService to use for emitting metrics.
   * @param dsFramework dsFramework to use for getting datasets.
   * @param discoveryServiceClient discoveryServiceClient used to do service discovery.
   * @param txClient txClient to do transaction operations.
   * @param pluginInstantiator {@link PluginInstantiator}
   * @param secureStore the {@link SecureStore} for this context
   * @param secureStoreManager the {@link SecureStoreManager} for this context
   * @param messagingService the {@link MessagingService} for interacting with TMS
   * @param artifactManager the {@link ArtifactManager} for getting artifacts information
   * @param metadataReader the {@link MetadataReader} for reading metadata
   * @param metadataPublisher the {@link MetadataPublisher} for writing out metadata
   * @param namespaceQueryAdmin the {@link NamespaceQueryAdmin} for querying namespace
   *     information
   * @param pluginFinder the {@link PluginFinder} for plugin discovery
   * @param fieldLineageWriter the {@link FieldLineageWriter} for writing out field level
   *     lineage
   */
  public BasicHttpServiceContext(Program program, ProgramOptions programOptions,
      CConfiguration cConf,
      @Nullable HttpServiceHandlerSpecification spec,
      int instanceId, AtomicInteger instanceCount,
      MetricsCollectionService metricsCollectionService,
      DatasetFramework dsFramework, DiscoveryServiceClient discoveryServiceClient,
      TransactionSystemClient txClient, @Nullable PluginInstantiator pluginInstantiator,
      SecureStore secureStore, SecureStoreManager secureStoreManager,
      MessagingService messagingService,
      ArtifactManager artifactManager, MetadataReader metadataReader,
      MetadataPublisher metadataPublisher,
      NamespaceQueryAdmin namespaceQueryAdmin,
      PluginFinder pluginFinder,
      FieldLineageWriter fieldLineageWriter, RemoteClientFactory remoteClientFactory,
      AppStateStoreProvider appStateStoreProvider) {
    super(program, programOptions, cConf,
        spec == null ? Collections.emptySet() : spec.getDatasets(),
        dsFramework, txClient, false,
        metricsCollectionService, createMetricsTags(spec, instanceId),
        secureStore, secureStoreManager, messagingService, pluginInstantiator, metadataReader,
        metadataPublisher,
        namespaceQueryAdmin, fieldLineageWriter, remoteClientFactory, appStateStoreProvider);
    this.cConf = cConf;
    this.artifactId = ProgramRunners.getArtifactId(programOptions);
    this.spec = spec;
    this.instanceId = instanceId;
    this.instanceCount = instanceCount;
    this.artifactManager = artifactManager;
    this.pluginFinder = pluginFinder;
    this.closeables = new ArrayList<>();
  }

  public static Map createMetricsTags(
      @Nullable HttpServiceHandlerSpecification spec, int instanceId) {
    Map tags = new HashMap<>();
    tags.put(Constants.Metrics.Tag.INSTANCE_ID, String.valueOf(instanceId));
    if (spec != null) {
      tags.put(Constants.Metrics.Tag.HANDLER, spec.getName());
    }
    return tags;
  }

  /**
   * @return the {@link HttpServiceHandlerSpecification} for this context or {@code null} if there
   *     is no service handler associated with this context.
   */
  @Nullable
  @Override
  public HttpServiceHandlerSpecification getSpecification() {
    return spec;
  }

  @Override
  public int getInstanceCount() {
    return instanceCount.get();
  }

  @Override
  public int getInstanceId() {
    return instanceId;
  }

  @Override
  public PluginConfigurer createPluginConfigurer() {
    return createServicePluginConfigurer(getNamespace());
  }

  @Override
  public PluginConfigurer createPluginConfigurer(String namespace) {
    return createServicePluginConfigurer(namespace);
  }

  @Override
  public ServicePluginConfigurer createServicePluginConfigurer() {
    return createServicePluginConfigurer(getNamespace());
  }

  @Override
  public ServicePluginConfigurer createServicePluginConfigurer(String namespace) {
    File tmpDir = new File(cConf.get(Constants.CFG_LOCAL_DATA_DIR),
        cConf.get(Constants.AppFabric.TEMP_DIR)).getAbsoluteFile();
    try {
      File pluginsDir = Files.createTempDirectory(tmpDir.toPath(), "plugins").toFile();
      PluginInstantiator instantiator = new PluginInstantiator(cConf, getProgram().getClassLoader(),
          pluginsDir);
      // this closeables will be closed after each execution of a handler event
      closeables.add(() -> {
        try {
          instantiator.close();
        } finally {
          DirUtils.deleteDirectoryContents(pluginsDir, true);
        }
      });
      return new DefaultServicePluginConfigurer(artifactId, new NamespaceId(namespace),
          instantiator, pluginFinder,
          getProgram().getClassLoader());
    } catch (IOException e) {
      throw new RuntimeException(e);
    }
  }

  @Override
  public List listArtifacts() throws IOException, AccessException {
    return artifactManager.listArtifacts();
  }

  @Override
  public List listArtifacts(String namespace) throws IOException, AccessException {
    return artifactManager.listArtifacts(namespace);
  }

  @Override
  public CloseableClassLoader createClassLoader(ArtifactInfo artifactInfo,
      @Nullable ClassLoader parentClassLoader)
      throws IOException, AccessException {
    return artifactManager.createClassLoader(artifactInfo, parentClassLoader);
  }

  @Override
  public CloseableClassLoader createClassLoader(String namespace, ArtifactInfo artifactInfo,
      @Nullable ClassLoader parentClassLoader)
      throws IOException, AccessException {
    return artifactManager.createClassLoader(namespace, artifactInfo, parentClassLoader);
  }

  /**
   * Releases resources that were created for an endpoint call but are no longer needed for future
   * calls.
   */
  public void releaseCallResources() {
    for (Closeable closeable : closeables) {
      try {
        closeable.close();
      } catch (IOException e) {
        LOG.warn("Error while cleaning up service resources.", e);
      }
    }
    closeables.clear();
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy