Skip to content

Commit

Permalink
Add documentation for column variants (#380)
Browse files Browse the repository at this point in the history
* Checkpoint

* Checkpoint

* Checkpoint

* Typo

* Fix awkward sentence

* Rename Time (UTC)

* Whitespace

* Grammar

* Wording

* CHeckpoint

* Add engine docs

* Edits

* Add global varaints example code

* Fix glossary TOC

* Fix glossary TOC again

* Wording

* Add ASCII diagrams

* Address comments

* Typos

* Remove runtime section
  • Loading branch information
mslukebo authored Oct 30, 2024
1 parent e4e94fa commit 78158c2
Show file tree
Hide file tree
Showing 4 changed files with 360 additions and 1 deletion.
13 changes: 12 additions & 1 deletion documentation/Glossary.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
* [Column](#column)
* [ColumnConfiguration](#columnconfiguration)
* [ColumnRole](#columnrole)
* [Column Variant](#column-variant)
* [CompositeDataCooker](#compositedatacooker)
* [CustomDataProcessor](#customdataprocessor)
* [DataCooker](#datacooker)
Expand Down Expand Up @@ -53,6 +54,16 @@ For example, a Column of `Timestamp` values relative to the start of a [DataSour

ColumnRoles are specified as part of a [Table Configuration](#tableconfiguration).

## Column Variant

An alternative [projection](#projection) for a [column](#column). While table authors are free to define column variants however they wish, variants are often used to provide alternate views of the same underlying data. For example, a "Process" column may define
* one `IProjection<int, string>` variant that projects to the process name
* one `IProjection<int, uint>` variant that projects to the process ID

> ⚠️ Note: column variants **cannot** be assigned as a [column role](#columnrole) or highlight entries within a [column configuration](#columnconfiguration); only the "base" projection associated with the column may be used for these purposes.
For more information on the different types of column variants, refer to the [Adding Column Variants](./Using-the-SDK/Advanced/Adding-Column-Variants.md) documentation.

## CompositeDataCooker

A [DataCooker](#datacooker) that receives input solely from other DataCookers.
Expand Down Expand Up @@ -109,7 +120,7 @@ If the "State" column above is a [Pivot Column](#pivot-column), the 5 rows would
| | 10023 | 60,998 |
| | 10025 | 94,600 |

NOTE: the SDK has no understanding of Pivot Tables. Tables created by a plugin are purely "flat" tables - i.e. tables similar the first one above. It is up to programs like [Windows Performance Analyzer](#windows-performance-analyzer) to use pivot information in a [Table Configuration](#tableconfiguration) to present a plugin's Table as a Pivot Table.
> ⚠️ Note: the SDK has no understanding of Pivot Tables. Tables created by a plugin are purely "flat" tables - i.e. tables similar the first one above. It is up to programs like [Windows Performance Analyzer](#windows-performance-analyzer) to use pivot information in a [Table Configuration](#tableconfiguration) to present a plugin's Table as a Pivot Table.
See also: [Wikipedia Pivot Tables](https://en.wikipedia.org/wiki/Pivot_table).

Expand Down
335 changes: 335 additions & 0 deletions documentation/Using-the-SDK/Advanced/Adding-Column-Variants.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,335 @@
# Abstract

This document outlines how to configure a [column](../../Glossary.md#column) with [column variants](../../Glossary.md#column-variant). Additionally, this document provides guidelines on when to use each type of column variant and when you should consider making create separate columns instead of column variants.

# Motivation

There are often instances where a single piece of data can be expressed in multiple ways. Prior to SDK version `1.3`, table authors were forced to add one column for each way the data can be displayed.

For example, suppose a table has a "Process" column whose data identifies which process is associated with each row. This data can be expressed in multiple ways:
* The process name
* The process identifier (PID)
* The process name + PID

Without column variants, the table author would have to add three independent columns to a table to expose these three representations of the same data. With column variants, however, the table author can add one "Process" column with three variants - one for each representation.

# When to Use Column Variants

Columns variants should be used when you wish to expose the data within a column in multiple ways. Some examples of this are:
* A "Process" column that has "Process Name" and "Process ID" variants, where each variant identifies the process associated with each row
* A "Time" column that has "Timestamp" and "DateTime" variants, where each variant identifies the point in time each row's data was recorded

Different column variants within the same column have no restrictions on the data type `T` that the variants expose. As such, table authors should take care to ensure column variants make sense to be grouped together under one column. There are two restrictions on column variants that help guide whether to expose projections as variants or new columns:

1. Column variants cannot supply a new [column configuration](../../Glossary.md#columnconfiguration) for the column that the variant is for. As such, column variants should only be used when the projected data makes sense given the base column's configuration. If a user would be confused how a column variant's data relates to its base column, you should consider exposing the variant as a separate dedicated column.

Note that in some cases it may make sense to edit a column configuration's name and description to accommodate new variants; for example, a column named "Process Name" could be renamed to "Process" and be expose variants for process name, PID, and process name + PID.

2. Column variants cannot be used as a [column role](../../Glossary.md#columnrole) or highlight entries within a [column configuration](../../Glossary.md#columnconfiguration). When column roles or highlight entries are defined, they are always associated with the column's base projection.

# How to Add Column Variants

Column variants are added to a column by adding the base column to an `ITableBuilder` with the `AddColumnWithVariants` methods introduced in SDK version `1.3`. These methods take a callback `Func<RootColumnBuilder, ColumnBuilder>` whose return value defines all the variants to associate with the added column. The supplied `Func` will be invoked with a `RootColumnBuilder` supplied by the SDK runtime and must return the final `ColumnBuilder` configured with variants.

Note that unlike the `ITableBuilder`, column builders are purely functional components: every column builder method returns a **new instance** of a column builder. For example, consider the code snippet below:

```cs
tableBuilder
.AddColumnWithVariants(baseConfig, projectionA, builder =>
{
// These modes are ignored because the return value from
// WithMode(modeB, projectionB) is being discarded
builder
.WithModes("Mode A")
.WithMode(modeB, projectionB);

// Will not contain the modes added above because the returned value is building
// off of the input builder
return builder
.WithToggle(toggleIdentifier, projectionToggle);
});
```

This code will produce the following variants tree:

```
(base column)
☑ toggle
```

**There will not be any modes associated with this column.** Even though `builder.WithModes` was called, the `ColumnBuilder` returned by the method was the supplied (empty) builder with one `WithToggle` added.

Any time a new projection is associated with a variant, a `ColumnVariantDescriptor` must be used to uniquely identify the added variant amongst all variants in the column. The `ColumnVariantDescriptor` is created either
* Explicitly by the table author and passed into a `ColumnBuilder` method, such as the case when calling `WithToggle`
* Implicitly by the SDK runtime by associating the added variant with the base column, such as the case when calling `WithModes`

## Column Variant Types

There are two primary types of column variants: toggles and modes. While all variant types expose the same information (an alternative [projection](../../Glossary.md#projection) for the base [column](../../Glossary.md#column)), they differ in how they relate to other defined variants.

### Toggle Column Variants
A toggle column variant is a *single* variant that is mutually exclusive to all of its *parent* projections. Conceptually, it offers users a way to "toggle" to an alternate view of the toggle's parent projection.

For example, suppose a base column is added that exposes UTC `DateTime` information. An "as local time" toggle may be added that projects each value to a `DateTime` relative to the computer's timezone:

```cs
ColumnConfiguration columnConfiguration =
new ColumnConfiguration(new ColumnMetadata(new Guid("..."), "Time"));

IProjection<int, DateTime> asUtc = GetUtcProjection(); // Omitted for brevity
tableBuilder
.AddColumnWithVariants(columnConfiguration, asUtc, builder =>
{
ColumnVariantDescriptor local = new(new Guid("..."), "As Local Time");

return builder
.WithToggle(
local,
asUtc.Compose(utc => utc.ToLocalTime()));
});
```

This code will produce the following variants tree:

```
(base column)
☑ As Local Time
```

Toggles can be added iteratively on top of each other: it is possible to have a toggle from projection `A` to `B`, then another toggle from `B` to `C`, and so on.

### Mode Column Variants
Mode column variants are *a collection* of variants that are mutually exclusive to *each other*. Conceptually, they offer users a way to choose between one or more alternate views. Unlike toggle variants that can be in on/off states, a collection of modes should have one mode selected at all times.

There are two ways to define a set of modes:

1. Using `WithModes`, define modes at the base column level. In this configuration, the base column's projection is the first available mode in the collection. The base variant's `Guid` will automatically be set to the `ColumnConfiguration`'s identifier.

```cs
ColumnConfiguration columnConfiguration =
new ColumnConfiguration(new ColumnMetadata(new Guid("..."), "Time"));

IProjection<int, DateTime> asUtc = GetUtcProjection(); // Omitted for brevity
tableBuilder
.AddColumnWithVariants(columnConfiguration, asUtc, builder =>
{
return builder
.WithModes("UTC") // The name of the mode to associate with the base column projection
.WithMode(
new ColumnVariantDescriptor(new Guid("..."), "Local"),
asUtc.Compose(utc => utc.ToLocalTime()))
.WithMode(
new ColumnVariantDescriptor(new Guid("..."), "Binary"),
asUtc.Compose(utc => utc.ToBinary()));
});
```

This code will produce the following variants tree:

```
(base column) // This is equivalent to the "UTC" mode
UTCLocalBinary
```

`WithModes` can *only* be invoked from a `RootColumnBuilder`. It is not possible to call, for example, `WithToggle` followed by `WithModes`. This restriction is in place because, if you could do this, the column variant associated with the modes' direct parent would be overshadowed by the set of modes. If you wish to expose, for UX purposes, a toggle that exposes a set of modes, use `WithToggledModes` described below.

2. Using `WithToggledModes`, define a set of modes that is, collectively, mutually exclusive to all of its parents. For example, a `Timestamp` column could offer an "as DateTime" toggle that itself allows users to select between UTC vs local time.

```cs
ColumnConfiguration columnConfiguration =
new ColumnConfiguration(new ColumnMetadata(new Guid("..."), "Timestamp"));

IProjection<int, Timestamp> timestampProjection = GetTimestampProjection(); // Omitted for brevity
tableBuilder
.AddColumnWithVariants(columnConfiguration, timestampProjection, builder =>
{
return builder
.WithToggledModes(
"As DateTime",
modesBuilder =>
{
ColumnVariantDescriptor utc = new(new Guid("..."), "UTC");
ColumnVariantDescriptor local = new(new Guid("..."), "Local");

IProjection<int, DateTime> utcProjection = ToUtcDateTime(timestampProjection);

return modesBuilder
.WithMode(utc, utcProjection)
.WithMode(local, utcProjection.Compose(utc => utc.ToLocalTime()));
});
});
```

This code will produce the following variants tree:

```
(base column)
As DateTime
UTCLocal
```

Note how the `WithToggledModes` does not take a projection. The above code will result in a toggle with no associated variant being exposed to SDK drivers, which defines how the set of modes should be displayed. Depending on the SDK driver, the "UTC" and "Local" modes may not be presented to the user unless the "As DateTime" toggle is in use.

`WithToggledModes` may be called after `WithToggle`, meaning you can end a chain of hierarchical toggles with a set of toggled modes. However, you are unable to continue adding regular toggles on top of toggled modes. If you wish to add a toggle after one or more modes in a collection, you must supply a callback `Func` when adding the mode(s), as explained below.

## Recursive Variants

Both `WithModes` and `WithMode` described above have overloaded methods that take a `Func<ToggleableColumnBuilder, ColumnBuilder>`. These methods allow you to add variants that are direct children of the mode that that is being added, letting you define "recursive" column variants.

For example, consider this code:

```cs
ColumnConfiguration columnConfiguration =
new ColumnConfiguration(new ColumnMetadata(new Guid("..."), "Time"));

IProjection<int, DateTime> asUtc = GetUtcProjection(); // Omitted for brevity
IProjection<int, DateTime> asLocal = asUtc.Compose(utc => utc.ToLocalTime();

tableBuilder
.AddColumnWithVariants(columnConfiguration, asUtc, builder =>
{
return builder
.WithModes("UTC") // The name of the mode to associate with the base column projection
.WithMode(
new ColumnVariantDescriptor(new Guid("..."), "Local"),
asLocal,
modeBuilder =>
{
ColumnVariantDescriptor withDST = new(new Guid("..."), "With DST");

return modeBuilder
.WithToggle(
withDST,
asLocal.Compose(local => FixDST(local)));
});
});
```

This code will create a "Time" column that, at the root level, has two available modes:
- "UTC" mode, which uses the base column's projection
- "Local" mode, which converts each UTC datetime to local time

In addition to these modes, this code adds a toggle "With DST" to the "Local" mode. Conceptually, this means that when a user is choosing to display the time as a local DateTime, they have the option to toggle DST on and off.

The variants tree therefore looks like
```
(base column)
UTCLocal
With DST
```

where "With DST" is a child of "Local" but not "UTC." Depending on the SDK driver, "With DST" may not be presented to the user unless the "Local" mode is selected.

If desired, it is also possible to define new sub-modes of a given mode using `WithToggledModes` in the callback.

> ❗ The ability to recursively define column variants within a mode makes it possible to define arbitrarily complex trees of column variants. For a better user experience, it is recommended to limit the number of levels of column variants; **if your column has a complex tree of variants, you should consider creating new columns instead**.

# Defining Default Column Variants
Starting in SDK version `1.3`, you can specify on a `ColumnConfiguration` the `Guid` of `ColumnVariantDescriptor` that should be used as the default presentation of the column. You may also add this property to any prebuilt table configuration JSON files, as long as your JSON file uses version `1.3` of the JSON schema.

# Globally Apply Variants Based on Data Type

In some cases table authors have many data columns of the same type available across many tables. For example, a plugin may have several tables that each have one or more `DateTime` columns.

In these cases, it may be useful to define a common set of variants for all columns of that specific type. Instead of duplicating calls to `AddColumnWithVariants`, a simple way to accomplish this is to create, at the plugin level, `ITableBuilderWithRowCount` extension methods that add these global variants.

For example, you could define extension methods for adding `DateTime` columns that always add a "local time" variant:

```cs
public static class TableBuilderExtensions
{
public static void AddColumn(
this ITableBuilderWithRowCount tableBuilder,
ColumnConfiguration columnConfiguration,
IProjection<int, DateTime> dateTimeProjection)
{
tableBuilder.AddColumn(new DataColumn<DateTime>(columnConfiguration, projection));
}

public static void AddColumn(
this ITableBuilderWithRowCount tableBuilder,
IDataColumn<DateTime> dateTimeProjection)
{
tableBuilder.AddColumnWithVariants(
column,
builder =>
{
return builder
.WithToggle(
new ColumnVariantDescriptor(new Guid("..."), "As Local Time"),
column.Projector.Compose(t => t.ToLocalTime()));
});
}
}
```

If there are cases where an added `DateTime` column needs additional variants outside of the ones applied in the extension method, you could define overloaded versions that take in an additional callback:

```cs
public static class TableBuilderExtensions
{
public static void AddColumn(
this ITableBuilderWithRowCount tableBuilder,
ColumnConfiguration columnConfiguration,
IProjection<int, DateTime> dateTimeProjection)
{
tableBuilder.AddColumnWithVariants(columnConfiguration, dateTimeProjection, null);
}

public static void AddColumnWithVariants(
this ITableBuilderWithRowCount tableBuilder,
ColumnConfiguration columnConfiguration,
IProjection<int, DateTime> dateTimeProjection,
Func<ToggleableColumnBuilder, ColumnBuilder> columnBuilder)
{
tableBuilder.AddColumn(
new DataColumn<DateTime>(columnConfiguration, dateTimeProjection),
columnBuilder);
}

public static void AddColumn(
this ITableBuilderWithRowCount tableBuilder,
IDataColumn<DateTime> dateTimeColumn)
{
tableBuilder.AddColumn(dateTimeColumn, null);
}

public static void AddColumnWithVariants(
this ITableBuilderWithRowCount tableBuilder,
IDataColumn<DateTime> dateTimeColumn,
Func<ToggleableColumnBuilder, ColumnBuilder> columnBuilder)
{
tableBuilder.AddColumnWithVariants(
dateTimeColumn,
builder =>
{
var withToggle = builder
.WithToggle(
new ColumnVariantDescriptor(new Guid("..."), "As Local Time"),
dateTimeColumn.Projector.Compose(t => t.ToLocalTime()));

if (columnBuilder != null)
{
return columnBuilder(withToggle);
}
else
{
return withToggle;
}
});
}
}
```

# Using Column Variants

Registered column variants are exposed as `IDataColumn` instances where
* The `IDataColumn.Configuration` is derived from the base column's configuration and the variant's `ColumnVariantDescriptor`
* The data column's projection is the variant's projection

For information on how to obtain `IDataColumn`s for column variants via the SDK Engine, please refer to the "Using Column Variants" section of the [Using the Engine](../Using-the-engine.md#using-column-variants) documentation.


Loading

0 comments on commit 78158c2

Please sign in to comment.