Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/edcarroll/class-discriminators

Aids Typescript type checking for class based discriminated unions
https://github.com/edcarroll/class-discriminators

decorators type-checking typescript

Last synced: about 2 months ago
JSON representation

Aids Typescript type checking for class based discriminated unions

Awesome Lists containing this project

README

        

# Class Discriminators

Aids Typescript type checking for class based discriminated unions.

Note that the object being tested doesn't need to be an instance of any of the classes, it only needs a discriminator value!

## Installation

```sh
$ npm install --save @edcarroll/class-discriminators
```

## Quickstart

Start by decorating your classes with their discriminator values:

```typescript
import { DiscriminatorValue } from "@edcarroll/class-discriminators";

export class Hello {
@DiscriminatorValue("hello")
public readonly type = "hello";

public sharedProp = 0;
public helloProp = 10;
}

export class World {
@DiscriminatorValue("world")
public readonly type = "world";

public sharedProp = 1;
public worldProp = 11;
}

// HelloWorld is the type union we are going to be narrowing.
export type HelloWorld = Hello | World;
```

Next, configure a guard factory using `configureGuardFactory` by providing the type we are narrowing from:

```typescript
import { configureGuardFactory } from "@edcarroll/class-discriminators";

const guardFactory = configureGuardFactory();
```

Note that you can optionally provide a property name to use as the discriminator property.

Now you can use the guard factory to test the type of various values.

### Array Filtering

The usefulness of this library shows itself when filtering arrays. This library works with the type checker to ensure that when you run `Array.filter` it returns an array typed as the union of the provided types, so that you can then run functions.

An example use case would be with Redux / ngrx and class based actions.

```typescript
const itemA = new Hello();
const itemB = new World();

// Take an array of multiple types
const arr = [itemA, itemB, itemA, itemB];

// You can use our configured guard factory to filter down to one or more specified types:
const hellos = arr
.filter(guardFactory(Hello))
.forEach(h => console.log(h.helloProp)); // prints 10, 10.

const worlds = console.log(arr
.filter(guardFactory(World))
.reduce((s, w) => w.worldProp + s, 0)); // prints 22.

// As mentioned before, `guardFactory` is a variadic higher order function,
// so you aren't limited to just 1 type:
const helloWorlds = arr
.filter(guardFactory(Hello, World))
.forEach(hw => console.log(hw.sharedProp)); // prints 0, 1, 0, 1.

// Also works well with Observables:
Observable
.of(itemA, itemB, itemA, itemB)
.filter(guardFactory(Hello))
.subscribe(i => console.log(i.helloProp)); // prints 10, 10
```

### Individual Guarding

Individual guarding is less useful as you can just `switch` on the discriminator property, however it works all the same:

```typescript
// i.e. we have an item but know only it's a hello or a world.
const item:HelloWorld = new Hello();

if (guardFactory(Hello)(item)) {
console.log(item.helloProp); // prints 10.
} else if (guardFactory(World)(item)) {
console.log(item.worldProp); // prints 11.
}

// The above will print 10 as item is a `Hello` (not a surprise for us,
// but at least the type checker is happy!)
```

### API

#### `@DiscriminatorValue(value:string):PropertyDecorator`

Property decorator that configures a class with the provided discriminator value. Decorator must be applied to the discriminator property as in the examples above.

#### `configureGuardFactory(discriminatorProperty?:keyof T):GuardFactory`

Takes an optional property name override (ommitting this value uses the property the `@DiscriminatorValue` decorator was applied to) to use for discriminator value checks, and returns a `GuardFactory` function.

##### `GuardFactory(...types:Class[]):Guard`

Takes `n` classes decorated with `@DiscriminatorValue` and returns a `Guard`.

*Note that the ...U above is not a real variadic kind, it's emulated using* ***a lot*** *of (up to 15) overflow declarations.*

##### `Guard(guardee:T) => guardee is U`

Takes an object of type `T` and returns `true` if its discriminator value is equal to one of the ones passed into the guard factory.

#### `extractDiscriminators(types:Class[]):string[]`

Takes an array of types and returns an array of their discriminator values.