Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/lue-bird/elm-no-record-type-alias-constructor-function

trick: no record `type alias` constructor function
https://github.com/lue-bird/elm-no-record-type-alias-constructor-function

constructor elm record type-alias

Last synced: about 1 month ago
JSON representation

trick: no record `type alias` constructor function

Awesome Lists containing this project

README

        

> trick: no record `type alias` constructor function

[forms example in guide.elm-lang.org](https://guide.elm-lang.org/architecture/forms.html):
```elm
type alias Model =
{ name : String
, password : String
, passwordAgain : String
}

init : Model
init =
Model "" "" ""
```
↑ Every directly aliased record type gets its default constructor function.

You can trick the compiler into not creating a `Model` record constructor function:

```elm
import RecordWithoutConstructorFunction exposing (RecordWithoutConstructorFunction)

type alias Model =
RecordWithoutConstructorFunction
{ name : String
, password : String
, passwordAgain : String
}

init : Model
init =
-- Model "" "" "" <- error
{ name = "", password = "", passwordAgain = "" }
```

where

```elm
type alias RecordWithoutConstructorFunction record =
record
```

### tips

- find & fix your current _usages_ of record `type alias` constructor functions: [`elm-review` rule `NoRecordAliasConstructor`](https://dark.elm.dmy.fr/packages/lue-bird/elm-review-record-alias-constructor/latest/NoRecordAliasConstructor)

- insert `RecordWithoutConstructorFunction`/... where necessary: [`elm-review` rule `NoRecordAliasWithConstructor`](https://dark.elm.dmy.fr/packages/lue-bird/elm-review-record-alias-constructor/latest/NoRecordAliasWithConstructor)

## why

Fields in a record don't have a "natural order".

```elm
{ age = 42, name = "Balsa" }
== { name = "Balsa", age = 42 }
--> True
```

So it shouldn't matter whether you write

```elm
type alias User =
{ name : String, age : Int }
or { age : Int, name : String }
```
as well.

```elm
User "Balsa" 42
```
however relies on a specific field order in the type and is more difficult to understand/read.
These constructors also open up the possibility for bugs to sneak in without the compiler warning you:

```elm
type alias User =
{ status : String
, name : String
}

decodeUser : Decoder User
decodeUser =
map2 User
(field "name" string)
(field "status" string)
```
Did you spot the mistake? [↑ a similar example](https://sporto.github.io/elm-patterns/advanced/pipeline-builder.html#caveat)

To avoid these kinds of bugs, just **forbid type alias constructors**:
```elm
type alias User =
RecordWithoutConstructorFunction ...
```

Problems don't end there.

### implicit magic

> ["There are worse things than being explicit" – Evan](https://twitter.com/evancz/status/928359289135046656)

Most record type aliases are not intended to work with positional arguments!
`Model` is the perfect example.

Even if you think it's ok currently, no one reminds you when you add new fields.

### there are better alternatives

It's so easy to create an explicit constructor

```elm
xy : Float -> Float -> { x : Float, y : Float }
```

As argued, unnamed arguments shouldn't be the _default_.

Additionally, your record will be more descriptive and type-safe as a `type`

```elm
type Cat
= Cat { mood : Mood, birthTime : Time.Posix }
```

to make wrapping, unwrapping and combining easier, you can try [typed-value](https://dark.elm.dmy.fr/packages/lue-bird/elm-typed-value/latest/).

### only works in very limited scenarios

```elm
type alias Record =
{ someField : () }

type alias Indirect =
Record
```

- `Record` has a constructor function
- `Indirect` doesn't have a constructor function

```elm
type alias Extended record =
{ record | someField : () }

type alias Constructed =
Extended {}
```

- `Constructed`, `Extended` don't have constructor functions

### name clash with variant

example adapted from [`Elm.Syntax.Exposing`](https://dark.elm.dmy.fr/packages/stil4m/elm-syntax/latest/Elm-Syntax-Exposing#TopLevelExpose)
```elm
type TopLevelExpose
= ...
| TypeExpose TypeExpose

type alias TypeExpose =
{ name : String
, open : Maybe Range
}
```
> NAME CLASH - multiple defined `TypeExpose` type constructors.
>
> How can I know which one you want? Rename one of them!

Either rename `type alias TypeExpose` to `TypeExposeData`/... or
```elm
type alias TypeExpose =
RecordWithoutConstructorFunction ...
```
and **get rid of the compile-time error**

### misunderstood as special magic constructors

[experience report by John Pavlick: LETWW, Part 2: "Regular expressions are quite confusing and difficult to use."](https://dev.to/jmpavlick/regular-expressions-are-quite-confusing-and-difficult-to-use-50l7) (read it):

> My prior lack of understanding was due to a mental disconnect between the two true sentences, "record type aliases come with implicit constructors" and "all constructors are functions"
>
> \[...\] After marinating for over a decade in a type system where type names are "special" and have to be invoked only in certain special-case contexts - I couldn't see \[...\]:
>
> Type names, just like everything else in Elm, are Not Special. They're constructors for a value [= functions].

### `succeed`/`constant` are misused

I'd consider `succeed`/`constant`/... with a constant value in record field value `Decoder`s/`Generator`s/... unidiomatic.

```elm
projectDecoder : Decoder Project
projectDecoder =
map3
(\name scale selected ->
{ name = name
, scale = scale
, selected = selected
}
)
(field "name" string)
(field "scale" float)
(succeed NothingSelected) -- weird
```

Constants should much rather be introduced explicitly in a translation step:

```elm
projectDecoder : Decoder Project
projectDecoder =
map2
(\name scale ->
{ name = name
, scale = scale
, selected = NothingSelected
}
)
(field "name" string)
(field "scale" float)
```

For record `Codec`s (from [MartinSStewart's `elm-serialize`](https://package.elm-lang.org/packages/MartinSStewart/elm-serialize/latest/) in this example) where we don't need to encode every field value:

```elm
serializeProject : Codec String Project
serializeProject =
record Project
|> field .name string
|> field .scale float
|> field .selected
(succeed NothingSelected)
|> finishRecord
```
`succeed` is a weird concept for codecs because some dummy value must be encoded which will never be read.

It does not exist in elm-serialize, but it does exist in [miniBill's `elm-codec`](https://package.elm-lang.org/packages/miniBill/elm-codec/latest) (, [prozacchiwawa's `elm-json-codec`](https://package.elm-lang.org/packages/prozacchiwawa/elm-json-codec/latest), ...):
> Create a Codec that produces null as JSON and always decodes as the same value.

Do you really want this behavior? If not, you'll need
```elm
serializeProject : Codec String Project
serializeProject =
record
(\name scale ->
{ name = name
, scale = scale
, selected = NothingSelected
}
)
|> field .name string
|> field .scale float
|> finishRecord
```
Why not consistently use this record constructing method?

This will also be used often for versioning
```elm
enum ProjectVersion0 [ ... ]
|> andThen
(\version ->
case version of
ProjectVersion0 ->
record
(\name -> { name = name, scale = 1 })
|> field .name string
|> finishRecord

...
)
```
Again: Why not consistently use this record constructing method?

## suggestions?
→ [contributing](https://github.com/lue-bird/elm-no-record-type-alias-constructor-function/blob/master/contributing.md).

## why a whole package

`RecordWithoutConstructorFunction.elm` can simply be copied to your project.

- [the `elm-review` rule that adds your `RecordWithoutConstructorFunction`](https://dark.elm.dmy.fr/packages/lue-bird/elm-review-record-alias-constructor/latest/NoRecordAliasWithConstructor) will still work

However, if you want

- no separate `RecordWithoutConstructorFunction`s hanging around
- a single place for up to date public documentation
- a common recognizable name
- safety that `RecordWithoutConstructorFunction` will never be aliased to a different type

consider
```monospace
elm install lue-bird/elm-no-record-type-alias-constructor-function
```

## fields constructor too verbose?

```elm
decodeUser =
map2 (\name status -> { name = name, status = status })
(field "name" string)
(field "status" string)
```
is rather verbose.

There are languages that introduce extra sugar:

### field addition

```elm
succeed {}
|> field &name "name" string
|> field &status "status" string
```

would be simple and neat. elm dropped this for simplicity.

### field "punning"

This is present in purescript and other languages

```elm
map2 (\name status -> { name, status })
(field "name" string)
(field "status" string)
```
Jeroen's made a convincing [argument on negative consequences for descriptiveness](https://discourse.elm-lang.org/t/record-creation-shortcut/6136/10) in contexts of functions growing larger.

### positional by fields

```elm
map2 { name, status }
(field "name" string)
(field "status" string)
```
with either
```elm
{ x, y } : Int -> Int -> { x : Int, y : Int }
{ x =, y = } : Int -> Int -> { x : Int, y : Int }
{ x = _, y = _ } : Int -> Int -> { x : Int, y : Int }
{ \x y } : Int -> Int -> { x : Int, y : Int }
```
The last one was proposed in a [very old discussion](https://github.com/elm/compiler/issues/73)

It's concise but quite limited in what it can do while not fixing many problems:

- problems with [`succeed`/`constant` misuse](https://dark.elm.dmy.fr/packages/lue-bird/elm-no-record-type-alias-constructor-function/latest/) remain
- → confusing `{ x, y = 0, z }` syntax necessary

- less intuitive?
- recognizing it as a function
- unlike punning in other languages

- less explicit than record field punning
a.k.a "doesn't scale well" (arguments are taken as they come, can't be combined, ...)

### `type alias` positional 1-field records

Explored in ["Safe and explicit records constructors exploration"](https://discourse.elm-lang.org/t/safe-and-explicit-records-constructors-exploration/4823).

Instead of
```elm
Point : Int -> Int -> Point

Point : { x : Int } -> { y : Int } -> Point
```

- problems with "doesn't scale:
can't expose extensible, extended indirect" remain (if not fixed somehow)
- defined field ordering should explicitly not matter
on by-nature-non-positional records
- could again seem
[magical and unintuitive](https://dev.to/jmpavlick/regular-expressions-are-quite-confusing-and-difficult-to-use-50l7)

## improve safety

```elm
Codec.group (\{ x } { y } -> { x = x, y = y })
|> Codec.part ( .x, \x -> { x = x })
Codec.int
|> Codec.part ( .y, \y -> { y = y })
Codec.int
```
or better, to always avoid shadowing errors:
```elm
Codec.group (\p0 p1 -> { x = .x p0, y = .y p1 })
|> Codec.part ( .x, \x -> { x = x })
Codec.int
|> Codec.part ( .y, \y -> { y = y })
Codec.int
```

Making this less verbose with not-necessarily-language-sugared-but-[code-generable](https://dark.elm.dmy.fr/packages/lue-bird/elm-review-missing-record-field-lens/latest/) field stuff:

```elm
Codec.group (\p0 p1 -> { x = .x p0, y = .y p1 })
|> Codec.part Record.x Codec.int
|> Codec.part Record.y Codec.int
```

with
```elm
import Record exposing (x, y)

x : Part x { x : x } { record_ | x : x }
x =
part
{ access = .x
, named = \x -> { x = x }
, alter = \alter r -> { r | x = r.x |> alter }
, description = "x"
}
```

I bet ideas like this won't be widely adopted (in packages) because setting up tools alongside might be seen as too much work (for beginners).