The Broadcast State Pattern

In this section you will learn about how to use broadcast state in practise. Please refer to Stateful Stream Processing to learn about the concepts behind stateful stream processing.

Provided APIs

To show the provided APIs, we will start with an example before presenting their full functionality. As our running example, we will use the case where we have a stream of objects of different colors and shapes and we want to find pairs of objects of the same color that follow a certain pattern, e.g. a rectangle followed by a triangle. We assume that the set of interesting patterns evolves over time.

In this example, the first stream will contain elements of type Item with a Color and a Shape property. The other stream will contain the Rules.

Starting from the stream of Items, we just need to key it by Color, as we want pairs of the same color. This will make sure that elements of the same color end up on the same physical machine.

  1. // key the items by color
  2. KeyedStream<Item, Color> colorPartitionedStream = itemStream
  3. .keyBy(new KeySelector<Item, Color>(){...});

Moving on to the Rules, the stream containing them should be broadcasted to all downstream tasks, and these tasks should store them locally so that they can evaluate them against all incoming Items. The snippet below will i) broadcast the stream of rules and ii) using the provided MapStateDescriptor, it will create the broadcast state where the rules will be stored.

  1. // a map descriptor to store the name of the rule (string) and the rule itself.
  2. MapStateDescriptor<String, Rule> ruleStateDescriptor = new MapStateDescriptor<>(
  3. "RulesBroadcastState",
  4. BasicTypeInfo.STRING_TYPE_INFO,
  5. TypeInformation.of(new TypeHint<Rule>() {}));
  6. // broadcast the rules and create the broadcast state
  7. BroadcastStream<Rule> ruleBroadcastStream = ruleStream
  8. .broadcast(ruleStateDescriptor);

Finally, in order to evaluate the Rules against the incoming elements from the Item stream, we need to:

  1. connect the two streams, and
  2. specify our match detecting logic.

Connecting a stream (keyed or non-keyed) with a BroadcastStream can be done by calling connect() on the non-broadcasted stream, with the BroadcastStream as an argument. This will return a BroadcastConnectedStream, on which we can call process() with a special type of CoProcessFunction. The function will contain our matching logic. The exact type of the function depends on the type of the non-broadcasted stream:

  • if that is keyed, then the function is a KeyedBroadcastProcessFunction.
  • if it is non-keyed, the function is a BroadcastProcessFunction.

Given that our non-broadcasted stream is keyed, the following snippet includes the above calls:

Attention: The connect should be called on the non-broadcasted stream, with the BroadcastStream as an argument.

  1. DataStream<String> output = colorPartitionedStream
  2. .connect(ruleBroadcastStream)
  3. .process(
  4. // type arguments in our KeyedBroadcastProcessFunction represent:
  5. // 1. the key of the keyed stream
  6. // 2. the type of elements in the non-broadcast side
  7. // 3. the type of elements in the broadcast side
  8. // 4. the type of the result, here a string
  9. new KeyedBroadcastProcessFunction<Color, Item, Rule, String>() {
  10. // my matching logic
  11. }
  12. );

BroadcastProcessFunction and KeyedBroadcastProcessFunction

As in the case of a CoProcessFunction, these functions have two process methods to implement; the processBroadcastElement() which is responsible for processing incoming elements in the broadcasted stream and the processElement() which is used for the non-broadcasted one. The full signatures of the methods are presented below:

  1. public abstract class BroadcastProcessFunction<IN1, IN2, OUT> extends BaseBroadcastProcessFunction {
  2. public abstract void processElement(IN1 value, ReadOnlyContext ctx, Collector<OUT> out) throws Exception;
  3. public abstract void processBroadcastElement(IN2 value, Context ctx, Collector<OUT> out) throws Exception;
  4. }
  1. public abstract class KeyedBroadcastProcessFunction<KS, IN1, IN2, OUT> {
  2. public abstract void processElement(IN1 value, ReadOnlyContext ctx, Collector<OUT> out) throws Exception;
  3. public abstract void processBroadcastElement(IN2 value, Context ctx, Collector<OUT> out) throws Exception;
  4. public void onTimer(long timestamp, OnTimerContext ctx, Collector<OUT> out) throws Exception;
  5. }

The first thing to notice is that both functions require the implementation of the processBroadcastElement() method for processing elements in the broadcast side and the processElement() for elements in the non-broadcasted side.

The two methods differ in the context they are provided. The non-broadcast side has a ReadOnlyContext, while the broadcasted side has a Context.

Both of these contexts (ctx in the following enumeration):

  1. give access to the broadcast state: ctx.getBroadcastState(MapStateDescriptor<K, V> stateDescriptor)
  2. allow to query the timestamp of the element: ctx.timestamp(),
  3. get the current watermark: ctx.currentWatermark()
  4. get the current processing time: ctx.currentProcessingTime(), and
  5. emit elements to side-outputs: ctx.output(OutputTag<X> outputTag, X value).

The stateDescriptor in the getBroadcastState() should be identical to the one in the .broadcast(ruleStateDescriptor) above.

