This Week in Fluvio #15
Welcome to This Week in Fluvio, our weekly newsletter for development updates to Fluvio open source. Fluvio is a distributed, programmable streaming platform written in Rust.
BANNER
New Release - Fluvio v0.9.13
This is a big release
Apple M1 support
This is the first release with official Apple M1 support. If you have an Apple M1 machine and you'd like to try out Fluvio, please read our Getting Started page for MacOS.
SmartModules
We previewed this feature a couple weeks ago and now it's here!
All SmartModule types (filter, map, array-map, filter-map, aggregate, joins) can saved into your Fluvio cluster and you can use them just by referring to them by name.
Example creating of SmartModule:
$ fluvio smartmodule create my-filter --wasm-file ./path/to/my-filter.wasm
Example usage:
$ fluvio consume my-topic --filter <name>
$ fluvio consume my-topic --map <name>
$ fluvio consume my-topic --array-map <name>
$ fluvio consume my-topic --filter-map <name>
$ fluvio consume my-topic --aggregate <name>
$ fluvio consume my-topic --join <name>
You can still use SmartModules the original way, by providing a path to your wasm file. But if you're using the SmartModule a lot, we think persistent SmartModules will be more convenient to use.
SmartConnectors
This feature was teased last week, but now it is ready to be tried out.
Check out the new Connector Developer guide for more information about how to create your own connectors.
FilterMap SmartModule
The FilterMap SmartModule enables you to do filtering and reshaping your data at the same time.
Example FilterMap code
In this example, we take in integers.
If those integers are positive, we want to divide it by two and return. Filter out inputs if they are odd.
use fluvio_smartmodule::{smartmodule, Record, RecordData, Result};
#[smartmodule(filter_map)]
pub fn filter_map(record: &Record) -> Result<Option<(Option<RecordData>, RecordData)>> {
    let key = record.key.clone();
    let string = String::from_utf8_lossy(record.value.as_ref()).to_string();
    let int: i32 = string.parse()?;
    if int % 2 == 0 {
        let output = int / 2;
        Ok(Some((key.clone(), RecordData::from(output.to_string()))))
    } else {
        Ok(None)
    }
}
Example input
$ fluvio produce filter-map-topic
> 19
Ok!
> 30
Ok!
> 29
Ok!
> -60
Ok!
> 23
Ok!
> 90
Ok!
> 17
Ok!
> ^C
Example output
$ fluvio consume filter-map-topic --filter-map divide-even-numbers
Consuming records from the end of topic 'filter-map-topic'. This will wait for new records
15
-30
45
For a deeper dive into FilterMap, check out our blog post which covers a use-case.
Join SmartModule
The Join SmartModule uses the stream you are consuming and the value at the end of another topic and allows you to return a new value
Example Join code
In this example, we have 2 topics, left-topic and right-topic, and our example Join SmartModule.
use fluvio_smartmodule::{smartmodule, Record, RecordData, Result};
#[smartmodule(join)]
pub fn join(left_record: &Record, right_record: &Record) -> Result<(Option<RecordData>, RecordData)> {
    let left_value: i32 = std::str::from_utf8(left_record.value.as_ref())?.parse()?;
    let right_value: i32 = std::str::from_utf8(right_record.value.as_ref())?.parse()?;
    let value = left_value + right_value;
    Ok((None, value.to_string().into()))
}
Example input for topic right-topic
$ fluvio produce right-topic
> 2
Ok!
Now, when new records come in to left-topic, we will add the new record to the newest record at right-topic.
$ fluvio produce left-topic
> 7
Ok!
> 11
Ok!
> 19
Ok!
Example consume output
$ fluvio consume left-topic --join example-join --join-topic right-topic
Consuming records from the end of topic 'left-topic'. This will wait for new records
9
13
21
And if we change the top value on right-topic
$ fluvio produce right-topic
> -5
Ok!
Then produce new values to left-topic
$ fluvio produce left-topic
> 3
Ok!
> 10
Ok!
> 27
Ok!
The resulting consume output will reflect the new values to left-topic adding itself to the new negative value at right-topic
$ fluvio consume left-topic --join example-join --join-topic right-topic
Consuming records from the end of topic 'left-topic'. This will wait for new records
9
13
21
-2
5
22
Fullscreen Consumer table
This is the first version of an interactive table display for the Consumer. It expects the same json object input as --output=table but the output is a full screen scrollable table. The columns are alphabetized, and the first column serves as a primary key for updating the row. This offers the possibility of viewing live updating data
First you need to create a tableformat to define how you want your table to be displayed. This would be highly dependent on the shape of your data. For example purposes, we will be displaying event-sourced data. This is what our example data looks like:
{"request_id":"123", "requester_name": "Alice", "state":"running", "run_time_minutes":"3"}
{"request_id":"456", "requester_name": "Alice", "state":"waiting", "run_time_minutes":"9"}
{"request_id":"789", "requester_name": "Bob", "state":"done", "run_time_minutes":"10"}
An example tableformat.
Here we only want to display the latest state of a request. We declare the request_id key as primaryKey, which means that any new events matching an existing request_id will update the row.
# tableformat-config.yaml
name: "current-requests"
input_format: "JSON"
columns:
    - headerLabel: "ID"
      keyPath: "request_id"
      primaryKey: true
    - headerLabel: "Runtime"
      keyPath: "run_time_minutes"
    - headerLabel: "State"
      keyPath: "state"
Create the tableformat
$ fluvio tableformat create --config tableformat-config.yaml
Consuming from the topic using the full-table output, and our tableformat
$ fluvio consume request-events --output full-table --tableformat current-requests
Output:
┌('c' to clear table | 'q' or ESC to exit) | Items: 3──┐
│ID                Runtime           State             │