You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			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); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | } |