https://github.com/lue-bird/elm-syntax-to-fsharp
  
  
    transpile elm to F# 
    https://github.com/lue-bird/elm-syntax-to-fsharp
  
elm fsharp transpiler
        Last synced: 7 months ago 
        JSON representation
    
transpile elm to F#
- Host: GitHub
- URL: https://github.com/lue-bird/elm-syntax-to-fsharp
- Owner: lue-bird
- License: mit
- Created: 2024-12-28T10:17:59.000Z (10 months ago)
- Default Branch: main
- Last Pushed: 2025-01-16T14:37:04.000Z (9 months ago)
- Last Synced: 2025-04-05T00:11:31.438Z (7 months ago)
- Topics: elm, fsharp, transpiler
- Language: Elm
- Homepage:
- Size: 49.8 KB
- Stars: 6
- Watchers: 1
- Forks: 0
- Open Issues: 0
- 
            Metadata Files:
            - Readme: README.md
- Changelog: changes.md
- License: LICENSE
 
Awesome Lists containing this project
README
          Print pure [`elm-syntax`](https://dark.elm.dmy.fr/packages/stil4m/elm-syntax/latest/) declarations as
[F#](https://fsharp.org/) code.
While that might already be useful on its own (wasm target etc),
you can even compile the generated F# further into rust etc using [fable](https://fable.io/).
```elm
import Elm.Parser
import ElmSyntaxToFsharp
"""module Sample exposing (..)
plus2 : Int -> Int
plus2 n =
    n + ([ 2 ] |> List.sum)
"""
    |> Elm.Parser.parseToFile
    |> Result.mapError (\_ -> "failed to parse elm source code")
    |> Result.map
        (\syntaxModule ->
            [ syntaxModule ]
                |> ElmSyntaxToFsharp.modules
                |> .declarations
                |> ElmSyntaxToFsharp.fsharpDeclarationsToModuleString
        )
-->
Ok """namespace global
module rec Elm =
    let samplePlus2 : float -> float =
        fun n ->
            basics_add n (List.sum [ 2 ])
    ..and some default declarations..
"""
```
To try it out, you can
run [this node script](https://github.com/lue-bird/elm-syntax-to-fsharp/tree/main/node-elm-to-fsharp).
### be aware
-   only a subset of elm is currently supported. not supported:
    -   `elm/regex`, `elm/file`, `elm/bytes`, `elm/http`, `elm/random`, `elm/url`, `elm/json`, `elm/parser`, `elm/virtual-dom`,
        `elm/html`, `elm/svg`, `elm/browser`, `elm/time`, `elm-explorations/markdown`, `elm-explorations/webgl`, `elm-explorations/benchmark`, `elm-explorations/linear-algebra`
    -   `Platform`, `Platform.Cmd`, `Platform.Sub`, `Task`, `Process`
    -   ports, glsl, the prefix operator functions `(>>)` and `(<<)`
    -   `++` will default to `List.append` unless one of the arguments is a string literal. So e.g. use `a ++ b ++ ""` to append string variables (which is also faster in elm)
    -   extensible record types. For example, these declarations won't work (at let or module level):
        ```elm
        -- inferred { r | name : name } -> name
        getName =
            .name
        
        -- inferred name -> { r | name : name } -> { r | name : name }
        setName new r =
            { r | name = new }
        
        -- even if used in explicit annotation
        getName : { r | name : name } -> name
        getName =
            .name
        
        -- even if the extensible record type is "fully constructed"
        type alias Named otherFields =
            { otherFields | name : String }
        
        getName : Named { email : Email } -> String
        getName =
            .name
        ```
        Allowed however are for example:
        ```elm
        type alias User =
            { name : String, email : Email }
        
        userGetName : User -> String
        userGetName =
            .name
        ```
        Incidentally, avoiding extensible record types
        also tends to improve your elm code because it's simpler and makes the compiler errors more concrete
    -   potential future candidates: `Basics.(<<)`, `Basics.(>>)`, `Basics.clamp`, `Basics.degrees`, `Basics.turns`,
        `Basics.radians`, `Basics.logBase`, `Basics.atan2`, `Basics.toPolar`, `Basics.fromPolar`, `Basics.never`, `String.reverse`, `List.map5`, `List.map4`, `Char.toLocaleLower`, `Char.toLocaleUpper`, `Char.isAlpha`, `Char.isAlphaNum`, `Char.isDigit`, `Char.isOctDigit`, `Char.isHexDigit`, `List.head`, `List.tail`, `List.unzip`, `Dict.update`, `Dict.merge`, `Dict.intersect`, `Bitwise`, `Set`, `Array`. Any help appreciated!
-   no checks are performed before transpiling to fsharp. So if you don't add a compile check of your elm input,
    you might e.g. get a running program that circumvents an elm opaque type or phantom type, or a fsharp program that can't be run
-   not much care has been put into making the resulting code readable or even conventionally formatted
    and comments are not preserved
Please [report any issues](https://github.com/lue-bird/elm-syntax-to-fsharp/issues/new) you notice <3
### why F#?
-   it runs decently fast and can directly target Wasm
-   it's pretty much a superset of elm which makes transpiling easy