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

org.apache.kafka.streams.kstream.TimeWindows Maven / Gradle / Ivy

There is a newer version: 3.7.1
Show newest version
/*
 * 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.kafka.streams.kstream;

import org.apache.kafka.streams.kstream.internals.TimeWindow;
import org.apache.kafka.streams.processor.TimestampExtractor;

import java.time.Duration;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Objects;

import static java.time.Duration.ofMillis;
import static org.apache.kafka.streams.internals.ApiUtils.prepareMillisCheckFailMsgPrefix;
import static org.apache.kafka.streams.internals.ApiUtils.validateMillisecondDuration;

/**
 * The fixed-size time-based window specifications used for aggregations.
 * 

* The semantics of time-based aggregation windows are: Every T1 (advance) milliseconds, compute the aggregate total for * T2 (size) milliseconds. *

    *
  • If {@code advance < size} a hopping windows is defined:
    * it discretize a stream into overlapping windows, which implies that a record maybe contained in one and or * more "adjacent" windows.
  • *
  • If {@code advance == size} a tumbling window is defined:
    * it discretize a stream into non-overlapping windows, which implies that a record is only ever contained in * one and only one tumbling window.
  • *
* Thus, the specified {@link TimeWindow}s are aligned to the epoch. * Aligned to the epoch means, that the first window starts at timestamp zero. * For example, hopping windows with size of 5000ms and advance of 3000ms, have window boundaries * [0;5000),[3000;8000),... and not [1000;6000),[4000;9000),... or even something "random" like [1452;6452),[4452;9452),... *

* For time semantics, see {@link TimestampExtractor}. * * @see SessionWindows * @see UnlimitedWindows * @see JoinWindows * @see KGroupedStream#windowedBy(Windows) * @see TimestampExtractor */ public final class TimeWindows extends Windows { /** The size of the windows in milliseconds. */ @SuppressWarnings("WeakerAccess") public final long sizeMs; /** * The size of the window's advance interval in milliseconds, i.e., by how much a window moves forward relative to * the previous one. */ @SuppressWarnings("WeakerAccess") public final long advanceMs; private final long graceMs; // flag to check if the grace is already set via ofSizeAndGrace or ofSizeWithNoGrace private final boolean hasSetGrace; private TimeWindows(final long sizeMs, final long advanceMs, final long graceMs, final boolean hasSetGrace) { this.sizeMs = sizeMs; this.advanceMs = advanceMs; this.graceMs = graceMs; this.hasSetGrace = hasSetGrace; if (sizeMs <= 0) { throw new IllegalArgumentException("Window size (sizeMs) must be larger than zero."); } if (advanceMs <= 0 || advanceMs > sizeMs) { throw new IllegalArgumentException(String.format("Window advancement interval should be more than zero " + "and less than window duration which is %d ms, but given advancement interval is: %d ms", sizeMs, advanceMs)); } if (graceMs < 0) { throw new IllegalArgumentException("Grace period must not be negative."); } } /** * Return a window definition with the given window size, and with the advance interval being equal to the window * size. * The time interval represented by the N-th window is: {@code [N * size, N * size + size)}. *

* This provides the semantics of tumbling windows, which are fixed-sized, gap-less, non-overlapping windows. * Tumbling windows are a special case of hopping windows with {@code advance == size}. *

* CAUTION: Using this method implicitly sets the grace period to zero, which means that any out-of-order * records arriving after the window ends are considered late and will be dropped. * * @param size The size of the window * @return a new window definition with default no grace period. Note that this means out-of-order records arriving after the window end will be dropped * @throws IllegalArgumentException if the specified window size is zero or negative or can't be represented as {@code long milliseconds} */ public static TimeWindows ofSizeWithNoGrace(final Duration size) throws IllegalArgumentException { return ofSizeAndGrace(size, ofMillis(NO_GRACE_PERIOD)); } /** * Return a window definition with the given window size, and with the advance interval being equal to the window * size. * The time interval represented by the N-th window is: {@code [N * size, N * size + size)}. *

* This provides the semantics of tumbling windows, which are fixed-sized, gap-less, non-overlapping windows. * Tumbling windows are a special case of hopping windows with {@code advance == size}. *

* Using this method explicitly sets the grace period to the duration specified by {@code afterWindowEnd}, which * means that only out-of-order records arriving more than the grace period after the window end will be dropped. * The window close, after which any incoming records are considered late and will be rejected, is defined as * {@code windowEnd + afterWindowEnd} * * @param size The size of the window. Must be larger than zero * @param afterWindowEnd The grace period to admit out-of-order events to a window. Must be non-negative. * @return a TimeWindows object with the specified size and the specified grace period * @throws IllegalArgumentException if {@code afterWindowEnd} is negative or can't be represented as {@code long milliseconds} */ public static TimeWindows ofSizeAndGrace(final Duration size, final Duration afterWindowEnd) throws IllegalArgumentException { final String sizeMsgPrefix = prepareMillisCheckFailMsgPrefix(size, "size"); final long sizeMs = validateMillisecondDuration(size, sizeMsgPrefix); final String afterWindowEndMsgPrefix = prepareMillisCheckFailMsgPrefix(afterWindowEnd, "afterWindowEnd"); final long afterWindowEndMs = validateMillisecondDuration(afterWindowEnd, afterWindowEndMsgPrefix); return new TimeWindows(sizeMs, sizeMs, afterWindowEndMs, true); } /** * Return a window definition with the given window size, and with the advance interval being equal to the window * size. * The time interval represented by the N-th window is: {@code [N * size, N * size + size)}. *

* This provides the semantics of tumbling windows, which are fixed-sized, gap-less, non-overlapping windows. * Tumbling windows are a special case of hopping windows with {@code advance == size}. * * @param size The size of the window * @return a new window definition without specifying the grace period (default to 24 hours minus window {@code size}) * @throws IllegalArgumentException if the specified window size is zero or negative or can't be represented as {@code long milliseconds} * @deprecated since 3.0. Use {@link #ofSizeWithNoGrace(Duration)} } instead */ @Deprecated public static TimeWindows of(final Duration size) throws IllegalArgumentException { final String msgPrefix = prepareMillisCheckFailMsgPrefix(size, "size"); final long sizeMs = validateMillisecondDuration(size, msgPrefix); return new TimeWindows(sizeMs, sizeMs, Math.max(DEPRECATED_DEFAULT_24_HR_GRACE_PERIOD - sizeMs, 0), false); } /** * Return a window definition with the original size, but advance ("hop") the window by the given interval, which * specifies by how much a window moves forward relative to the previous one. * The time interval represented by the N-th window is: {@code [N * advance, N * advance + size)}. *

* This provides the semantics of hopping windows, which are fixed-sized, overlapping windows. * * @param advance The advance interval ("hop") of the window, with the requirement that {@code 0 < advance.toMillis() <= sizeMs}. * @return a new window definition with default maintain duration of 1 day * @throws IllegalArgumentException if the advance interval is negative, zero, or larger than the window size */ public TimeWindows advanceBy(final Duration advance) { final String msgPrefix = prepareMillisCheckFailMsgPrefix(advance, "advance"); final long advanceMs = validateMillisecondDuration(advance, msgPrefix); return new TimeWindows(sizeMs, advanceMs, graceMs, false); } @Override public Map windowsFor(final long timestamp) { long windowStart = (Math.max(0, timestamp - sizeMs + advanceMs) / advanceMs) * advanceMs; final Map windows = new LinkedHashMap<>(); while (windowStart <= timestamp) { final TimeWindow window = new TimeWindow(windowStart, windowStart + sizeMs); windows.put(windowStart, window); windowStart += advanceMs; } return windows; } @Override public long size() { return sizeMs; } /** * Reject out-of-order events that arrive more than {@code millisAfterWindowEnd} * after the end of its window. *

* Delay is defined as (stream_time - record_timestamp). * * @param afterWindowEnd The grace period to admit out-of-order events to a window. * @return this updated builder * @throws IllegalArgumentException if {@code afterWindowEnd} is negative or can't be represented as {@code long milliseconds} * @throws IllegalStateException if {@link #grace(Duration)} is called after {@link #ofSizeAndGrace(Duration, Duration)} or {@link #ofSizeWithNoGrace(Duration)} * @deprecated since 3.0. Use {@link #ofSizeAndGrace(Duration, Duration)} instead */ @Deprecated public TimeWindows grace(final Duration afterWindowEnd) throws IllegalArgumentException { if (this.hasSetGrace) { throw new IllegalStateException( "Cannot call grace() after setting grace value via ofSizeAndGrace or ofSizeWithNoGrace."); } final String msgPrefix = prepareMillisCheckFailMsgPrefix(afterWindowEnd, "afterWindowEnd"); final long afterWindowEndMs = validateMillisecondDuration(afterWindowEnd, msgPrefix); return new TimeWindows(sizeMs, advanceMs, afterWindowEndMs, false); } @Override public long gracePeriodMs() { return graceMs; } @Override public boolean equals(final Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } final TimeWindows that = (TimeWindows) o; return sizeMs == that.sizeMs && advanceMs == that.advanceMs && graceMs == that.graceMs; } @Override public int hashCode() { return Objects.hash(sizeMs, advanceMs, graceMs); } @Override public String toString() { return "TimeWindows{" + ", sizeMs=" + sizeMs + ", advanceMs=" + advanceMs + ", graceMs=" + graceMs + '}'; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy