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

io.ray.streaming.operator.chain.ChainedOperator Maven / Gradle / Ivy

package io.ray.streaming.operator.chain;

import com.google.common.base.Preconditions;
import io.ray.streaming.api.Language;
import io.ray.streaming.api.collector.Collector;
import io.ray.streaming.api.context.RuntimeContext;
import io.ray.streaming.api.function.Function;
import io.ray.streaming.api.function.impl.SourceFunction.SourceContext;
import io.ray.streaming.message.Record;
import io.ray.streaming.operator.OneInputOperator;
import io.ray.streaming.operator.Operator;
import io.ray.streaming.operator.OperatorType;
import io.ray.streaming.operator.SourceOperator;
import io.ray.streaming.operator.StreamOperator;
import io.ray.streaming.operator.TwoInputOperator;
import java.io.Serializable;
import java.lang.reflect.Proxy;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

/** Abstract base class for chained operators. */
public abstract class ChainedOperator extends StreamOperator {

  protected final List operators;
  protected final Operator headOperator;
  protected final Operator tailOperator;
  private final List> configs;

  public ChainedOperator(List operators, List> configs) {
    Preconditions.checkArgument(
        operators.size() >= 2, "Need at lease two operators to be chained together");
    operators.stream()
        .skip(1)
        .forEach(operator -> Preconditions.checkArgument(operator instanceof OneInputOperator));
    this.operators = operators;
    this.configs = configs;
    this.headOperator = operators.get(0);
    this.tailOperator = operators.get(operators.size() - 1);
  }

  @Override
  public void open(List collectorList, RuntimeContext runtimeContext) {
    // Dont' call super.open() as we `open` every operator separately.
    List succeedingCollectors =
        operators.stream()
            .skip(1)
            .map(operator -> new ForwardCollector((OneInputOperator) operator))
            .collect(Collectors.toList());
    for (int i = 0; i < operators.size() - 1; i++) {
      StreamOperator operator = operators.get(i);
      List forwardCollectors =
          Collections.singletonList(succeedingCollectors.get(i));
      operator.open(forwardCollectors, createRuntimeContext(runtimeContext, i));
    }
    // tail operator send data to downstream using provided collectors.
    tailOperator.open(collectorList, createRuntimeContext(runtimeContext, operators.size() - 1));
  }

  @Override
  public OperatorType getOpType() {
    return headOperator.getOpType();
  }

  @Override
  public Language getLanguage() {
    return headOperator.getLanguage();
  }

  @Override
  public String getName() {
    return operators.stream().map(Operator::getName).collect(Collectors.joining(" -> ", "[", "]"));
  }

  public List getOperators() {
    return operators;
  }

  public Operator getHeadOperator() {
    return headOperator;
  }

  public Operator getTailOperator() {
    return tailOperator;
  }

  @Override
  public Serializable saveCheckpoint() {
    Serializable[] checkpoints = new Serializable[operators.size()];
    for (int i = 0; i < operators.size(); ++i) {
      checkpoints[i] = operators.get(i).saveCheckpoint();
    }
    return checkpoints;
  }

  @Override
  public void loadCheckpoint(Serializable checkpointObject) {
    Serializable[] checkpoints = (Serializable[]) checkpointObject;
    for (int i = 0; i < operators.size(); ++i) {
      operators.get(i).loadCheckpoint(checkpoints[i]);
    }
  }

  private RuntimeContext createRuntimeContext(RuntimeContext runtimeContext, int index) {
    return (RuntimeContext)
        Proxy.newProxyInstance(
            runtimeContext.getClass().getClassLoader(),
            new Class[] {RuntimeContext.class},
            (proxy, method, methodArgs) -> {
              if (method.getName().equals("getConfig")) {
                return configs.get(index);
              } else {
                return method.invoke(runtimeContext, methodArgs);
              }
            });
  }

  public static ChainedOperator newChainedOperator(
      List operators, List> configs) {
    switch (operators.get(0).getOpType()) {
      case SOURCE:
        return new ChainedSourceOperator(operators, configs);
      case ONE_INPUT:
        return new ChainedOneInputOperator(operators, configs);
      case TWO_INPUT:
        return new ChainedTwoInputOperator(operators, configs);
      default:
        throw new IllegalArgumentException(
            "Unsupported operator type " + operators.get(0).getOpType());
    }
  }

  static class ChainedSourceOperator extends ChainedOperator implements SourceOperator {

    private final SourceOperator sourceOperator;

    @SuppressWarnings("unchecked")
    ChainedSourceOperator(List operators, List> configs) {
      super(operators, configs);
      sourceOperator = (SourceOperator) headOperator;
    }

    @Override
    public void fetch() {
      sourceOperator.fetch();
    }

    @Override
    public SourceContext getSourceContext() {
      return sourceOperator.getSourceContext();
    }
  }

  static class ChainedOneInputOperator extends ChainedOperator implements OneInputOperator {

    private final OneInputOperator inputOperator;

    @SuppressWarnings("unchecked")
    ChainedOneInputOperator(List operators, List> configs) {
      super(operators, configs);
      inputOperator = (OneInputOperator) headOperator;
    }

    @Override
    public void processElement(Record record) throws Exception {
      inputOperator.processElement(record);
    }
  }

  static class ChainedTwoInputOperator extends ChainedOperator
      implements TwoInputOperator {

    private final TwoInputOperator inputOperator;

    @SuppressWarnings("unchecked")
    ChainedTwoInputOperator(List operators, List> configs) {
      super(operators, configs);
      inputOperator = (TwoInputOperator) headOperator;
    }

    @Override
    public void processElement(Record record1, Record record2) {
      inputOperator.processElement(record1, record2);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy