org.apache.flink.streaming.runtime.tasks.ChainingOutput 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.flink.streaming.runtime.tasks;
import org.apache.flink.metrics.Counter;
import org.apache.flink.metrics.Gauge;
import org.apache.flink.metrics.SimpleCounter;
import org.apache.flink.metrics.groups.OperatorMetricGroup;
import org.apache.flink.streaming.api.operators.Input;
import org.apache.flink.streaming.api.watermark.Watermark;
import org.apache.flink.streaming.runtime.io.RecordProcessorUtils;
import org.apache.flink.streaming.runtime.metrics.WatermarkGauge;
import org.apache.flink.streaming.runtime.streamrecord.LatencyMarker;
import org.apache.flink.streaming.runtime.streamrecord.RecordAttributes;
import org.apache.flink.streaming.runtime.streamrecord.StreamRecord;
import org.apache.flink.streaming.runtime.watermarkstatus.WatermarkStatus;
import org.apache.flink.util.OutputTag;
import org.apache.flink.util.function.ThrowingConsumer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.annotation.Nullable;
class ChainingOutput
implements WatermarkGaugeExposingOutput>,
OutputWithChainingCheck> {
private static final Logger LOG = LoggerFactory.getLogger(ChainingOutput.class);
protected final Input input;
protected final Counter numRecordsOut;
protected final Counter numRecordsIn;
protected final WatermarkGauge watermarkGauge = new WatermarkGauge();
@Nullable protected final OutputTag outputTag;
protected WatermarkStatus announcedStatus = WatermarkStatus.ACTIVE;
protected final ThrowingConsumer, Exception> recordProcessor;
public ChainingOutput(
Input input,
@Nullable Counter prevNumRecordsOut,
OperatorMetricGroup curOperatorMetricGroup,
@Nullable OutputTag outputTag) {
this.input = input;
if (prevNumRecordsOut != null) {
this.numRecordsOut = prevNumRecordsOut;
} else {
// Uses a dummy counter here to avoid checking the existence of numRecordsOut on the
// per-record path.
this.numRecordsOut = new SimpleCounter();
}
this.numRecordsIn = curOperatorMetricGroup.getIOMetricGroup().getNumRecordsInCounter();
this.outputTag = outputTag;
this.recordProcessor = RecordProcessorUtils.getRecordProcessor(input);
}
@Override
public void collect(StreamRecord record) {
if (this.outputTag != null) {
// we are not responsible for emitting to the main output.
return;
}
pushToOperator(record);
}
@Override
public void collect(OutputTag outputTag, StreamRecord record) {
if (OutputTag.isResponsibleFor(this.outputTag, outputTag)) {
pushToOperator(record);
}
}
@Override
public boolean collectAndCheckIfChained(StreamRecord record) {
collect(record);
return false;
}
@Override
public boolean collectAndCheckIfChained(OutputTag outputTag, StreamRecord record) {
collect(outputTag, record);
return false;
}
protected void pushToOperator(StreamRecord record) {
try {
// we know that the given outputTag matches our OutputTag so the record
// must be of the type that our operator expects.
@SuppressWarnings("unchecked")
StreamRecord castRecord = (StreamRecord) record;
numRecordsOut.inc();
numRecordsIn.inc();
recordProcessor.accept(castRecord);
} catch (Exception e) {
throw new ExceptionInChainedOperatorException(e);
}
}
@Override
public void emitWatermark(Watermark mark) {
if (announcedStatus.isIdle()) {
return;
}
try {
watermarkGauge.setCurrentWatermark(mark.getTimestamp());
input.processWatermark(mark);
} catch (Exception e) {
throw new ExceptionInChainedOperatorException(e);
}
}
@Override
public void emitLatencyMarker(LatencyMarker latencyMarker) {
try {
input.processLatencyMarker(latencyMarker);
} catch (Exception e) {
throw new ExceptionInChainedOperatorException(e);
}
}
@Override
public void close() {
// nothing is owned by ChainingOutput and should be closed, see FLINK-20888
}
@Override
public Gauge getWatermarkGauge() {
return watermarkGauge;
}
@Override
public void emitWatermarkStatus(WatermarkStatus watermarkStatus) {
if (!announcedStatus.equals(watermarkStatus)) {
announcedStatus = watermarkStatus;
try {
input.processWatermarkStatus(watermarkStatus);
} catch (Exception e) {
throw new ExceptionInChainedOperatorException(e);
}
}
}
@Override
public void emitRecordAttributes(RecordAttributes recordAttributes) {
try {
input.processRecordAttributes(recordAttributes);
} catch (Exception e) {
throw new ExceptionInChainedOperatorException(e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy