System.Net.Http.HttpClient.PostAsync блокирует и никогда не возвращает
У меня есть приложение Windows Forms.NET framework с формой, которая имеет этот код:
using System;
using System.Collections.Generic;
using System.IO;
using System.Net.Http;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace test
{
public partial class Main : Form
{
public int exitCode = 1;
private Options opts;
CancellationTokenSource cancellationSource = new CancellationTokenSource();
public Main(Options opts)
{
InitializeComponent();
this.opts = opts;
}
private void btnCancel_Click(object sender, EventArgs e)
{
exitCode = 1;
cancellationSource.Cancel();
Close();
}
async Task doUpload()
{
using (var content = new MultipartFormDataContent())
{
List<FileStream> streams = new List<FileStream>();
try
{
foreach (string fPath in opts.InputFiles)
{
FileStream stream = new FileStream(fPath, FileMode.Open, FileAccess.Read);
streams.Add(stream);
content.Add(new StreamContent(stream), fPath);
}
var progressContent = new ProgressableStreamContent(
content,
4096,
(sent, total) =>
{
double percent = 100 * sent / total;
progressBar.Value = (int)percent;
});
using (var client = new HttpClient())
{
using (var response = await client.PostAsync(opts.URL, progressContent, cancellationSource.Token))
{
if (response.IsSuccessStatusCode)
{
exitCode = 0;
}
else
{
MessageBox.Show(
response.Content.ToString(),
"Error " + response.StatusCode,
MessageBoxButtons.OK, MessageBoxIcon.Error
);
}
Close();
}
}
}
finally
{
foreach (FileStream stream in streams)
{
stream.Close();
}
}
}
}
private void Main_Load(object sender, EventArgs e)
{
}
private void Main_FormClosing(object sender, FormClosingEventArgs e)
{
e.Cancel = !cancellationSource.IsCancellationRequested;
}
private void Main_Shown(object sender, EventArgs e)
{
doUpload();
}
}
}
ProgressableStreamContent - это то же самое, что было указано здесь: C#: HttpClient, Ход загрузки файла при загрузке нескольких файлов как MultipartFormDataContent
Проблема в том, что ответ не возвращается. Другими словами: ожидание postAsync никогда не завершается. Кроме того, обратный вызов прогресса никогда не вызывается. Даже если я попытаюсь использовать URL-адрес POST, содержащий несуществующий домен, ничего не произойдет. Думаю, это тупик, но не понимаю, как? Результат async Task никогда нигде не используется и его не ждут.
Он отличается от примера async/await, который вызывает взаимоблокировку, потому что.Result не используется и метод никогда не ожидается, а также кажется, что вызов ConfigureAwait(false) не имеет никакого эффекта.
ОБНОВЛЕНИЕ: я создал новое репозиторий github для этого вопроса, поэтому любой может его протестировать:
https://github.com/nagylzs/csharp_http_post_example
ОБНОВЛЕНИЕ: наконец-то это работает. ConfigureAwait не требуется. Все операции обновления пользовательского интерфейса должны быть размещены внутри Invoke. Я обновил тестовое репо до рабочей версии. Также добавлена поддержка TLSv1.2 (по умолчанию отключена).
2 ответа
PostAsync
в опубликованном вами коде не блокируется (но на самом деле никогда не возвращается!). Выдает исключение:
System.InvalidOperationException: Cross-thread operation not valid: Control 'progressBar' accessed from a thread other than the thread it was created on.
Это причина того, что точки останова у вас не сработали. Правильным решением будет:
var progressContent = new ProgressableStreamContent(
content,
4096,
(sent, total) =>
{
Invoke((Action) (() => {
double percent = 100 * sent / total;
progressBar.Value = (int) percent;
}));
});
(либо добавить Invoke
или BeginInvoke
к обратному вызову)
Обратные вызовы HTTP-клиента вызываются в фоновом потоке, и вы должны поместить их в четную очередь вашего окна, если вы хотите, чтобы они имели доступ к элементам управления пользовательского интерфейса.
.ConfigureAwait(false)
не имеет ничего общего с этой проблемой, вы не должны использовать его в контексте пользовательского интерфейса (как раз наоборот: вы хотите, чтобы он помещал продолжение в поток пользовательского интерфейса, поэтому вам не следует его использовать).
Вам нужно изменить это:
client.PostAsync(opts.URL, progressContent, cancellationSource.Token)
к
client.PostAsync(opts.URL, progressContent, cancellationSource.Token).ConfigureAwait(false)
Это уже обсуждается, поэтому вы можете найти дополнительные ресурсы в сети, но это должно быть хорошей отправной точкой.