You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			292 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			292 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //-----------------------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //-----------------------------------------------------------------------------
 | |
| 
 | |
| namespace System.IdentityModel.Tokens
 | |
| {
 | |
|     using System.Collections.Generic;
 | |
|     using System.Collections.ObjectModel;
 | |
|     using System.IdentityModel.Claims;
 | |
|     using System.IdentityModel.Selectors;
 | |
|     using System.Xml;
 | |
|     
 | |
|     public class SamlAttribute
 | |
|     {
 | |
|         string name;
 | |
|         string nameSpace;
 | |
|         readonly ImmutableCollection<string> attributeValues = new ImmutableCollection<string>();
 | |
|         string originalIssuer;
 | |
|         string attributeValueXsiType = System.Security.Claims.ClaimValueTypes.String;
 | |
| 
 | |
|         List<Claim> claims;
 | |
|         string claimType;
 | |
|         bool isReadOnly = false;
 | |
| 
 | |
|         public SamlAttribute()
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         public SamlAttribute(string attributeNamespace, string attributeName, IEnumerable<string> attributeValues)
 | |
|         {
 | |
|             if (string.IsNullOrEmpty(attributeName))
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SAMLAttributeNameAttributeRequired));
 | |
| 
 | |
|             if (string.IsNullOrEmpty(attributeNamespace))
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SAMLAttributeNamespaceAttributeRequired));
 | |
| 
 | |
|             if (attributeValues == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("attributeValues");
 | |
| 
 | |
|             this.name = StringUtil.OptimizeString(attributeName);
 | |
|             this.nameSpace = StringUtil.OptimizeString(attributeNamespace);
 | |
|             this.claimType = string.IsNullOrEmpty(this.nameSpace) ? this.name : this.nameSpace + "/" + this.name;
 | |
| 
 | |
|             foreach (string value in attributeValues)
 | |
|             {
 | |
|                 if (value == null)
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SAMLAttributeValueCannotBeNull));
 | |
| 
 | |
|                 this.attributeValues.Add(value);
 | |
|             }
 | |
| 
 | |
|             if (this.attributeValues.Count == 0)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SAMLAttributeShouldHaveOneValue));
 | |
|         }
 | |
| 
 | |
|         public SamlAttribute(Claim claim)
 | |
|         {
 | |
|             if (claim == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("claim");
 | |
| 
 | |
|             if (!(claim.Resource is String))
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SamlAttributeClaimResourceShouldBeAString));
 | |
| 
 | |
|             if (claim.Right != Rights.PossessProperty)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SamlAttributeClaimRightShouldBePossessProperty));
 | |
| 
 | |
| #pragma warning suppress 56506 // claim.CalimType can never be null.
 | |
|             int lastSlashIndex = claim.ClaimType.LastIndexOf('/');
 | |
|             if ((lastSlashIndex == -1) || (lastSlashIndex == 0) || (lastSlashIndex == claim.ClaimType.Length - 1))
 | |
|             {
 | |
|                 this.nameSpace = String.Empty;
 | |
|                 this.name = claim.ClaimType;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 this.nameSpace = StringUtil.OptimizeString(claim.ClaimType.Substring(0, lastSlashIndex));
 | |
|                 this.name = StringUtil.OptimizeString(claim.ClaimType.Substring(lastSlashIndex + 1, claim.ClaimType.Length - (lastSlashIndex + 1)));
 | |
|             }
 | |
|             this.claimType = claim.ClaimType;
 | |
|             this.attributeValues.Add(claim.Resource as string);
 | |
|         }
 | |
| 
 | |
|         public string Name
 | |
|         {
 | |
|             get { return this.name; }
 | |
|             set
 | |
|             {
 | |
|                 if (isReadOnly)
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.ObjectIsReadOnly)));
 | |
| 
 | |
|                 if (string.IsNullOrEmpty(value))
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SAMLAttributeNameAttributeRequired));
 | |
| 
 | |
