580 lines
21 KiB
C#
580 lines
21 KiB
C#
|
//------------------------------------------------------------
|
||
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
||
|
//------------------------------------------------------------
|
||
|
namespace System.ServiceModel.Security
|
||
|
{
|
||
|
using System.Diagnostics;
|
||
|
using System.IO;
|
||
|
using System.Runtime;
|
||
|
using System.ServiceModel;
|
||
|
using System.ServiceModel.Channels;
|
||
|
using System.ServiceModel.Diagnostics;
|
||
|
using System.Xml;
|
||
|
|
||
|
sealed class SecurityVerifiedMessage : DelegatingMessage
|
||
|
{
|
||
|
byte[] decryptedBuffer;
|
||
|
XmlDictionaryReader cachedDecryptedBodyContentReader;
|
||
|
XmlAttributeHolder[] envelopeAttributes;
|
||
|
XmlAttributeHolder[] headerAttributes;
|
||
|
XmlAttributeHolder[] bodyAttributes;
|
||
|
string envelopePrefix;
|
||
|
bool bodyDecrypted;
|
||
|
BodyState state = BodyState.Created;
|
||
|
string bodyPrefix;
|
||
|
bool isDecryptedBodyStatusDetermined;
|
||
|
bool isDecryptedBodyFault;
|
||
|
bool isDecryptedBodyEmpty;
|
||
|
XmlDictionaryReader cachedReaderAtSecurityHeader;
|
||
|
readonly ReceiveSecurityHeader securityHeader;
|
||
|
XmlBuffer messageBuffer;
|
||
|
bool canDelegateCreateBufferedCopyToInnerMessage;
|
||
|
|
||
|
public SecurityVerifiedMessage(Message messageToProcess, ReceiveSecurityHeader securityHeader)
|
||
|
: base(messageToProcess)
|
||
|
{
|
||
|
this.securityHeader = securityHeader;
|
||
|
if (securityHeader.RequireMessageProtection)
|
||
|
{
|
||
|
XmlDictionaryReader messageReader;
|
||
|
BufferedMessage bufferedMessage = this.InnerMessage as BufferedMessage;
|
||
|
if (bufferedMessage != null && this.Headers.ContainsOnlyBufferedMessageHeaders)
|
||
|
{
|
||
|
messageReader = bufferedMessage.GetMessageReader();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
this.messageBuffer = new XmlBuffer(int.MaxValue);
|
||
|
XmlDictionaryWriter writer = this.messageBuffer.OpenSection(this.securityHeader.ReaderQuotas);
|
||
|
this.InnerMessage.WriteMessage(writer);
|
||
|
this.messageBuffer.CloseSection();
|
||
|
this.messageBuffer.Close();
|
||
|
messageReader = this.messageBuffer.GetReader(0);
|
||
|
}
|
||
|
MoveToSecurityHeader(messageReader, securityHeader.HeaderIndex, true);
|
||
|
this.cachedReaderAtSecurityHeader = messageReader;
|
||
|
this.state = BodyState.Buffered;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
this.envelopeAttributes = XmlAttributeHolder.emptyArray;
|
||
|
this.headerAttributes = XmlAttributeHolder.emptyArray;
|
||
|
this.bodyAttributes = XmlAttributeHolder.emptyArray;
|
||
|
this.canDelegateCreateBufferedCopyToInnerMessage = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public override bool IsEmpty
|
||
|
{
|
||
|
get
|
||
|
{
|
||
|
if (this.IsDisposed)
|
||
|
{
|
||
|
// PreSharp Bug: Property get methods should not throw exceptions.
|
||
|
#pragma warning suppress 56503
|
||
|
throw TraceUtility.ThrowHelperError(CreateMessageDisposedException(), this);
|
||
|
}
|
||
|
if (!this.bodyDecrypted)
|
||
|
{
|
||
|
return this.InnerMessage.IsEmpty;
|
||
|
}
|
||
|
|
||
|
EnsureDecryptedBodyStatusDetermined();
|
||
|
|
||
|
return this.isDecryptedBodyEmpty;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public override bool IsFault
|
||
|
{
|
||
|
get
|
||
|
{
|
||
|
if (this.IsDisposed)
|
||
|
{
|
||
|
// PreSharp Bug: Property get methods should not throw exceptions.
|
||
|
#pragma warning suppress 56503
|
||
|
throw TraceUtility.ThrowHelperError(CreateMessageDisposedException(), this);
|
||
|
}
|
||
|
if (!this.bodyDecrypted)
|
||
|
{
|
||
|
return this.InnerMessage.IsFault;
|
||
|
}
|
||
|
|
||
|
EnsureDecryptedBodyStatusDetermined();
|
||
|
|
||
|
return this.isDecryptedBodyFault;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal byte[] PrimarySignatureValue
|
||
|
{
|
||
|
get { return this.securityHeader.PrimarySignatureValue; }
|
||
|
}
|
||
|
|
||
|
internal ReceiveSecurityHeader ReceivedSecurityHeader
|
||
|
{
|
||
|
get { return this.securityHeader; }
|
||
|
}
|
||
|
|
||
|
Exception CreateBadStateException(string operation)
|
||
|
{
|
||
|
return new InvalidOperationException(SR.GetString(SR.MessageBodyOperationNotValidInBodyState,
|
||
|
operation, this.state));
|
||
|
}
|
||
|
|
||
|
public XmlDictionaryReader CreateFullBodyReader()
|
||
|
{
|
||
|
switch (this.state)
|
||
|
{
|
||
|
case BodyState.Buffered:
|
||
|
return CreateFullBodyReaderFromBufferedState();
|
||
|
case BodyState.Decrypted:
|
||
|
return CreateFullBodyReaderFromDecryptedState();
|
||
|
default:
|
||
|
throw TraceUtility.ThrowHelperError(CreateBadStateException("CreateFullBodyReader"), this);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
XmlDictionaryReader CreateFullBodyReaderFromBufferedState()
|
||
|
{
|
||
|
if (this.messageBuffer != null)
|
||
|
{
|
||
|
XmlDictionaryReader reader = this.messageBuffer.GetReader(0);
|
||
|
MoveToBody(reader);
|
||
|
return reader;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return ((BufferedMessage) this.InnerMessage).GetBufferedReaderAtBody();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
XmlDictionaryReader CreateFullBodyReaderFromDecryptedState()
|
||
|
{
|
||
|
XmlDictionaryReader reader = XmlDictionaryReader.CreateTextReader(this.decryptedBuffer, 0, this.decryptedBuffer.Length, this.securityHeader.ReaderQuotas);
|
||
|
MoveToBody(reader);
|
||
|
return reader;
|
||
|
}
|
||
|
|
||
|
void EnsureDecryptedBodyStatusDetermined()
|
||
|
{
|
||
|
if (!this.isDecryptedBodyStatusDetermined)
|
||
|
{
|
||
|
XmlDictionaryReader reader = CreateFullBodyReader();
|
||
|
if (Message.ReadStartBody(reader, this.InnerMessage.Version.Envelope, out this.isDecryptedBodyFault, out this.isDecryptedBodyEmpty))
|
||
|
{
|
||
|
this.cachedDecryptedBodyContentReader = reader;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
reader.Close();
|
||
|
}
|
||
|
this.isDecryptedBodyStatusDetermined = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public XmlAttributeHolder[] GetEnvelopeAttributes()
|
||
|
{
|
||
|
return this.envelopeAttributes;
|
||
|
}
|
||
|
|
||
|
public XmlAttributeHolder[] GetHeaderAttributes()
|
||
|
{
|
||
|
return this.headerAttributes;
|
||
|
}
|
||
|
|
||
|
XmlDictionaryReader GetReaderAtEnvelope()
|
||
|
{
|
||
|
if (this.messageBuffer != null)
|
||
|
{
|
||
|
return this.messageBuffer.GetReader(0);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return ((BufferedMessage) this.InnerMessage).GetMessageReader();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public XmlDictionaryReader GetReaderAtFirstHeader()
|
||
|
{
|
||
|
XmlDictionaryReader reader = GetReaderAtEnvelope();
|
||
|
MoveToHeaderBlock(reader, false);
|
||
|
reader.ReadStartElement();
|
||
|
return reader;
|
||
|
}
|
||
|
|
||
|
public XmlDictionaryReader GetReaderAtSecurityHeader()
|
||
|
{
|
||
|
if (this.cachedReaderAtSecurityHeader != null)
|
||
|
{
|
||
|
XmlDictionaryReader result = this.cachedReaderAtSecurityHeader;
|
||
|
this.cachedReaderAtSecurityHeader = null;
|
||
|
return result;
|
||
|
}
|
||
|
return this.Headers.GetReaderAtHeader(this.securityHeader.HeaderIndex);
|
||
|
}
|
||
|
|
||
|
void MoveToBody(XmlDictionaryReader reader)
|
||
|
{
|
||
|
if (reader.NodeType != XmlNodeType.Element)
|
||
|
{
|
||
|
reader.MoveToContent();
|
||
|
}
|
||
|
reader.ReadStartElement();
|
||
|
if (reader.IsStartElement(XD.MessageDictionary.Header, this.Version.Envelope.DictionaryNamespace))
|
||
|
{
|
||
|
reader.Skip();
|
||
|
}
|
||
|
if (reader.NodeType != XmlNodeType.Element)
|
||
|
{
|
||
|
reader.MoveToContent();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void MoveToHeaderBlock(XmlDictionaryReader reader, bool captureAttributes)
|
||
|
{
|
||
|
if (reader.NodeType != XmlNodeType.Element)
|
||
|
{
|
||
|
reader.MoveToContent();
|
||
|
}
|
||
|
if (captureAttributes)
|
||
|
{
|
||
|
this.envelopePrefix = reader.Prefix;
|
||
|
this.envelopeAttributes = XmlAttributeHolder.ReadAttributes(reader);
|
||
|
}
|
||
|
reader.ReadStartElement();
|
||
|
reader.MoveToStartElement(XD.MessageDictionary.Header, this.Version.Envelope.DictionaryNamespace);
|
||
|
if (captureAttributes)
|
||
|
{
|
||
|
this.headerAttributes = XmlAttributeHolder.ReadAttributes(reader);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void MoveToSecurityHeader(XmlDictionaryReader reader, int headerIndex, bool captureAttributes)
|
||
|
{
|
||
|
MoveToHeaderBlock(reader, captureAttributes);
|
||
|
reader.ReadStartElement();
|
||
|
while (true)
|
||
|
{
|
||
|
if (reader.NodeType != XmlNodeType.Element)
|
||
|
{
|
||
|
reader.MoveToContent();
|
||
|
}
|
||
|
if (headerIndex == 0)
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
reader.Skip();
|
||
|
headerIndex--;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected override void OnBodyToString(XmlDictionaryWriter writer)
|
||
|
{
|
||
|
if (this.state == BodyState.Created)
|
||
|
{
|
||
|
base.OnBodyToString(writer);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
OnWriteBodyContents(writer);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected override void OnClose()
|
||
|
{
|
||
|
|
||
|
if (this.cachedDecryptedBodyContentReader != null)
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
this.cachedDecryptedBodyContentReader.Close();
|
||
|
}
|
||
|
catch (System.IO.IOException exception)
|
||
|
{
|
||
|
//
|
||
|
// We only want to catch and log the I/O exception here
|
||
|
// assuming reader only throw those exceptions
|
||
|
//
|
||
|
DiagnosticUtility.TraceHandledException(exception, TraceEventType.Warning);
|
||
|
}
|
||
|
finally
|
||
|
{
|
||
|
this.cachedDecryptedBodyContentReader = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (this.cachedReaderAtSecurityHeader != null)
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
this.cachedReaderAtSecurityHeader.Close();
|
||
|
}
|
||
|
catch (System.IO.IOException exception)
|
||
|
{
|
||
|
//
|
||
|
// We only want to catch and log the I/O exception here
|
||
|
// assuming reader only throw those exceptions
|
||
|
//
|
||
|
DiagnosticUtility.TraceHandledException(exception, TraceEventType.Warning);
|
||
|
}
|
||
|
finally
|
||
|
{
|
||
|
this.cachedReaderAtSecurityHeader = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.messageBuffer = null;
|
||
|
this.decryptedBuffer = null;
|
||
|
this.state = BodyState.Disposed;
|
||
|
this.InnerMessage.Close();
|
||
|
}
|
||
|
|
||
|
protected override XmlDictionaryReader OnGetReaderAtBodyContents()
|
||
|
{
|
||
|
if (this.state == BodyState.Created)
|
||
|
{
|
||
|
return this.InnerMessage.GetReaderAtBodyContents();
|
||
|
}
|
||
|
if (this.bodyDecrypted)
|
||
|
{
|
||
|
EnsureDecryptedBodyStatusDetermined();
|
||
|
}
|
||
|
if (this.cachedDecryptedBodyContentReader != null)
|
||
|
{
|
||
|
XmlDictionaryReader result = this.cachedDecryptedBodyContentReader;
|
||
|
this.cachedDecryptedBodyContentReader = null;
|
||
|
return result;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
XmlDictionaryReader reader = CreateFullBodyReader();
|
||
|
reader.ReadStartElement();
|
||
|
reader.MoveToContent();
|
||
|
return reader;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected override MessageBuffer OnCreateBufferedCopy(int maxBufferSize)
|
||
|
{
|
||
|
if (this.canDelegateCreateBufferedCopyToInnerMessage && this.InnerMessage is BufferedMessage)
|
||
|
{
|
||
|
return this.InnerMessage.CreateBufferedCopy(maxBufferSize);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return base.OnCreateBufferedCopy(maxBufferSize);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal void OnMessageProtectionPassComplete(bool atLeastOneHeaderOrBodyEncrypted)
|
||
|
{
|
||
|
this.canDelegateCreateBufferedCopyToInnerMessage = !atLeastOneHeaderOrBodyEncrypted;
|
||
|
}
|
||
|
|
||
|
internal void OnUnencryptedPart(string name, string ns)
|
||
|
{
|
||
|
if (ns == null)
|
||
|
{
|
||
|
throw TraceUtility.ThrowHelperError(new MessageSecurityException(SR.GetString(SR.RequiredMessagePartNotEncrypted, name)), this);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
throw TraceUtility.ThrowHelperError(new MessageSecurityException(SR.GetString(SR.RequiredMessagePartNotEncryptedNs, name, ns)), this);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal void OnUnsignedPart(string name, string ns)
|
||
|
{
|
||
|
if (ns == null)
|
||
|
{
|
||
|
throw TraceUtility.ThrowHelperError(new MessageSecurityException(SR.GetString(SR.RequiredMessagePartNotSigned, name)), this);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
throw TraceUtility.ThrowHelperError(new MessageSecurityException(SR.GetString(SR.RequiredMessagePartNotSignedNs, name, ns)), this);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected override void OnWriteStartBody(XmlDictionaryWriter writer)
|
||
|
{
|
||
|
if (this.state == BodyState.Created)
|
||
|
{
|
||
|
this.InnerMessage.WriteStartBody(writer);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
XmlDictionaryReader reader = CreateFullBodyReader();
|
||
|
reader.MoveToContent();
|
||
|
writer.WriteStartElement(reader.Prefix, reader.LocalName, reader.NamespaceURI);
|
||
|
writer.WriteAttributes(reader, false);
|
||
|
reader.Close();
|
||
|
}
|
||
|
|
||
|
protected override void OnWriteBodyContents(XmlDictionaryWriter writer)
|
||
|
{
|
||
|
if (this.state == BodyState.Created)
|
||
|
{
|
||
|
this.InnerMessage.WriteBodyContents(writer);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
XmlDictionaryReader reader = CreateFullBodyReader();
|
||
|
reader.ReadStartElement();
|
||
|
while (reader.NodeType != XmlNodeType.EndElement)
|
||
|
writer.WriteNode(reader, false);
|
||
|
reader.ReadEndElement();
|
||
|
reader.Close();
|
||
|
}
|
||
|
|
||
|
public void SetBodyPrefixAndAttributes(XmlDictionaryReader bodyReader)
|
||
|
{
|
||
|
this.bodyPrefix = bodyReader.Prefix;
|
||
|
this.bodyAttributes = XmlAttributeHolder.ReadAttributes(bodyReader);
|
||
|
}
|
||
|
|
||
|
public void SetDecryptedBody(byte[] decryptedBodyContent)
|
||
|
{
|
||
|
if (this.state != BodyState.Buffered)
|
||
|
{
|
||
|
throw TraceUtility.ThrowHelperError(CreateBadStateException("SetDecryptedBody"), this);
|
||
|
}
|
||
|
|
||
|
MemoryStream stream = new MemoryStream();
|
||
|
XmlDictionaryWriter writer = XmlDictionaryWriter.CreateTextWriter(stream);
|
||
|
|
||
|
writer.WriteStartElement(this.envelopePrefix, XD.MessageDictionary.Envelope, this.Version.Envelope.DictionaryNamespace);
|
||
|
XmlAttributeHolder.WriteAttributes(this.envelopeAttributes, writer);
|
||
|
|
||
|
writer.WriteStartElement(this.bodyPrefix, XD.MessageDictionary.Body, this.Version.Envelope.DictionaryNamespace);
|
||
|
XmlAttributeHolder.WriteAttributes(this.bodyAttributes, writer);
|
||
|
writer.WriteString(" "); // ensure non-empty element
|
||
|
writer.WriteEndElement();
|
||
|
writer.WriteEndElement();
|
||
|
writer.Flush();
|
||
|
|
||
|
this.decryptedBuffer = ContextImportHelper.SpliceBuffers(decryptedBodyContent, stream.GetBuffer(), (int) stream.Length, 2);
|
||
|
|
||
|
this.bodyDecrypted = true;
|
||
|
this.state = BodyState.Decrypted;
|
||
|
}
|
||
|
|
||
|
enum BodyState
|
||
|
{
|
||
|
Created,
|
||
|
Buffered,
|
||
|
Decrypted,
|
||
|
Disposed,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Adding wrapping tags using a writer is a temporary feature to
|
||
|
// support interop with a partner. Eventually, the serialization
|
||
|
// team will add a feature to XmlUTF8TextReader to directly
|
||
|
// support the addition of outer namespaces before creating a
|
||
|
// Reader. This roundabout way of supporting context-sensitive
|
||
|
// decryption can then be removed.
|
||
|
static class ContextImportHelper
|
||
|
{
|
||
|
internal static XmlDictionaryReader CreateSplicedReader(byte[] decryptedBuffer,
|
||
|
XmlAttributeHolder[] outerContext1, XmlAttributeHolder[] outerContext2, XmlAttributeHolder[] outerContext3, XmlDictionaryReaderQuotas quotas)
|
||
|
{
|
||
|
const string wrapper1 = "x";
|
||
|
const string wrapper2 = "y";
|
||
|
const string wrapper3 = "z";
|
||
|
const int wrappingDepth = 3;
|
||
|
|
||
|
MemoryStream stream = new MemoryStream();
|
||
|
XmlDictionaryWriter writer = XmlDictionaryWriter.CreateTextWriter(stream);
|
||
|
writer.WriteStartElement(wrapper1);
|
||
|
WriteNamespaceDeclarations(outerContext1, writer);
|
||
|
writer.WriteStartElement(wrapper2);
|
||
|
WriteNamespaceDeclarations(outerContext2, writer);
|
||
|
writer.WriteStartElement(wrapper3);
|
||
|
WriteNamespaceDeclarations(outerContext3, writer);
|
||
|
writer.WriteString(" "); // ensure non-empty element
|
||
|
writer.WriteEndElement();
|
||
|
writer.WriteEndElement();
|
||
|
writer.WriteEndElement();
|
||
|
writer.Flush();
|
||
|
|
||
|
byte[] splicedBuffer = SpliceBuffers(decryptedBuffer, stream.GetBuffer(), (int) stream.Length, wrappingDepth);
|
||
|
XmlDictionaryReader reader = XmlDictionaryReader.CreateTextReader(splicedBuffer, quotas);
|
||
|
reader.ReadStartElement(wrapper1);
|
||
|
reader.ReadStartElement(wrapper2);
|
||
|
reader.ReadStartElement(wrapper3);
|
||
|
if (reader.NodeType != XmlNodeType.Element)
|
||
|
{
|
||
|
reader.MoveToContent();
|
||
|
}
|
||
|
return reader;
|
||
|
}
|
||
|
|
||
|
internal static string GetPrefixIfNamespaceDeclaration(string prefix, string localName)
|
||
|
{
|
||
|
if (prefix == "xmlns")
|
||
|
{
|
||
|
return localName;
|
||
|
}
|
||
|
if (prefix.Length == 0 && localName == "xmlns")
|
||
|
{
|
||
|
return string.Empty;
|
||
|
}
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
static bool IsNamespaceDeclaration(string prefix, string localName)
|
||
|
{
|
||
|
return GetPrefixIfNamespaceDeclaration(prefix, localName) != null;
|
||
|
}
|
||
|
|
||
|
internal static byte[] SpliceBuffers(byte[] middle, byte[] wrapper, int wrapperLength, int wrappingDepth)
|
||
|
{
|
||
|
const byte openChar = (byte) '<';
|
||
|
int openCharsFound = 0;
|
||
|
int openCharIndex;
|
||
|
for (openCharIndex = wrapperLength - 1; openCharIndex >= 0; openCharIndex--)
|
||
|
{
|
||
|
if (wrapper[openCharIndex] == openChar)
|
||
|
{
|
||
|
openCharsFound++;
|
||
|
if (openCharsFound == wrappingDepth)
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Fx.Assert(openCharIndex > 0, "");
|
||
|
|
||
|
byte[] splicedBuffer = DiagnosticUtility.Utility.AllocateByteArray(checked(middle.Length + wrapperLength - 1));
|
||
|
int offset = 0;
|
||
|
int count = openCharIndex - 1;
|
||
|
Buffer.BlockCopy(wrapper, 0, splicedBuffer, offset, count);
|
||
|
offset += count;
|
||
|
count = middle.Length;
|
||
|
Buffer.BlockCopy(middle, 0, splicedBuffer, offset, count);
|
||
|
offset += count;
|
||
|
count = wrapperLength - openCharIndex;
|
||
|
Buffer.BlockCopy(wrapper, openCharIndex, splicedBuffer, offset, count);
|
||
|
|
||
|
return splicedBuffer;
|
||
|
}
|
||
|
|
||
|
static void WriteNamespaceDeclarations(XmlAttributeHolder[] attributes, XmlWriter writer)
|
||
|
{
|
||
|
if (attributes != null)
|
||
|
{
|
||
|
for (int i = 0; i < attributes.Length; i++)
|
||
|
{
|
||
|
XmlAttributeHolder a = attributes[i];
|
||
|
if (IsNamespaceDeclaration(a.Prefix, a.LocalName))
|
||
|
{
|
||
|
a.WriteTo(writer);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|