AnswerWhy is this code broken?
I asked why this code is broken, and now is the time to dig into this. The issue is in this block of code. Take a look at that for a moment, if you please:
The code is trying to gather the async upload of all the files, and then it will await them. This code compile and runs successfully, but it will not do what you expect it to do. Let’s break it up a bit to understand what is going on:
We are passing the variable task to the list of tasks we have. We just extract a variable, nothing much going on here. Let’s explore further, what is the type of task? We know it must be a subtype of Task, since that is what the tasks collection will take. It turns out that this isn’t that simple:
What is that, Task<Task> thingie? Well, let’s look at the signature of Task.Factory.StartNew(), shall we?
public Task<TResult> StartNew<TResult>(Func<TResult> function);
Just from the signature, we can tell what is going on. StartNew() accepts a function that returns a value and will then return a task for the eventual result of this function. However, the function we are actually passing to the StartNew() doesn’t produce a value. Except that it does…
Let’s explore that thing for a bit:
var func = async () => { };
What is the type of func in this case?
Func<Task> func = async() => {};
The idea is that when the compiler sees the async keyword, it transforms the function to one that returns a Task. Combining both those features together means that our original code actually registers the start of the asynchronous process to happen and will return as soon as it got started. Basically, we’ll only wait for the actual opening of the file, not for the actual network work that has to happen here.
The right way to express what we want here is:
Task.Run(async () => {});
The signature for this is:
public static Task Run<TResult>(Func<Task> function);
You can see here that we get a function that returns a task, but we aren’t actually wrapping that in another Task instance. The task that will be returned will be completed once the full work has been completed.
It is an interesting pitfall in the API, and can be quite hard to figure out exactly what is going on. Mostly because there are several different things happening all at once.
More posts in "Answer" series:
- (22 Jan 2025) What does this code do?
- (05 Jan 2023) what does this code print?
- (15 Dec 2022) What does this code print?
- (07 Apr 2022) Why is this code broken?
- (20 Jan 2017) What does this code do?
- (16 Aug 2011) Modifying execution approaches
- (30 Apr 2011) Stopping the leaks
- (24 Dec 2010) This code should never hit production
- (21 Dec 2010) Your own ThreadLocal
- (11 Feb 2010) Debugging a resource leak
- (03 Sep 2009) The lazy loaded inheritance many to one association OR/M conundrum
- (04 Sep 2008) Don't stop with the first DSL abstraction
- (12 Jun 2008) How many tests?
Comments
Task.Run seeems more natural, but given the original implementation, I was expecting the change to be removing the async operator on the argument to Task.Factory.StartNew, i.e.
I think this would work the same from the consumer's standpoint, but I'm guessing StartNew() might always allocate a new task, whereas Task.Run() might re-use an existing task if one is available. I'm not sure if that is the case or not, so I'm hoping somebody more knowledgeable than me can chime in.
Chris,
If you made this a sync call, you'll kill the system, see:
https://ayende.com/blog/166913/mixing-sync-async-calls?msclkid=3f2288c4b68011ec8ade150d41a92c5b
I know it's obviously not the point, but why not simply do:
The error would also more easily apparent! Seems like an odd use of task.run ...
Interesting how the error is masked by the usage of
var
and becomes much more obvious with the explicit type shown. That should be one more good reason to have an.editorconfig
that forces explicit variable type declarations when the type is not obvious:@Konstantine You can't just do that because of the
using
statement. But you can still get a similar effect:I assume
StartNew
/Run
is an effort to escape from the current synchronization context, or if you need an overload withTaskCreationOptions
, otherwise I am not sure what value it's adding here.Separately there is a very (mis?)use of StreamContent: it's
IDisposable
so it should get its ownusing var ...
Konstantine,
The actual scenario that we ran into was a bit more complex. It was multiple async calls, so wrapping that in an async method and invoking in this manner was easier.
Bertrand,
Yes, I do a lot of code reviews in GitHub PR (no IDE there), and I found that to be essential to understand what is going on at a glance
@Oren you should try pressing . inside your PR, it does give a better environment to see everything
@Oren you should try JetBrains Rider to make Github PR reviews directly in a full blown IDE
@Steve,
Wow, I wasn't aware that they had this feature. That is pretty awesome!
@Steve, what do you mean by pressing . ? I don't see anything mapped to it in https://docs.github.com/en/get-started/using-github/keyboard-shortcuts#issues-and-pull-requests nor anything happens when I press it :(
@Milosz I found it here: https://dev.to/lostintangent/10-awesome-things-you-can-do-with-github-dev-5fm7
@Milosz it gives you a visual studio code in the browser. You can also trigger it explicitly by changing github.com to github.dev in your URL
Thanks, it didn't work for me at first because I wasn't logged-in :)
Comment preview