RFC: Nested record definitions

Today, if you want to have nested records, you need to define each record as a distinct type, in the correct order, and plug them together. An example:

type persistOptionsFileConfig = {
  extension?: string
}
type persistOptions = {
  fileName: string,
  path?: string,
  fileConfig?: persistOptionsFileConfig
}

type options = {
  startFrom: float,
  persist?: persistOptions
}

@module("some-package")
external someFn: (string, ~options: options=?) => unit = "someFn"

This is because each record is a distinct, nominal type, that needs a proper definition. We have objects ({ "someField": bool }) which are structural, and where you can nest the definition directly. But, most often what you want is a record, because of the other characteristics of records.

This has some pretty large cons:

  • Having to write out each record definition becomes tedious and annoying
  • Types in these scenarios are often “one shot” as in the types aren’t intended to be reusable anyway
  • It’s harder to read (you usually see type names instead of the actual structure), and the tooling doesn’t always print all of the relevant record types in hovers etc for you to quickly be able to see and understand the structure
  • Copy/pasting types directly from TS is harder because in TS you usually nest definitions (because the structural type system lets you do it) whereas in ReScript you need to break the structure down into its parts.This also makes LLMs worse at converting TS and ReScript

Suggestion: Allow nested record definitions

What if you could do this instead of the above:

type options = {
  startFrom: float,
  persist?: {
    fileName: string,
    path?: string,
    fileConfig?: {
      extension?: string
    }
  }
}

@module("some-package")
external someFn: (string, ~options: options=?) => unit = "someFn"

Just defining your records inline. They still need to be actual, defined types, so the above could essentially be sugar for this:

type \"options.persist.fileConfig\" = {
  extension?: string
}
type \"options.persist\" = {
  fileName: string,
  path?: string,
  fileConfig?: \"options.persist.fileConfig\"
}
type options = {
  startFrom: float,
  persist?: \"options.persist\"
}

@module("some-package")
external someFn: (string, ~options: options=?) => unit = "someFn"

Real types, with autogenerated names that aren’t possible to reference unless using escaped identifiers (a reminder that you should probably break it out to a real type definition if you want to reuse it).

One could also imagining supporting the @as attribute to set the name for a generated type:

type options = {
  startFrom: float,
  persist?: @as("persistOptions") { // makes the type `type persistOptions` instead of `type \"options.persist\"
    fileName: string,
    path?: string,
    fileConfig?: {
      extension?: string
    }
  }
}

Tooling

The editor tooling could:

  • Give an easy way of “persisting”/extracting a generated type to its own type definition that you can
  • Print the nested records directly in hovers etc, so you always get to see the full structure whenever you have a nested record

Thoughts and feedback

So, what are your thoughts and feedback on this? Do you think it’d be valuable?

14 Likes

+1 I have already wanted this more than once.

In your options example, would you be able to create or reference any of the nested types by themselves, or would they always need to be a part of option?

type options = {
  persist?: {
    fileName: string
  }
}

let persist = {
  fileName: "foo"
}

let persist = opt => opt.fileName
3 Likes

They’d work by themselves, they’ll be an actual defined type to the compiler. It’s just that (in this proposal as of now) referring to them by typename can’t be done without using escaped identifiers, or by using the @as annotation. Your example above would work just fine because it’d be inferred to be the persist type.

2 Likes

I’ve had the need for this quite often when writing bindings!

1 Like

This rfc will make rescript more easily to use!

2 Likes

This would be great!

A PoC of this that soon will be usable (hopefully) exists here: PoC of nested record definitions by zth · Pull Request #7241 · rescript-lang/rescript · GitHub

1 Like

Real types, with autogenerated names that aren’t possible to reference unless using escaped identifiers (a reminder that you should probably break it out to a real type definition if you want to reuse it).

One could also imagining supporting the attribute to set the name for a generated type:@as

I think it’s right that reuse is explicit. What case does the @as addition proposal needed specifically?

It reminds me of the exportFragmentSpreadSubTypes option in the GraphQL code generator. It was quickly abused at scale, so it had to be disabled.

Yeah I lean towards that as well - make it a concrete definition if you need reuse. We can make the tooling make it super easy to extract to a concrete definition.

1 Like

Would nested variants be a stretch or do you want to keep this scoped to records?

Definitely not ruling out that, but we should start with records and see how that’s used. Variants are slightly different because of things like @unboxed and @tag where you often want/need to configure the variant at the type definition level. At a glance I think it’s a bit unclear how to achieve that in an intuitive way inline without introducing new syntax. But definitely not out of the question.

1 Like

I’m a little bit skeptic about using the @as attribute for renaming the underline types.

Because currently it exists to adjust the runtime representation of ReScript data and simply putting the @as to a wrong place will result in a completely different result:

type options = {
  startFrom: float,
  @as("persistOptions") persist?: { // makes the type `type persistOptions` instead of `type \"options.persist\"
    fileName: string,
    path?: string,
    fileConfig?: {
      extension?: string
    }
  }
}

I’d like to keep @as only for changing the runtime representation.

3 Likes