The difference lies in the type of access each one gives to the broadcast state. The broadcasted side has read-write access to it, while the non-broadcast side has read-only access (thus the names). The reason for this is that in Flink there is no cross-task communication. So, to guarantee that the contents in the Broadcast State are the same across all parallel instances of our operator, we give read-write access only to the broadcast side, which sees the same elements across all tasks, and we require the computation on each incoming element on that side to be identical across all tasks. Ignoring this rule would break the consistency guarantees of the state, leading to inconsistent and often difficult to debug results.

Attention: The logic implemented in `processBroadcastElement()` must have the same deterministic behavior across all parallel instances!

Finally, due to the fact that the KeyedBroadcastProcessFunction is operating on a keyed stream, it exposes some functionality which is not available to the BroadcastProcessFunction. That is:

  1. the ReadOnlyContext in the processElement() method gives access to Flink’s underlying timer service, which allows to register event and/or processing time timers. When a timer fires, the onTimer() (shown above) is invoked with an OnTimerContext which exposes the same functionality as the ReadOnlyContext plus
    • the ability to ask if the timer that fired was an event or processing time one and
    • to query the key associated with the timer.
  2. the Context in the processBroadcastElement() method contains the method applyToKeyedState(StateDescriptor<S, VS> stateDescriptor, KeyedStateFunction<KS, S> function). This allows to register a KeyedStateFunction to be applied to all states of all keys associated with the provided stateDescriptor.

Attention: Registering timers is only possible at `processElement()` of the `KeyedBroadcastProcessFunction` and only there. It is not possible in the `processBroadcastElement()` method, as there is no key associated to the broadcasted elements.

Coming back to our original example, our KeyedBroadcastProcessFunction could look like the following:

  1. new KeyedBroadcastProcessFunction<Color, Item, Rule, String>() {
  2. // store partial matches, i.e. first elements of the pair waiting for their second element
  3. // we keep a list as we may have many first elements waiting
  4. private final MapStateDescriptor<String, List<Item>> mapStateDesc =
  5. new MapStateDescriptor<>(
  6. "items",
  7. BasicTypeInfo.STRING_TYPE_INFO,
  8. new ListTypeInfo<>(Item.class));
  9. // identical to our ruleStateDescriptor above
  10. private final MapStateDescriptor<String, Rule> ruleStateDescriptor =
  11. new MapStateDescriptor<>(
  12. "RulesBroadcastState",
  13. BasicTypeInfo.STRING_TYPE_INFO,
  14. TypeInformation.of(new TypeHint<Rule>() {}));
  15. @Override
  16. public void processBroadcastElement(Rule value,
  17. Context ctx,
  18. Collector<String> out) throws Exception {
  19. ctx.getBroadcastState(ruleStateDescriptor).put(value.name, value);
  20. }
  21. @Override
  22. public void processElement(Item value,
  23. ReadOnlyContext ctx,
  24. Collector<String> out) throws Exception {
  25. final MapState<String, List<Item>> state = getRuntimeContext().getMapState(mapStateDesc);
  26. final Shape shape = value.getShape();
  27. for (Map.Entry<String, Rule> entry :
  28. ctx.getBroadcastState(ruleStateDescriptor).immutableEntries()) {
  29. final String ruleName = entry.getKey();
  30. final Rule rule = entry.getValue();
  31. List<Item> stored = state.get(ruleName);
  32. if (stored == null) {
  33. stored = new ArrayList<>();
  34. }
  35. if (shape == rule.second && !stored.isEmpty()) {
  36. for (Item i : stored) {
  37. out.collect("MATCH: " + i + " - " + value);
  38. }
  39. stored.clear();
  40. }
  41. // there is no else{} to cover if rule.first == rule.second
  42. if (shape.equals(rule.first)) {
  43. stored.add(value);
  44. }
  45. if (stored.isEmpty()) {
  46. state.remove(ruleName);
  47. } else {
  48. state.put(ruleName, stored);
  49. }
  50. }
  51. }
  52. }

Important Considerations

After describing the offered APIs, this section focuses on the important things to keep in mind when using broadcast state. These are:

  • There is no cross-task communication: As stated earlier, this is the reason why only the broadcast side of a (Keyed)-BroadcastProcessFunction can modify the contents of the broadcast state. In addition, the user has to make sure that all tasks modify the contents of the broadcast state in the same way for each incoming element. Otherwise, different tasks might have different contents, leading to inconsistent results.

  • Order of events in Broadcast State may differ across tasks: Although broadcasting the elements of a stream guarantees that all elements will (eventually) go to all downstream tasks, elements may arrive in a different order to each task. So the state updates for each incoming element MUST NOT depend on the ordering of the incoming events.

  • All tasks checkpoint their broadcast state: Although all tasks have the same elements in their broadcast state when a checkpoint takes place (checkpoint barriers do not overpass elements), all tasks checkpoint their broadcast state, and not just one of them. This is a design decision to avoid having all tasks read from the same file during a restore (thus avoiding hotspots), although it comes at the expense of increasing the size of the checkpointed state by a factor of p (= parallelism). Flink guarantees that upon restoring/rescaling there will be no duplicates and no missing data. In case of recovery with the same or smaller parallelism, each task reads its checkpointed state. Upon scaling up, each task reads its own state, and the remaining tasks (p_new-p_old) read checkpoints of previous tasks in a round-robin manner.

  • No RocksDB state backend: Broadcast state is kept in-memory at runtime and memory provisioning should be done accordingly. This holds for all operator states.