Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

https://github.com/anakic/jot

Jot is a library for persisting and applying .NET application state.
https://github.com/anakic/jot

csharp dotnet ioc library settings settings-storage

Last synced: 18 days ago
JSON representation

Jot is a library for persisting and applying .NET application state.

Lists

README

        

# Jot - a .NET library for state persistence

[![Build status](https://ci.appveyor.com/api/projects/status/3p31q9b15v46sudk/branch/master?svg=true)](https://ci.appveyor.com/project/anakic/jot/branch/master)

## Introduction
Almost every application needs to keep track of its own state, regardless of what it otherwise does. This typically includes:

1. Sizes and locations of movable/resizable elements of the UI (forms, tool windows, draggable toolbars...)
1. Last entered data (e.g. username, selected tab indexes, recently opened files...)
1. Settings and user preferences

A common approach is to store this data in a .settings file and read and update it as needed. This involves writing a lot of boilerplate code to copy that data back and forth. This code is generally tedious, error-prone and no fun to write.

With Jot, you only need to declare which properties of which objects you want to track, and when to persist and apply data. This is a better abstraction for this requirement, resulting in more readable and concise code.

## Installation

Jot is available on NuGet and can be installed from the package manager console:

`install-package Jot`

## Example: Persisting the size and location of a Window
To illustrate the basic idea, let's compare two ways of dealing with this requirement: .settings file (Scenario A) versus Jot (Scenario B).

### Scenario A (.settings file)

**Step 1:** Define settings

![](http://www.codeproject.com/KB/cs/475498/settings.jpg)

**Step 2:** Apply previously stored data

``` C#
public MainWindow()
{
InitializeComponent();

this.Left = MySettings.Default.MainWindowLeft;
this.Top = MySettings.Default.MainWindowTop;
this.Width = MySettings.Default.MainWindowWidth;
this.Height = MySettings.Default.MainWindowHeight;
this.WindowState = MySettings.Default.MainWindowWindowState;
}
```

**Step 3:** Persist updated data before the window is closed

``` C#
protected override void OnClosed(EventArgs e)
{
MySettings.Default.MainWindowLeft = this.Left;
MySettings.Default.MainWindowTop = this.Top;
MySettings.Default.MainWindowWidth = this.Width;
MySettings.Default.MainWindowHeight = this.Height;
MySettings.Default.MainWindowWindowState = this.WindowState;

MySettings.Default.Save();

base.OnClosed(e);
}
```

This is a lot of work, even for a single window. If there were 10 resizable/movable elements of the UI, the settings file would become a jungle of similarly named properties, making this code quite tedious and error prone to write.

Also notice that for each property of the window, we need to mention it in five places (in the settings file, twice in the constructor and twice in OnClosed).

### Scenario B (Jot)

**Step 1:** Create and configure the tracker

``` C#
// Expose services as static class to keep the example simple
static class Services
{
// expose the tracker instance
public static Tracker Tracker = new Tracker();

static Services()
{
// tell Jot how to track Window objects
Tracker.Configure()
.Id(w => w.Name)
.Properties(w => new { w.Height, w.Width, w.Left, w.Top, w.WindowState })
.PersistOn(nameof(Window.WindowClosed))
}
}
```

**Step 2:** Track the window instance

``` C#
public MainWindow()
{
InitializeComponent();

// Start tracking the Window instance.
// This will apply any previously stored data and start listening for "WindowClosed" event to persist new data.
Services.Tracker.Track(this);
}

```

That's it. We've set up tracking for all window objects in one place, so that all we need to to is call `tracker.Track(window)` on each window instance to preserve it's size and location. It's concise, the intent is clear, and there's no repetition. Notice also that we've mentioned each property only once, and it would be trivial to track additional properties.

### Real world form/window tracking

The above code (both scenarios) works but it doesn't account for a few things. The first one is multiple displays. Screens can be unplugged, and we never want to position a window onto a screen that's no longer there. We can get around this problem very easily if we make the screen resolution part of the identifier. Jot will then track the same window separately for each screen configuration.

#### WPF Window
Here's how to properly track a WPF window:

``` csharp
// 1. tell the tracker how to track Window objects (this goes in a startup class)
tracker.Configure()
.Id(w => w.Name, SystemInformation.VirtualScreen.Size) // <-- include the screen resolution in the id
.Properties(w => new { w.Top, w.Width, w.Height, w.Left, w.WindowState })
.PersistOn(nameof(Window.Closing))
.StopTrackingOn(nameof(Window.Closing));

// 2. in the Window constructor
public Window1()
{
// fetch the tracker instance e.g. via IOC or static property
var tracker = Services.Tracker;
tracker.Track(this);
}
```

The `Id` method has a `params object []` parameter that can be used to define a namespace for the id. These parameters simply get *ToString-ed* and concatenated to the Id. By using the screen resolution as the namespace, we ensure that we maintain separate configurations for different resolutions.

#### Windows forms

Winforms have a few additional caveats:
- Forms will return bogus size/location data for maximized/minimized forms, so we have to cancel persisting those
- Tracking needs to be applied during `OnLoad` since `Top` and `Left` properties set in the constructor are ignored

Here's how to properly track (Windows) Forms:

``` C#
// tell the tracker how to track Form objects (this goes in a startup class)
tracker.Configure()
.Id(f => f.Name, SystemInformation.VirtualScreen.Size) // <-- include the screen resolution in the id
.Properties(f => new { f.Top, f.Width, f.Height, f.Left, f.WindowState })
.PersistOn(nameof(Form.Move), nameof(Form.Resize), nameof(Form.FormClosing))
.WhenPersistingProperty((f, p) => p.Cancel = (f.WindowState != FormWindowState.Normal && (p.Property == nameof(Form.Height) || p.Property == nameof(Form.Width) || p.Property == nameof(Form.Top) || p.Property == nameof(Form.Left)))) // do not track form size and location when minimized/maximized
.StopTrackingOn(nameof(Form.FormClosing)); // <-- a form should not be persisted after it is closed since properties will be empty

// in the form code
protected override void OnLoad(EventArgs e)
{
// fetch the tracker instance e.g. via IOC or static property
var tracker = Services.Tracker;
tracker.Track(this);
}
```

## Which properties to track

There are two methods (and several overloads) for telling Jot which properties of a given type to track.

The `Properties` method accepts an expression that projects the target properties as an anonymous object:
```csharp
tracker.Configure()
.Properties(p => new
{
p.Name,
p.LastName,
MothersMaidenName = p.Mother.LastName // <-- can navigate object graph
})
```
The `Property` method is used to add propreties one by one. It allows specifying a name and a default value for each property. Since the property name can be passed as a string, this overload is useful for situations where the properties to track are determined at runtime.
```csharp
tracker.Configure()
.Property(p => p.Name)
.Property(p => p.LastName)
.Property(p => p.Age, -1) // if there's no value in the store, -1 will be set
.Property(p => p.Mother.LastName, "MothersMaidenName") // <-- a name must be provided so it does not colide with p.LastName
```

The expressions you provide to these methods are used to specify which properties to track. The properties usually belong to the target object itself but they can also navigate through other objects (e.g. `p.Mother.LastName`). Based on these expressions, Jot will dynamically generate *getter* and *setter* methods for reading and writing the data. Both methods (`Properties` and `Property`) are cumulative: they add properties to track, rather than overwrite previous calls.

## When is the data persisted?

Jot needs to know when a target's data has changed so it can save the updated data to the store. You can tell Jot to automatically persist a target whenever it (the target) fires an event:
```csharp
tracker.Configure()
.Properties(...)
.PersistOn(nameof(Foo.SomeEvent)) <-- the event that should trigger persisting
```
You can optionally specify another object as the source of the event:
```csharp
PersistOn("SomeEvent", otherObject)
```
You can also explicitly tell Jot to persist a target using the `Persist` method:
```csharp
tracker.Persist(targetObj);
```
To tell Jot to persist all tracked objects, use the `PersistAll` method:
```csharp
tracker.PersistAll();
```
Usually, this would be during an application shutdown or at the end of a web request. Jot maintains a list of weak references to target objects. Targets that are already garbage collected are ignored.

Some objects survive until the end of the application without being in a usable state. For example, a disposed form can still be referenced (and thus not garbage collected). We do not want to continue tracking that form after it is disposed because it will have bogus property values which we do not want to save to the store. For such cases, we can tell Jot to stop tracking a particular object by calling `StopTracking`:
```csharp
tracker.StopTracking(targetObj);
```
We can also tell Jot to automatically stop tracking an object when it raises a certain event:
```csharp
tracker.Configure()
.Properties(...)
.PersistOn(...)
.StopTrackingOn(nameof(Form.Closed)) <-- the event that should cause the tracker to stop tracking the target
```

## Where is the data stored?

The `Tracker` class constructor has an optional parameter that allows you to specify where the data will be stored.

``` C#
Tracker(IStore store)
```
Jot comes with a built-in implementation of `IStore` called `JsonFileStore`. If the `IStore` argument is not provided, the data will be stored in json files in the following folder: `%AppData%\[company name]\[application name]`. The *company name* and *application name* are read from the entry assembly's attributes). For each target object, there will be a separate file. Data is stored in separate files in order to make reading and writing data fast.

To keep using the JSON file store, but store the data in a per-machine folder (e.g. `CommonApplicationData`), configure the tracker like so:

``` C#
var tracker = new Tracker(new JsonFileStore(Environment.SpecialFolder.CommonApplicationData));
```

Or specify the storage folder explicitly:

``` C#
var tracker = new Tracker(new JsonFileStore(@"c:\example\path\"));
```

Here's what the stored data looks like:

![](http://i.imgur.com/xUVaVMh.png)

### Custom storage

The `IStore` interface is very simple. For a given Id, it needs to be able to store and retrieve a dictionary of values.

```C#
public interface IStore
{
void SetData(string id, IDictionary values);
IDictionary GetData(string id);
}
```

You can use this interface to make Jot store data anywhere you like e.g. in the cloud (to share settings for a user between machines) or a database.

## Value conversions and cancellation

Jot lets you hook into the Apply and Persist operations. You can use this to perform value conversion and cancel persisting or applying data. As we've seen in the WinForms example, we can cancel applying size/location properties for Forms that are maximized or minimized:

```csharp
tracker.Configure()
.Id(...)
.Properties(...)
.WhenPersistingProperty((f, p) => p.Cancel = (f.WindowState != FormWindowState.Normal && (p.Property == nameof(Form.Height) || p.Property == nameof(Form.Width) || p.Property == nameof(Form.Top) || p.Property == nameof(Form.Left))))
```

There are four hooks you can use: `WhenPersistingProperty`, `WhenApplyingProperty`, `WhenAppliedState` and `WhenPersisted`.

## Tracking and inheritance

Tracking is configured per-type, meaning that a separate `TrackingConfiguration` object will need to be defined for each type of object we track. This configuration object tells Jot how to track objects of that type, but it also applies to objects of derived types.

When configuring tracking for a derived type, Jot will examine the inheritance hierarchy of that type and look for the closest ancestor type for which a tracking configuration already exists. If it finds one, it will first create a copy of the base type's tracking configuration which you can then further customize.

For example, let's suppose you define a class called `MyForm` that derives from `Form`. In addition to tracking the size and location, you also want to track the selected tab of a TabControl that's part of `MyForm`. Here's what that might look like:

``` csharp
// configure tracking for Form
tracker.Configure()
.Id(f => f.Name, SystemInformation.VirtualScreen.Size)
.Properties(f => new { f.Height, f.Width, f.Left, f.Top, f.WindowState})
.PersistOn(nameof(Form.Closing))
.StopTrackingOn(nameof(Form.Closed))
.WhenPersistingProperty((f, p) => p.Cancel = (f.WindowState != FormWindowState.Normal && p.Property != nameof(Form.WindowState)))

// add the selected tab index for MyForm (everything else is already copied from the configuration for Form)
tracker.Configure()
.Properties(f => f.tabControl1.SelectedIndex);
```
We do not have to repeat the tracking configuration for size and location. Since `MyForm` derives from `Form`, the configuration for `MyForm` will be copied from the configuration for `Form` and we only need to add the additional `f.tabControl1.SelectedTabIndex` property.

Furthermore, if we configure tracking for `Form` but not for `MyForm`, Jot will track `MyForm` instances using the tracking configuration for `Form`.

## The ITrackingAware interface

Sometimes we cannot know at compile time which properties to track. In those situations, we need to configure tracking on a per-instance basis at runtime. To do this, our tracked objects can implement the `ITrackingAware` interface.

```csharp
public interface ITrackingAware
{
void ConfigureTracking(TrackingConfiguration configuration);
}
```
In the `ConfigureTracking` method, the object can dynamically specify which properties to track. The `configuration` parameter is specific to that instance (and not the type) so each instance can independently adjust its tracking configuration.

For example, let's assume we have a form that has a datagrid, and we want to track the widths of grid columns. We could track each grid column object as a separate object, but we can also track those columns as part of tracking the form. Here's what that might look like:

```csharp
public class MyFormWithDataGrid : ITrackingAware
{
protected override void OnLoad(EventArgs e)
{
Services.Tracker.Track(this);
}

public void InitConfiguration(TrackingConfiguration configuration)
{
// include data grid column widths when tracking this form
for (int i = 0; i < dataGridView1.Columns.Count; i++)
{
var idx = i; // capture i into a variable (cannot use i directly since it changes in each iteration)
configuration.Property("grid_c_" + dataGridView1.Columns[idx].Name, f => f.dataGridView1.Columns[idx].Width);
}
}
}
```

# IOC integration

Once we've explained to Jot how to track different types of objects, all that's needed in order for Jot to track instances of those types is to call:

``` C#
tracker.Track(obj);
```

Here's the really cool part... When using an IOC container, many objects in the application will be created by the container. This gives us an opportunity to automatically track all created objects by hooking into the container.

For example, with [SimpleInjector](https://simpleinjector.org/index.html) we can do this quite easily, with a single line of code:

``` C#
var tracker = new Jot.Tracker();
var container = new SimpleInjector.Container();

//configure tracking and apply previously stored data to all created objects
container.RegisterInitializer(d => { tracker.Track(d.Instance); }, cx => true);
```

With this in place, we can easily make any property of any object persistent, just by modifying the tracking configuration for its type. Neat!

# Demos

Demo projects for WPF and WinForms are included in the repository.

# Contributing

You can contribute to this project in the usual way:

1. First of all, don't forget to star the project
1. Fork the project
1. Push your commits to your fork
1. Make a pull request

# TODO
- Async support
- IOC demos
- aspnet core (demo + readme section)
- readme topics: unity ioc integration, attribute based configuration