question-mark
Stuck on an issue?

Lightrun Answers was designed to reduce the constant googling that comes with debugging 3rd party libraries. It collects links to all the places you might be looking at while hunting down a tough bug.

And, if you’re still stuck at the end, we’re happy to hop on a call to see how we can help out.

FS3118 and FS0010 when combined with task CE and function name is too long

See original GitHub issue

Please accept my apologies if this is a known issue, or not reproducible.

Repro steps

Use Visual Studio to create a new .NET 6.0 F# class library. Modify the default Library.fs file by giving it this content:

module ClassLibrary1

open System.Threading.Tasks

let f2345 () = task {
    return () } :> Task

Expected behavior

The code compiles.

Actual behavior

Compilation fails with two error messages:

error FS0010: Unexpected symbol ':>' in binding. Expected incomplete structured construct at or before this point
or other token.

error FS3118: Incomplete value or function definition. If this is in an expression, the body of the expression must
be indented to the same column as the 'let' keyword.

Known workarounds

There are (at least) two workarounds to this problem:

  1. Make the function name shorter. With the given example, change the function name to f234, and the code compiles. The above code is a minimal repro of an issue I’ve run into more than once, and while I don’t understand the exact relationship, the maximum function name length seems to be somehow related to the last expression in the task CE. (Here, it’s just return (), but with a longer expression, the name can be longer, too.)
  2. Format the code like this:
module ClassLibrary1

open System.Threading.Tasks

let f2345 () =
    task {
        return () } :> Task

This alternative formatting seems to resolve the problem, and I haven’t seen any problems with it.

This syntax does, however, introduce an extra line of code, and what is worse, an extra level of indentation, which I’d like to avoid.

Related information

If you’re wondering about why anyone would want to upcast Task<'a> to Task this is only a minimal repro. My actual use case is asynchronous integration tests written with xUnit.net. The xUnit.net framework can execute Task-valued operations, but not Task<'a>-valued operations.

I suspect that the repro syntax may strictly speaking not be valid F# syntax, but usually it works. In any case it’s confusing that it compiles when the function name is short, but stops compiling when the name exceeds a threshold.

I’ve discovered this issue while working with F# in Visual Studio (details below), but it also reproduces with dotnet build from the command line. I don’t know if it reproduces in Visual Studio Code.

Edition Windows 11 Pro Version 22H2 Installed on ‎27.‎01.‎2023 OS build 22621.1848 Experience Windows Feature Experience Pack 1000.22642.1000.0

.NET 6

Microsoft Visual Studio Community 2022 Version 17.6.3 VisualStudio.17.Release/17.6.3+33801.468 Microsoft .NET Framework Version 4.8.09032

Installed Version: Community

Visual C++ 2022 00482-90000-00000-AA879 Microsoft Visual C++ 2022

ASP.NET and Web Tools 17.6.326.62524 ASP.NET and Web Tools

Azure App Service Tools v3.0.0 17.6.326.62524 Azure App Service Tools v3.0.0

Azure Functions and Web Jobs Tools 17.6.326.62524 Azure Functions and Web Jobs Tools

C# Tools 4.6.0-3.23259.8+c3cc1d0ceeab1a65da0217e403851a1e8a30086a C# components used in the IDE. Depending on your project type and settings, a different version of the compiler may be used.

CodeMaintainability 2022 1.2 Extension for tracking code maintainability of your methods. Instead of often performing report driven code analysis tools, you can use this extension to view in real-time maintainability score.

Common Azure Tools 1.10 Provides common services for use by Azure Mobile Services and Microsoft Azure Tools.

Microsoft JVM Debugger 1.0 Provides support for connecting the Visual Studio debugger to JDWP compatible Java Virtual Machines

NuGet Package Manager 6.6.0 NuGet Package Manager in Visual Studio. For more information about NuGet, visit https://docs.nuget.org/

Razor (ASP.NET Core) 17.6.0.2327201+a6a61fdfa748eaa65aab53dab583276e26af4a3e Provides languages services for ASP.NET Core Razor.

SQL Server Data Tools 17.6.13.0 Microsoft SQL Server Data Tools

Surrounder 1.0.12 Surrounds any selected texts with matching quotation or brace pairs

TypeScript Tools 17.0.20329.2001 TypeScript Tools for Microsoft Visual Studio

Visual Basic Tools 4.6.0-3.23259.8+c3cc1d0ceeab1a65da0217e403851a1e8a30086a Visual Basic components used in the IDE. Depending on your project type and settings, a different version of the compiler may be used.

Visual F# Tools 17.6.0-beta.23174.5+0207bea1afae48d9351ac26fb51afc8260de0a97 Microsoft Visual F# Tools

Visual Studio IntelliCode 2.2 AI-assisted development for Visual Studio.

Issue Analytics

  • State:open
  • Created 3 months ago
  • Reactions:3
  • Comments:7 (6 by maintainers)

github_iconTop GitHub Comments

2reactions
auduchinokcommented, Jun 19, 2023

It’s not related to the name length in any way, and the same error can be seen without the binding: Screenshot 2023-06-19 at 11 21 40

The issue happens due to how indentation syntax works in F#. Normally, an operator requires at least the same indentation as its operands, but there’s a special case allowing it to be less indented by its length + one space:

    1
  + 1
    ""
 :> obj

If the operator is indented less than that, then there’s this parsing error:

    ""
:> obj

What makes the case in this issue harder to analyze is the fact the braces allow additional deindentation inside them too. That doesn’t affect the subsequent code after them, and :> ends up being less indented than required by the context (seemingly started at {). The more idiomatic way to format it would probably be something like this:

module ClassLibrary1

open System.Threading.Tasks

let f2345 () =
    task {
        return ()
    } :> Task

In theory it should be possible to allow cases where a token could also have less indentation when continuing a deintented context before it, but it would require a change in the language, so in that case it would go to F# lang suggestions repo first and would be discussed and carefully designed there.

0reactions
TheAngryByrdcommented, Jul 5, 2023

Yeah this was a small detail I brought up in https://github.com/fsharp/fslang-design/issues/706#issuecomment-1284657617 so this applies to more than just casting.

Read more comments on GitHub >

github_iconTop Results From Across the Web

[style-guide] Stroustrup bracket style - fsharp/fslang-design
OK someTupledFunction ( 478815516, "A very long string making all of ... FS3118 and FS0010 when combined with task CE and function name...
Read more >
Cannot declare type in F#, Error FS0010, Error FS3118
I've been trying to fix this but the main fixes for the FS0010 error are just suggestions to indent the code correctly. I...
Read more >
Should I use AND in a function name?
Having a composite function name is acceptable, so which naming scheme you use depends on your context. There are purists who will tell...
Read more >
Are my method or variable names too long? [closed]
Calling a function with a long name and passing it two or three long variable names can make for one long line of...
Read more >
Function name is not following coding convention
Reports the functions' names that are either too short, too long, or do not follow the specified regular expression pattern.
Read more >

github_iconTop Related Medium Post

No results found

github_iconTop Related StackOverflow Question

No results found

github_iconTroubleshoot Live Code

Lightrun enables developers to add logs, metrics and snapshots to live code - no restarts or redeploys required.
Start Free

github_iconTop Related Reddit Thread

No results found

github_iconTop Related Hackernoon Post

No results found

github_iconTop Related Tweet

No results found

github_iconTop Related Dev.to Post

No results found

github_iconTop Related Hashnode Post

No results found