|                 this.name = StringUtil.OptimizeString(value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public string Namespace
 | |
|         {
 | |
|             get { return this.nameSpace; }
 | |
|             set
 | |
|             {
 | |
|                 if (isReadOnly)
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.ObjectIsReadOnly)));
 | |
| 
 | |
|                 if (string.IsNullOrEmpty(value))
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SAMLAttributeNamespaceAttributeRequired));
 | |
| 
 | |
|                 this.nameSpace = StringUtil.OptimizeString(value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public IList<string> AttributeValues
 | |
|         {
 | |
|             get { return this.attributeValues; }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Gets or Sets the string that represents the OriginalIssuer of the SAML Attribute.
 | |
|         /// </summary>
 | |
|         public string OriginalIssuer
 | |
|         {
 | |
|             get { return this.originalIssuer; }
 | |
|             set
 | |
|             {
 | |
|                 if (value == String.Empty)
 | |
|                 {
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument("value", SR.GetString(SR.ID4251));
 | |
|                 }
 | |
| 
 | |
|                 this.originalIssuer = StringUtil.OptimizeString(value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Gets or sets the xsi:type of the values contained in the SAML Attribute.
 | |
|         /// </summary>
 | |
|         public string AttributeValueXsiType
 | |
|         {
 | |
|             get { return attributeValueXsiType; }
 | |
|             set
 | |
|             {
 | |
|                 if (string.IsNullOrEmpty(value))
 | |
|                 {
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument("value", SR.GetString(SR.ID4254));
 | |
|                 }
 | |
| 
 | |
|                 int indexOfHash = value.IndexOf('#');
 | |
|                 if (indexOfHash == -1)
 | |
|                 {
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument("value", SR.GetString(SR.ID4254));
 | |
|                 }
 | |
| 
 | |
|                 string prefix = value.Substring(0, indexOfHash);
 | |
|                 if (prefix.Length == 0)
 | |
|                 {
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument("value", SR.GetString(SR.ID4254));
 | |
|                 }
 | |
| 
 | |
|                 string suffix = value.Substring(indexOfHash + 1);
 | |
|                 if (suffix.Length == 0)
 | |
|                 {
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument("value", SR.GetString(SR.ID4254));
 | |
|                 }
 | |
| 
 | |
|                 attributeValueXsiType = value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool IsReadOnly
 | |
|         {
 | |
|             get { return this.isReadOnly; }
 | |
|         }
 | |
| 
 | |
|         public void MakeReadOnly()
 | |
|         {
 | |
|             if (!this.isReadOnly)
 | |
|             {
 | |
|                 this.attributeValues.MakeReadOnly();
 | |
| 
 | |
|                 this.isReadOnly = true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual ReadOnlyCollection<Claim> ExtractClaims()
 | |
|         {
 | |
|             if (this.claims == null)
 | |
|             {
 | |
|                 List<Claim> tempClaims = new List<Claim>(this.attributeValues.Count);
 | |
| 
 | |
|                 for (int i = 0; i < this.attributeValues.Count; i++)
 | |
|                 {
 | |
|                     if (this.attributeValues[i] == null)
 | |
|                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SAMLAttributeValueCannotBeNull));
 | |
| 
 | |
|                     tempClaims.Add(new Claim(this.claimType, this.attributeValues[i], Rights.PossessProperty));
 | |
|                 }
 | |
|                 this.claims = tempClaims;
 | |
|             }
 | |
| 
 | |
|             return this.claims.AsReadOnly();
 | |
|         }
 | |
| 
 | |
|         void CheckObjectValidity()
 | |
|         {
 | |
|             if (string.IsNullOrEmpty(this.name))
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new SecurityTokenException(SR.GetString(SR.SAMLAttributeNameAttributeRequired)));
 | |
| 
 | |
|             if (string.IsNullOrEmpty(this.nameSpace))
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new SecurityTokenException(SR.GetString(SR.SAMLAttributeNamespaceAttributeRequired)));
 | |
| 
 | |
|             if (this.attributeValues.Count == 0)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new SecurityTokenException(SR.GetString(SR.SAMLAttributeShouldHaveOneValue)));
 | |
|         }
 | |
| 
 | |
|         public virtual void ReadXml(XmlDictionaryReader reader, SamlSerializer samlSerializer, SecurityTokenSerializer keyInfoSerializer, SecurityTokenResolver outOfBandTokenResolver)
 | |
|         {
 | |
|             if (reader == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("reader"));
 | |
| 
 | |
|             if (samlSerializer == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("samlSerializer"));
 | |
| 
 | |
| #pragma warning suppress 56506 // samlSerializer.DictionaryManager is never null.
 | |
|             SamlDictionary dictionary = samlSerializer.DictionaryManager.SamlDictionary;
 | |
| 
 | |
|             this.name = reader.GetAttribute(dictionary.AttributeName, null);
 | |
|             if (string.IsNullOrEmpty(this.name))
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new SecurityTokenException(SR.GetString(SR.SAMLAttributeMissingNameAttributeOnRead)));
 | |
| 
 | |
|             this.nameSpace = reader.GetAttribute(dictionary.AttributeNamespace, null);
 | |
|             if (string.IsNullOrEmpty(this.nameSpace))
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new SecurityTokenException(SR.GetString(SR.SAMLAttributeMissingNamespaceAttributeOnRead)));
 | |
| 
 | |
|             this.claimType = string.IsNullOrEmpty(this.nameSpace) ? this.name : this.nameSpace + "/" + this.name;
 | |
| 
 | |
|             reader.MoveToContent();
 | |
|             reader.Read();
 | |
|             while (reader.IsStartElement(dictionary.AttributeValue, dictionary.Namespace))
 | |
|             {
 | |
|                 // We will load all Attributes as a string value by default.
 | |
|                 string attrValue = reader.ReadString();
 | |
|                 this.attributeValues.Add(attrValue);
 | |
| 
 | |
|                 reader.MoveToContent();
 | |
|                 reader.ReadEndElement();
 | |
|             }
 | |
| 
 | |
|             if (this.attributeValues.Count == 0)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new SecurityTokenException(SR.GetString(SR.SAMLAttributeShouldHaveOneValue)));
 | |
| 
 | |
|             reader.MoveToContent();
 | |
|             reader.ReadEndElement();
 | |
|         }
 | |
| 
 | |
|         public virtual void WriteXml(XmlDictionaryWriter writer, SamlSerializer samlSerializer, SecurityTokenSerializer keyInfoSerializer)
 | |
|         {
 | |
|             CheckObjectValidity();
 | |
| 
 | |
|             if (writer == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("writer"));
 | |
| 
 | |
|             if (samlSerializer == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("samlSerializer"));
 | |
| 
 | |
| #pragma warning suppress 56506 // samlSerializer.DictionaryManager is never null.
 | |
|             SamlDictionary dictionary = samlSerializer.DictionaryManager.SamlDictionary;
 | |
| 
 | |
|             writer.WriteStartElement(dictionary.PreferredPrefix.Value, dictionary.Attribute, dictionary.Namespace);
 | |
| 
 | |
|             writer.WriteStartAttribute(dictionary.AttributeName, null);
 | |
|             writer.WriteString(this.name);
 | |
|             writer.WriteEndAttribute();
 | |
|             writer.WriteStartAttribute(dictionary.AttributeNamespace, null);
 | |
|             writer.WriteString(this.nameSpace);
 | |
|             writer.WriteEndAttribute();
 | |
| 
 | |
|             for (int i = 0; i < this.attributeValues.Count; i++)
 | |
|             {
 | |
|                 if (this.attributeValues[i] == null)
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SAMLAttributeValueCannotBeNull));
 | |
| 
 | |
|                 writer.WriteElementString(dictionary.PreferredPrefix.Value, dictionary.AttributeValue, dictionary.Namespace, this.attributeValues[i]);
 | |
|             }
 | |
| 
 | |
|             writer.WriteEndElement();
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
| }
 |