npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

ts-lenses

v0.9.4

Published

A statically typed lens library for TypeScript

Downloads

1,970

Readme

Type-safe lenses in TypeScript

By Dr. Giuseppe Maggiore

Modern Single Page Applications (SPA's) built in TypeScript and JavaScript need to manage more and more complex and nested data structures. For example, we could need to manage a state such as the following:

interface AppState {
  loginForm:{
    firstPage:{
      userName:string
      password:string
    },
    secondPage:{
      email:string
      accountType:number
    }
  }
}

Whenever we have to immutably update the userName as a result of a user action, we might end up writing code looking like the following:

const setUserName = (newUserName:string) => (s0:AppState) : AppState => ({
  ...s0,
  loginForm:{
    ...s0.loginForm,
    firstPage:{
      ...s0.loginForm.firstPage,
      userName:newUserName
    }
  }
})

While it is, I believe, pretty awesome that TypeScript offers us a type-safe spread operator, the nesting is a bit painful to watch. The fact that we cannot take advantage of the implicit context that we are copying s0, and as such we want to stick to it when updating its nested objects such as loginForm and firstPage requires repetition, which increases cognitive load when maintaining the code, and is error-prone (one might mistakenly write s1.loginForm.firstPage if the scope contains another state, which sometimes is the case, without the compiler being able to offer any helpful warnings).

In order to tackle this problem, I have built a simple lenses library that takes on the task of performing updates on TypeScript records in a way that is type-safe, and as contextually smart as possible.

Simple example

In order to run this example, please first install the package via npm install ts-lenses, and add import { Entity } from "ts-lenses" to the top of your file!

Consider a simple, shallow type such as:

interface Person {
  name:string,
  surname:string,
  age:number
}

We can wrap an object of type Person into a lazy Entity that can be updated with some smarter operators:

const p1 = Entity<Person>({ name:"John", surname:"Doe", age:27 })

We can now set values as follows:

const q1 = p1.set("age", a => a+1).set("name", _ => "Jane")

Notice that, in order to enable method chaining, the set operator does not return the final result, but rather a new Entity on which further operations can be performed. Of course, set is type-safe: "age" must be a valid attribute, and the setter function that updates the value must process an input and produce an output of the correct type.

When we are done with chaining operations, we can commit and then we get the resulting object with the values set correctly:

const q1 = p1.set("age", a => a+1).set("name", _ => "Jane").commit()

We can also do some nice things like change the structure (and thus the type) of the resulting record:

const q2 = p1.rename("age", "birthday", x => new Date("1-1-2001")).commit()

The type of q2 now has no age attribute anymore, and instead has a birthday of type Date:

q2 : {
  name:string,
  surname:string,
  birthday:Date
}

This means that further operator chaining after a rename cannot access the old attribute, but rather only the new:

IntelliSense after rename

More complex example

We can also work on nested objects. For example, consider a fictitious type such as:

interface NestedState {
  nesting1:{
    nesting2:{
      nesting3:{
        nesting4:{
          nesting5:{
            obscenelyNestedValueWeNeedToUpdate:number
          },
          slightlyLessObscenelyNestedValueWeNeedToUpdate:number
        }
      }
    }
  }
}

Imagine that we were tasked with incrementing both nested numbers by 1. A bit of a daunting task, especially if we consider the amount of repetition involved. Writing something like {...s0.nesting1.nesting2.nesting3.nesting4.nesting5, obscenelyNestedValueWeNeedToUpdate:s0.nesting1.nesting2.nesting3.nesting4.nesting5.obscenelyNestedValueWeNeedToUpdate+1} is not exactly that paragon of elegance that gives most developers that feeling of "yes! I love my job" :)

The library can help us a bit. The setIn operator facilitates setting nested values. The resulting code would look as follows:

const p2 = Entity<NestedState>({ nesting1:{ nesting2:{ nesting3:{ nesting4:{ slightlyLessObscenelyNestedValueWeNeedToUpdate:0, nesting5:{ obscenelyNestedValueWeNeedToUpdate:0 } } } } } })
const q21 = p2.setIn("nesting1", e => 
  e.setIn("nesting2", e => 
    e.setIn("nesting3", e => 
    e.setIn("nesting4", e => 
      e.set("slightlyLessObscenelyNestedValueWeNeedToUpdate", v => v + 2)
        .setIn("nesting5", e => 
          e.set("obscenelyNestedValueWeNeedToUpdate", v => v+1)
        )
      )
    )
  )
).commit()

Of course, we enjoy type-safety all the way down, and we can mix and match set and setIn as needed. We could even rename some of the nested attributes, in order to both update and restructure the input state for update-and-convert tasks:

IntelliSense, nesting, and renaming

The original issue

The original "challenging" bit of code then becomes:

const setUserName = (newUserName:string) => (s0:AppState) : AppState => 
  Entity(s0)
    .setIn("loginForm", e => e
    .setIn("firstPage", e => e
    .set("userName", _ => newUserName)))
    .commit()

Of course, it is a matter of personal preference, but I find this much more attractive than the original version!

Conclusion

Managing immutable update operations on complex nested states is a recurring challenge. In this article I present a small, new library that wraps these operations in a type-safe way, inspired from the lenses concept from Haskell.

Thanks to this library, which can be found on npm, you can process data quickly and easily, with enhanced productivity and less bugs.

Thank you for coming all the way to the end, I hope you enjoyed reading this article as much as I enjoyed writing it ;)

Appendix: about the author

Hi! I am Giuseppe Maggiore. I have an academic background (PhD) in Computer Science, specifically compilers and functional programming (not so surprising eh...). I am now CTO of Hoppinger, a wonderful software development company in the heart of Rotterdam (Netherlands).

I am always looking for talented software engineers who get excited at the thought of type safety, reliable software, functional programming, and so on. If that is the case, do get in touch with us, we always have open positions for smart people!