Reputation: 10875
Using HttpWebRequest, I can call XDocument.Save() to write directly to the request stream:
XDocument doc = ...;
var request = (HttpWebRequest)WebCreate.Create(uri);
request.method = "POST";
Stream requestStream = request.GetRequestStream();
doc.Save(requestStream);
Is it possible to do the same thing with HttpClient
? The straight-forward way is
XDocument doc = ...;
Stream stream = new MemoryStream();
doc.Save(stream);
var content = new System.Net.Http.StreamContent(stream);
var client = new HttpClient();
client.Post(uri, content);
But this creates another copy of the XDocument
in the MemoryStream
.
Upvotes: 5
Views: 3115
Reputation: 245066
XDocument.Save()
expects a Stream
that can be written to. StreamContent
expects a stream that can be read. So, you can use a two Stream
s, where one acts as as a forwarder for the other one. I don't think such type exists in the framework, but you can write one yourself:
class ForwardingStream
{
private readonly ReaderStream m_reader;
private readonly WriterStream m_writer;
public ForwardingStream()
{
// bounded, so that writing too much data blocks
var buffers = new BlockingCollection<byte[]>(10);
m_reader = new ReaderStream(buffers);
m_writer = new WriterStream(buffers);
}
private class ReaderStream : Stream
{
private readonly BlockingCollection<byte[]> m_buffers;
private byte[] m_currentBuffer;
private int m_readFromCurrent;
public ReaderStream(BlockingCollection<byte[]> buffers)
{
m_buffers = buffers;
}
public override void Flush()
{}
public override long Seek(long offset, SeekOrigin origin)
{
throw new NotSupportedException();
}
public override void SetLength(long value)
{
throw new NotSupportedException();
}
public override int Read(byte[] buffer, int offset, int count)
{
if (m_currentBuffer == null)
{
if (!m_buffers.TryTake(out m_currentBuffer, -1))
{
return 0;
}
m_readFromCurrent = 0;
}
int toRead = Math.Min(count, m_currentBuffer.Length - m_readFromCurrent);
Array.Copy(m_currentBuffer, m_readFromCurrent, buffer, offset, toRead);
m_readFromCurrent += toRead;
if (m_readFromCurrent == m_currentBuffer.Length)
m_currentBuffer = null;
return toRead;
}
public override void Write(byte[] buffer, int offset, int count)
{
throw new NotSupportedException();
}
public override bool CanRead
{
get { return true; }
}
public override bool CanSeek
{
get { return false; }
}
public override bool CanWrite
{
get { return false; }
}
public override long Length
{
get { throw new NotSupportedException(); }
}
public override long Position
{
get { throw new NotSupportedException(); }
set { throw new NotSupportedException(); }
}
}
private class WriterStream : Stream
{
private readonly BlockingCollection<byte[]> m_buffers;
public WriterStream(BlockingCollection<byte[]> buffers)
{
m_buffers = buffers;
}
public override void Flush()
{}
public override long Seek(long offset, SeekOrigin origin)
{
throw new NotSupportedException();
}
public override void SetLength(long value)
{
throw new NotSupportedException();
}
public override int Read(byte[] buffer, int offset, int count)
{
throw new NotSupportedException();
}
public override void Write(byte[] buffer, int offset, int count)
{
if (count == 0)
return;
var copied = new byte[count];
Array.Copy(buffer, offset, copied, 0, count);
m_buffers.Add(copied);
}
public override bool CanRead
{
get { return false; }
}
public override bool CanSeek
{
get { return false; }
}
public override bool CanWrite
{
get { return true; }
}
public override long Length
{
get { throw new NotSupportedException(); }
}
public override long Position
{
get { throw new NotSupportedException(); }
set { throw new NotSupportedException(); }
}
protected override void Dispose(bool disposing)
{
m_buffers.CompleteAdding();
base.Dispose(disposing);
}
}
public Stream Reader
{
get { return m_reader; }
}
public Stream Writer
{
get { return m_writer; }
}
}
Unfortunately, you can't read and write to those streams at the same time from the same thread. But you can use Task
to write from another thread:
XDocument doc = …;
var forwardingStream = new ForwardingStream();
var client = new HttpClient();
var content = new StreamContent(forwardingStream.Reader);
Task.Run(() => doc.Save(forwardingStream.Writer));
var response = client.Post(url, content);
Upvotes: 2