我正在使用XmlWriter
发送XMPP(Jingle)流。 XMPP协议在协商TLS时替换XML流,这意味着XML最终会像:
<stream>
<features>
...
</features>
...TLS gets negotiated...
<stream>
...
</stream>
由于有两个流启动标记,因此XML最终没有很好地形成。
我想要做的是抛弃我在TLS协商之前使用的XmlWriter
并创建一个全新的,它允许我更好地模块化我的代码。但是,当我调用myXmlWriter.Close()
以确保它被处置时,它将发送关闭流结束元素,这会破坏XMPP协议。
无论如何我可以关闭XmlWriter
而不发送未完成的结束元素吗?
答案 0 :(得分:1)
创建一个可用于断开XmlWriter与基本流的中间流。
这不是最优雅的解决方案,下面的代码需要工作,所以在将其投入生产之前对其进行测试,但这是关于这个想法。
public class DummyStream : Stream
{
public DummyStream(Stream baseStream)
{
if (baseStream == null)
throw new ArgumentNullException("baseStream");
BaseStream = baseStream;
}
public Stream BaseStream { get; private set; }
public void DisconnectBaseStream()
{
BaseStream = null;
}
private Stream GetBaseStream()
{
return BaseStream ?? Stream.Null;
}
public override IAsyncResult BeginRead(byte[] buffer, int offset, int count, AsyncCallback callback, object state)
{
return GetBaseStream().BeginRead(buffer, offset, count, callback, state);
}
public override IAsyncResult BeginWrite(byte[] buffer, int offset, int count, AsyncCallback callback, object state)
{
return GetBaseStream().BeginWrite(buffer, offset, count, callback, state);
}
public override bool CanRead
{
get { return GetBaseStream().CanRead; }
}
public override bool CanSeek
{
get { return GetBaseStream().CanSeek; }
}
public override bool CanTimeout
{
get { return GetBaseStream().CanTimeout; }
}
public override bool CanWrite
{
get { return GetBaseStream().CanWrite; }
}
public override void Close()
{
// We do not close the BaseStream because this stream
// is just a wrapper.
// GetBaseStream().Close();
}
public override ObjRef CreateObjRef(Type requestedType)
{
return GetBaseStream().CreateObjRef(requestedType);
}
protected override void Dispose(bool disposing)
{
base.Dispose(disposing);
// We do not dispose the BaseStream because this stream
// is just a wrapper.
}
public override int EndRead(IAsyncResult asyncResult)
{
return GetBaseStream().EndRead(asyncResult);
}
public override void EndWrite(IAsyncResult asyncResult)
{
GetBaseStream().EndWrite(asyncResult);
}
public override bool Equals(object obj)
{
return GetBaseStream().Equals(obj);
}
public override void Flush()
{
GetBaseStream().Flush();
}
public override int GetHashCode()
{
return GetBaseStream().GetHashCode();
}
public override object InitializeLifetimeService()
{
return GetBaseStream().InitializeLifetimeService();
}
public override long Length
{
get { return GetBaseStream().Length; }
}
public override long Position
{
get { return GetBaseStream().Position; }
set { GetBaseStream().Position = value; }
}
public override int Read(byte[] buffer, int offset, int count)
{
return GetBaseStream().Read(buffer, offset, count);
}
public override int ReadByte()
{
return GetBaseStream().ReadByte();
}
public override int ReadTimeout
{
get { return GetBaseStream().ReadTimeout; }
set { GetBaseStream().ReadTimeout = value; }
}
public override long Seek(long offset, SeekOrigin origin)
{
return GetBaseStream().Seek(offset, origin);
}
public override void SetLength(long value)
{
GetBaseStream().SetLength(value);
}
public override string ToString()
{
return GetBaseStream().ToString();
}
public override void Write(byte[] buffer, int offset, int count)
{
GetBaseStream().Write(buffer, offset, count);
}
public override void WriteByte(byte value)
{
GetBaseStream().WriteByte(value);
}
public override int WriteTimeout
{
get { return GetBaseStream().WriteTimeout; }
set { GetBaseStream().WriteTimeout = value; }
}
}
此类用作XmlWriter
和XmlWriter
输出到的流之间的流。此类只是将所有来自XmlWriter
的呼叫转发到基本流,但是一旦您呼叫DisconnectBaseStream
,它就会停止转发它们,XmlWriter
无法再控制基本流。
您可以像这样使用此类:
using (var stream = /* stream used to communicate with */)
{
using (var wrapperStream = new DummyStream(stream))
using (var writer = XmlWriter.Create(wrapperStream))
{
// Do you work here.
// Now, disconnect the dummy stream so that the XML writer
// cannot send more data.
wrapperStream.DisconnectBaseStream();
// End of the using block will close the XmlWriter and it
// cannot send more data to the base stream.
}
// Perform TLS negotiation etc...
}
同样,DummyStream
是一个起点,需要一些工作。例如,您需要确保XmlWriter
在断开连接后没有进行调用,因此您需要进行一些检查,例如: Write
方法,BaseStream
是否为null
,如果是,请跳过此呼叫。