You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			1058 lines
		
	
	
		
			54 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1058 lines
		
	
	
		
			54 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //-----------------------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //-----------------------------------------------------------------------------
 | |
| 
 | |
| namespace System.ServiceModel.Description
 | |
| {
 | |
|     using System.CodeDom;
 | |
|     using System.Collections.Generic;
 | |
|     using System.Net.Security;
 | |
|     using System.Reflection;
 | |
|     using System.Runtime;
 | |
|     using System.Runtime.Serialization;
 | |
|     using System.ServiceModel.Channels;
 | |
|     using System.ServiceModel.Security;
 | |
|     using System.ComponentModel;
 | |
|     using System.Threading.Tasks;
 | |
| 
 | |
|     enum MessageContractType { None, WrappedMessageContract, BareMessageContract }
 | |
|     interface IWrappedBodyTypeGenerator
 | |
|     {
 | |
|         void ValidateForParameterMode(OperationDescription operationDescription);
 | |
|         void AddMemberAttributes(XmlName messageName, MessagePartDescription part, CodeAttributeDeclarationCollection attributesImported, CodeAttributeDeclarationCollection typeAttributes, CodeAttributeDeclarationCollection fieldAttributes);
 | |
|         void AddTypeAttributes(string messageName, string typeNS, CodeAttributeDeclarationCollection typeAttributes, bool isEncoded);
 | |
|     }
 | |
| 
 | |
|     class OperationGenerator //: IOperationBehavior, IOperationContractGenerationExtension
 | |
|     {
 | |
|         Dictionary<MessagePartDescription, CodeTypeReference> parameterTypes;
 | |
|         Dictionary<MessagePartDescription, CodeAttributeDeclarationCollection> parameterAttributes;
 | |
|         Dictionary<MessagePartDescription, string> specialPartName;
 | |
| 
 | |
|         internal OperationGenerator()
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         internal Dictionary<MessagePartDescription, CodeAttributeDeclarationCollection> ParameterAttributes
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (this.parameterAttributes == null)
 | |
|                     this.parameterAttributes = new Dictionary<MessagePartDescription, CodeAttributeDeclarationCollection>();
 | |
|                 return this.parameterAttributes;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal Dictionary<MessagePartDescription, CodeTypeReference> ParameterTypes
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (this.parameterTypes == null)
 | |
|                     this.parameterTypes = new Dictionary<MessagePartDescription, CodeTypeReference>();
 | |
|                 return this.parameterTypes;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal Dictionary<MessagePartDescription, string> SpecialPartName
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (specialPartName == null)
 | |
|                     this.specialPartName = new Dictionary<MessagePartDescription, string>();
 | |
|                 return specialPartName;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal void GenerateOperation(OperationContractGenerationContext context, ref OperationFormatStyle style, bool isEncoded, IWrappedBodyTypeGenerator wrappedBodyTypeGenerator, Dictionary<MessagePartDescription, ICollection<CodeTypeReference>> knownTypes)
 | |
|         {
 | |
|             if (context == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("context"));
 | |
|             if (context.Operation == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.OperationPropertyIsRequiredForAttributeGeneration)));
 | |
| 
 | |
|             MethodSignatureGenerator methodSignatureGenerator = new MethodSignatureGenerator(this, context, style, isEncoded, wrappedBodyTypeGenerator, knownTypes);
 | |
|             methodSignatureGenerator.GenerateSyncSignature(ref style);
 | |
| 
 | |
|             if (context.IsTask)
 | |
|             {
 | |
|                 methodSignatureGenerator.GenerateTaskSignature(ref style);
 | |
|             }
 | |
| 
 | |
|             if (context.IsAsync)
 | |
|             {
 | |
|                 methodSignatureGenerator.GenerateAsyncSignature(ref style);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal static CodeAttributeDeclaration GenerateAttributeDeclaration(ServiceContractGenerator generator, Attribute attribute)
 | |
|         {
 | |
|             return CustomAttributeHelper.GenerateAttributeDeclaration(generator, attribute);
 | |
|         }
 | |
| 
 | |
|         class MethodSignatureGenerator
 | |
|         {
 | |
|             readonly OperationGenerator Parent;
 | |
|             readonly OperationContractGenerationContext Context;
 | |
|             readonly OperationFormatStyle Style;
 | |
|             readonly bool IsEncoded;
 | |
|             readonly IWrappedBodyTypeGenerator WrappedBodyTypeGenerator;
 | |
|             readonly Dictionary<MessagePartDescription, ICollection<CodeTypeReference>> KnownTypes;
 | |
| 
 | |
|             CodeMemberMethod Method;
 | |
|             CodeMemberMethod EndMethod;
 | |
| 
 | |
|             readonly string ContractName;
 | |
|             string DefaultName;
 | |
|             readonly string ContractNS;
 | |
|             readonly string DefaultNS;
 | |
| 
 | |
|             readonly bool Oneway;
 | |
| 
 | |
|             readonly MessageDescription Request;
 | |
|             readonly MessageDescription Response;
 | |
| 
 | |
|             bool IsNewRequest;
 | |
|             bool IsNewResponse;
 | |
|             bool IsTaskWithOutputParameters;
 | |
|             MessageContractType MessageContractType;
 | |
| 
 | |
|             IPartCodeGenerator BeginPartCodeGenerator;
 | |
|             IPartCodeGenerator EndPartCodeGenerator;
 | |
| 
 | |
|             internal MethodSignatureGenerator(OperationGenerator parent, OperationContractGenerationContext context, OperationFormatStyle style, bool isEncoded, IWrappedBodyTypeGenerator wrappedBodyTypeGenerator, Dictionary<MessagePartDescription, ICollection<CodeTypeReference>> knownTypes)
 | |
|             {
 | |
|                 this.Parent = parent;
 | |
|                 this.Context = context;
 | |
|                 this.Style = style;
 | |
|                 this.IsEncoded = isEncoded;
 | |
|                 this.WrappedBodyTypeGenerator = wrappedBodyTypeGenerator;
 | |
|                 this.KnownTypes = knownTypes;
 | |
|                 this.MessageContractType = context.ServiceContractGenerator.OptionsInternal.IsSet(ServiceContractGenerationOptions.TypedMessages) ? MessageContractType.WrappedMessageContract : MessageContractType.None;
 | |
| 
 | |
|                 this.ContractName = context.Contract.Contract.CodeName;
 | |
|                 this.ContractNS = context.Operation.DeclaringContract.Namespace;
 | |
|                 this.DefaultNS = (style == OperationFormatStyle.Rpc) ? string.Empty : this.ContractNS;
 | |
|                 this.Oneway = (context.Operation.IsOneWay);
 | |
|                 this.Request = context.Operation.Messages[0];
 | |
|                 this.Response = this.Oneway ? null : context.Operation.Messages[1];
 | |
| 
 | |
|                 this.IsNewRequest = true;
 | |
|                 this.IsNewResponse = true;
 | |
|                 this.BeginPartCodeGenerator = null;
 | |
|                 this.EndPartCodeGenerator = null;
 | |
|                 this.IsTaskWithOutputParameters = context.IsTask && context.Operation.HasOutputParameters;
 | |
| 
 | |
|                 Fx.Assert(this.Oneway == (this.Response == null), "OperationContractGenerationContext.Operation cannot contain a null response message when the operation is not one-way");
 | |
|             }
 | |
| 
 | |
|             internal void GenerateSyncSignature(ref OperationFormatStyle style)
 | |
|             {
 | |
|                 this.Method = this.Context.SyncMethod;
 | |
|                 this.EndMethod = this.Context.SyncMethod;
 | |
|                 this.DefaultName = this.Method.Name;
 | |
|                 GenerateOperationSignatures(ref style);
 | |
|             }
 | |
| 
 | |
|             internal void GenerateAsyncSignature(ref OperationFormatStyle style)
 | |
|             {
 | |
|                 this.Method = this.Context.BeginMethod;
 | |
|                 this.EndMethod = this.Context.EndMethod;
 | |
|                 this.DefaultName = this.Method.Name.Substring(5);
 | |
|                 GenerateOperationSignatures(ref style);
 | |
|             }
 | |
| 
 | |
|             void GenerateOperationSignatures(ref OperationFormatStyle style)
 | |
|             {
 | |
|                 if (this.MessageContractType != MessageContractType.None || this.GenerateTypedMessageForTaskWithOutputParameters())
 | |
|                 {
 | |
|                     CheckAndSetMessageContractTypeToBare();
 | |
|                     this.GenerateTypedMessageOperation(false /*hideFromEditor*/, ref style);
 | |
|                 }
 | |
|                 else if (!this.TryGenerateParameterizedOperation())
 | |
|                 {
 | |
|                     this.GenerateTypedMessageOperation(true /*hideFromEditor*/, ref style);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             bool GenerateTypedMessageForTaskWithOutputParameters()
 | |
|             {
 | |
|                 if (this.IsTaskWithOutputParameters)
 | |
|                 {
 | |
|                     if (this.Method == this.Context.TaskMethod)
 | |
|                     {
 | |
|                         this.Method.Comments.Add(new CodeCommentStatement(SR.GetString(SR.SFxCodeGenWarning, SR.GetString(SR.SFxCannotImportAsParameters_OutputParameterAndTask))));
 | |
|                     }
 | |
|                     
 | |
|                     return true;
 | |
|                 }
 | |
| 
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             void CheckAndSetMessageContractTypeToBare()
 | |
|             {
 | |
|                 if (this.MessageContractType == MessageContractType.BareMessageContract)
 | |
|                     return;
 | |
|                 try
 | |
|                 {
 | |
|                     this.WrappedBodyTypeGenerator.ValidateForParameterMode(this.Context.Operation);
 | |
|                 }
 | |
|                 catch (ParameterModeException ex)
 | |
|                 {
 | |
|                     this.MessageContractType = ex.MessageContractType;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             bool TryGenerateParameterizedOperation()
 | |
|             {
 | |
|                 CodeParameterDeclarationExpressionCollection methodParameters, endMethodParameters = null;
 | |
|                 methodParameters = new CodeParameterDeclarationExpressionCollection(this.Method.Parameters);
 | |
|                 if (this.EndMethod != null)
 | |
|                     endMethodParameters = new CodeParameterDeclarationExpressionCollection(this.EndMethod.Parameters);
 | |
| 
 | |
|                 try
 | |
|                 {
 | |
|                     GenerateParameterizedOperation();
 | |
|                 }
 | |
|                 catch (ParameterModeException ex)
 | |
|                 {
 | |
|                     this.MessageContractType = ex.MessageContractType;
 | |
|                     CodeMemberMethod method = this.Method;
 | |
|                     method.Comments.Add(new CodeCommentStatement(SR.GetString(SR.SFxCodeGenWarning, ex.Message)));
 | |
|                     method.Parameters.Clear();
 | |
|                     method.Parameters.AddRange(methodParameters);
 | |
|                     if (this.Context.IsAsync)
 | |
|                     {
 | |
|                         CodeMemberMethod endMethod = this.EndMethod;
 | |
|                         endMethod.Parameters.Clear();
 | |
|                         endMethod.Parameters.AddRange(endMethodParameters);
 | |
|                     }
 | |
|                     return false;
 | |
|                 }
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             void GenerateParameterizedOperation()
 | |
|             {
 | |
| 
 | |
|                 ParameterizedMessageHelper.ValidateProtectionLevel(this);
 | |
|                 CreateOrOverrideActionProperties();
 | |
| 
 | |
|                 if (this.HasUntypedMessages)
 | |
|                 {
 | |
|                     if (!this.IsCompletelyUntyped)
 | |
|                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_Message, this.Context.Operation.CodeName)));
 | |
| 
 | |
|                     CreateUntypedMessages();
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
| 
 | |
|                     ParameterizedMessageHelper.ValidateWrapperSettings(this);
 | |
|                     ParameterizedMessageHelper.ValidateNoHeaders(this);
 | |
|                     this.WrappedBodyTypeGenerator.ValidateForParameterMode(this.Context.Operation);
 | |
| 
 | |
|                     ParameterizedMethodGenerator generator = new ParameterizedMethodGenerator(this.Method, this.EndMethod);
 | |
|                     this.BeginPartCodeGenerator = generator.InputGenerator;
 | |
|                     this.EndPartCodeGenerator = generator.OutputGenerator;
 | |
| 
 | |
|                     if (!this.Oneway && this.Response.Body.ReturnValue != null)
 | |
|                     {
 | |
|                         this.EndMethod.ReturnType = GetParameterType(this.Response.Body.ReturnValue);
 | |
|                         ParameterizedMessageHelper.GenerateMessageParameterAttribute(this.Response.Body.ReturnValue, this.EndMethod.ReturnTypeCustomAttributes, TypeLoader.GetReturnValueName(this.DefaultName), this.DefaultNS);
 | |
|                         AddAdditionalAttributes(this.Response.Body.ReturnValue, this.EndMethod.ReturnTypeCustomAttributes, this.IsEncoded);
 | |
|                     }
 | |
| 
 | |
|                     GenerateMessageBodyParts(false /*generateTypedMessages*/);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void GenerateTypedMessageOperation(bool hideFromEditor, ref OperationFormatStyle style)
 | |
|             {
 | |
|                 CreateOrOverrideActionProperties();
 | |
| 
 | |
|                 if (this.HasUntypedMessages)
 | |
|                 {
 | |
|                     CreateUntypedMessages();
 | |
|                     if (this.IsCompletelyUntyped)
 | |
|                         return;
 | |
|                 }
 | |
| 
 | |
|                 CodeNamespace ns = this.Context.ServiceContractGenerator.NamespaceManager.EnsureNamespace(this.ContractNS);
 | |
| 
 | |
|                 if (!this.Request.IsUntypedMessage)
 | |
|                 {
 | |
|                     CodeTypeReference typedReqMessageRef = GenerateTypedMessageHeaderAndReturnValueParts(ns, this.DefaultName + "Request", this.Request, false /*isReply*/, hideFromEditor, ref this.IsNewRequest, out this.BeginPartCodeGenerator);
 | |
|                     this.Method.Parameters.Insert(0, new CodeParameterDeclarationExpression(typedReqMessageRef, "request"));
 | |
|                 }
 | |
| 
 | |
|                 if (!this.Oneway && !this.Response.IsUntypedMessage)
 | |
|                 {
 | |
|                     CodeTypeReference typedRespMessageRef = GenerateTypedMessageHeaderAndReturnValueParts(ns, this.DefaultName + "Response", this.Response, true /*isReply*/, hideFromEditor, ref this.IsNewResponse, out this.EndPartCodeGenerator);
 | |
|                     this.EndMethod.ReturnType = typedRespMessageRef;
 | |
|                 }
 | |
| 
 | |
|                 GenerateMessageBodyParts(true /*generateTypedMessages*/);
 | |
| 
 | |
|                 if (!this.IsEncoded)
 | |
|                     style = OperationFormatStyle.Document;
 | |
|             }
 | |
| 
 | |
|             CodeTypeReference GenerateTypedMessageHeaderAndReturnValueParts(CodeNamespace ns, string defaultName, MessageDescription message, bool isReply, bool hideFromEditor, ref bool isNewMessage, out IPartCodeGenerator partCodeGenerator)
 | |
|             {
 | |
|                 CodeTypeReference typedMessageRef;
 | |
|                 if (TypedMessageHelper.FindGeneratedTypedMessage(this.Context.Contract, message, out typedMessageRef))
 | |
|                 {
 | |
|                     partCodeGenerator = null;
 | |
|                     isNewMessage = false;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     UniqueCodeNamespaceScope namespaceScope = new UniqueCodeNamespaceScope(ns);
 | |
| 
 | |
|                     CodeTypeDeclaration typedMessageDecl = Context.Contract.TypeFactory.CreateClassType();
 | |
|                     string messageName = XmlName.IsNullOrEmpty(message.MessageName) ? null : message.MessageName.DecodedName;
 | |
|                     typedMessageRef = namespaceScope.AddUnique(typedMessageDecl, messageName, defaultName);
 | |
| 
 | |
|                     TypedMessageHelper.AddGeneratedTypedMessage(this.Context.Contract, message, typedMessageRef);
 | |
| 
 | |
|                     if (this.MessageContractType == MessageContractType.BareMessageContract && message.Body.WrapperName != null)
 | |
|                         WrapTypedMessage(ns, typedMessageDecl.Name, message, isReply, this.Context.IsInherited, hideFromEditor);
 | |
| 
 | |
|                     partCodeGenerator = new TypedMessagePartCodeGenerator(typedMessageDecl);
 | |
| 
 | |
|                     if (hideFromEditor)
 | |
|                     {
 | |
|                         TypedMessageHelper.AddEditorBrowsableAttribute(typedMessageDecl.CustomAttributes);
 | |
|                     }
 | |
|                     TypedMessageHelper.GenerateWrapperAttribute(message, partCodeGenerator);
 | |
|                     TypedMessageHelper.GenerateProtectionLevelAttribute(message, partCodeGenerator);
 | |
| 
 | |
|                     foreach (MessageHeaderDescription setting in message.Headers)
 | |
|                         GenerateHeaderPart(setting, partCodeGenerator);
 | |
| 
 | |
|                     if (isReply && message.Body.ReturnValue != null)
 | |
|                     {
 | |
|                         GenerateBodyPart(0, message.Body.ReturnValue, partCodeGenerator, true, this.IsEncoded, this.DefaultNS);
 | |
|                     }
 | |
|                 }
 | |
|                 return typedMessageRef;
 | |
|             }
 | |
| 
 | |
| 
 | |
|             bool IsCompletelyUntyped
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     bool isRequestMessage = this.Request != null && this.Request.IsUntypedMessage;
 | |
|                     bool isResponseMessage = this.Response != null && this.Response.IsUntypedMessage;
 | |
| 
 | |
|                     if (isRequestMessage && isResponseMessage)
 | |
|                         return true;
 | |
|                     else if (isResponseMessage && Request == null || IsEmpty(Request))
 | |
|                         return true;
 | |
|                     else if (isRequestMessage && Response == null || IsEmpty(Response))
 | |
|                         return true;
 | |
|                     else
 | |
|                         return false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             bool IsEmpty(MessageDescription message)
 | |
|             {
 | |
|                 return (message.Body.Parts.Count == 0 && message.Headers.Count == 0);
 | |
|             }
 | |
| 
 | |
|             bool HasUntypedMessages
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     bool isRequestMessage = this.Request != null && this.Request.IsUntypedMessage;
 | |
|                     bool isResponseMessage = this.Response != null && this.Response.IsUntypedMessage;
 | |
|                     return (isRequestMessage || isResponseMessage);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void CreateUntypedMessages()
 | |
|             {
 | |
|                 bool isRequestMessage = this.Request != null && this.Request.IsUntypedMessage;
 | |
|                 bool isResponseMessage = this.Response != null && this.Response.IsUntypedMessage;
 | |
| 
 | |
|                 if (isRequestMessage)
 | |
|                     this.Method.Parameters.Insert(0, new CodeParameterDeclarationExpression(Context.ServiceContractGenerator.GetCodeTypeReference((typeof(Message))), "request"));
 | |
|                 if (isResponseMessage)
 | |
|                     this.EndMethod.ReturnType = Context.ServiceContractGenerator.GetCodeTypeReference(typeof(Message));
 | |
|             }
 | |
| 
 | |
|             void CreateOrOverrideActionProperties()
 | |
|             {
 | |
|                 if (this.Request != null)
 | |
|                 {
 | |
|                     CustomAttributeHelper.CreateOrOverridePropertyDeclaration(
 | |
|                         CustomAttributeHelper.FindOrCreateAttributeDeclaration<OperationContractAttribute>(this.Method.CustomAttributes), OperationContractAttribute.ActionPropertyName, this.Request.Action);
 | |
|                 }
 | |
|                 if (this.Response != null)
 | |
|                 {
 | |
|                     CustomAttributeHelper.CreateOrOverridePropertyDeclaration(
 | |
|                         CustomAttributeHelper.FindOrCreateAttributeDeclaration<OperationContractAttribute>(this.Method.CustomAttributes), OperationContractAttribute.ReplyActionPropertyName, this.Response.Action);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             interface IPartCodeGenerator
 | |
|             {
 | |
|                 CodeAttributeDeclarationCollection AddPart(CodeTypeReference type, ref string name);
 | |
|                 CodeAttributeDeclarationCollection MessageLevelAttributes { get; }
 | |
|                 void EndCodeGeneration();
 | |
|             }
 | |
| 
 | |
|             class ParameterizedMethodGenerator
 | |
|             {
 | |
|                 ParametersPartCodeGenerator ins;
 | |
|                 ParametersPartCodeGenerator outs;
 | |
|                 bool isSync;
 | |
| 
 | |
|                 internal ParameterizedMethodGenerator(CodeMemberMethod beginMethod, CodeMemberMethod endMethod)
 | |
|                 {
 | |
|                     this.ins = new ParametersPartCodeGenerator(this, beginMethod.Name, beginMethod.Parameters, beginMethod.CustomAttributes, FieldDirection.In);
 | |
|                     this.outs = new ParametersPartCodeGenerator(this, beginMethod.Name, endMethod.Parameters, beginMethod.CustomAttributes, FieldDirection.Out);
 | |
|                     this.isSync = (beginMethod == endMethod);
 | |
|                 }
 | |
| 
 | |
|                 internal CodeParameterDeclarationExpression GetOrCreateParameter(CodeTypeReference type, string name, FieldDirection direction, ref int index, out bool createdNew)
 | |
|                 {
 | |
|                     Fx.Assert(System.CodeDom.Compiler.CodeGenerator.IsValidLanguageIndependentIdentifier(name), String.Format(System.Globalization.CultureInfo.InvariantCulture, "Type name '{0}' is not ValidLanguageIndependentIdentifier!", name));
 | |
|                     ParametersPartCodeGenerator existingParams = direction != FieldDirection.In ? ins : outs;
 | |
|                     int i = index;
 | |
|                     CodeParameterDeclarationExpression existing = existingParams.GetParameter(name, ref i);
 | |
|                     bool isRef = existing != null && existing.Type.BaseType == type.BaseType;
 | |
|                     if (isRef)
 | |
|                     {
 | |
|                         existing.Direction = FieldDirection.Ref;
 | |
|                         if (isSync)
 | |
|                         {
 | |
|                             index = i + 1;
 | |
|                             createdNew = false;
 | |
|                             return existing;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     CodeParameterDeclarationExpression paramDecl = new CodeParameterDeclarationExpression();
 | |
|                     paramDecl.Name = name;
 | |
|                     paramDecl.Type = type;
 | |
|                     paramDecl.Direction = direction;
 | |
|                     if (isRef)
 | |
|                         paramDecl.Direction = FieldDirection.Ref;
 | |
| 
 | |
|                     createdNew = true;
 | |
| 
 | |
|                     return paramDecl;
 | |
|                 }
 | |
| 
 | |
|                 internal IPartCodeGenerator InputGenerator
 | |
|                 {
 | |
|                     get
 | |
|                     {
 | |
|                         return this.ins;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 internal IPartCodeGenerator OutputGenerator
 | |
|                 {
 | |
|                     get
 | |
|                     {
 | |
|                         return this.outs;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 class ParametersPartCodeGenerator : IPartCodeGenerator
 | |
|                 {
 | |
|                     ParameterizedMethodGenerator parent;
 | |
|                     FieldDirection direction;
 | |
|                     CodeParameterDeclarationExpressionCollection parameters;
 | |
|                     CodeAttributeDeclarationCollection messageAttrs;
 | |
|                     string methodName;
 | |
|                     int index;
 | |
| 
 | |
|                     internal ParametersPartCodeGenerator(ParameterizedMethodGenerator parent, string methodName, CodeParameterDeclarationExpressionCollection parameters, CodeAttributeDeclarationCollection messageAttrs, FieldDirection direction)
 | |
|                     {
 | |
|                         this.parent = parent;
 | |
|                         this.methodName = methodName;
 | |
|                         this.parameters = parameters;
 | |
|                         this.messageAttrs = messageAttrs;
 | |
|                         this.direction = direction;
 | |
|                         this.index = 0;
 | |
|                     }
 | |
| 
 | |
|                     public bool NameExists(string name)
 | |
|                     {
 | |
|                         if (String.Compare(name, methodName, StringComparison.OrdinalIgnoreCase) == 0)
 | |
|                             return true;
 | |
|                         int index = 0;
 | |
|                         return GetParameter(name, ref index) != null;
 | |
|                     }
 | |
| 
 | |
|                     CodeAttributeDeclarationCollection IPartCodeGenerator.AddPart(CodeTypeReference type, ref string name)
 | |
|                     {
 | |
|                         bool createdNew;
 | |
|                         name = UniqueCodeIdentifierScope.MakeValid(name, "param");
 | |
|                         CodeParameterDeclarationExpression paramDecl = parent.GetOrCreateParameter(type, name, this.direction, ref index, out createdNew);
 | |
|                         if (createdNew)
 | |
|                         {
 | |
|                             paramDecl.Name = GetUniqueParameterName(paramDecl.Name, this);
 | |
|                             parameters.Insert(this.index++, paramDecl);
 | |
|                         }
 | |
| 
 | |
|                         name = paramDecl.Name;
 | |
|                         if (!createdNew)
 | |
|                             return null;
 | |
|                         return paramDecl.CustomAttributes;
 | |
|                     }
 | |
| 
 | |
| 
 | |
|                     internal CodeParameterDeclarationExpression GetParameter(string name, ref int index)
 | |
|                     {
 | |
|                         for (int i = index; i < parameters.Count; i++)
 | |
|                         {
 | |
|                             CodeParameterDeclarationExpression parameter = parameters[i];
 | |
|                             if (String.Compare(parameter.Name, name, StringComparison.OrdinalIgnoreCase) == 0)
 | |
|                             {
 | |
|                                 index = i;
 | |
|                                 return parameter;
 | |
|                             }
 | |
|                         }
 | |
| 
 | |
|                         return null;
 | |
|                     }
 | |
| 
 | |
|                     CodeAttributeDeclarationCollection IPartCodeGenerator.MessageLevelAttributes
 | |
|                     {
 | |
|                         get
 | |
|                         {
 | |
|                             return messageAttrs;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     void IPartCodeGenerator.EndCodeGeneration()
 | |
|                     {
 | |
|                     }
 | |
| 
 | |
|                     static string GetUniqueParameterName(string name, ParametersPartCodeGenerator parameters)
 | |
|                     {
 | |
|                         return NamingHelper.GetUniqueName(name, DoesParameterNameExist, parameters);
 | |
|                     }
 | |
|                     static bool DoesParameterNameExist(string name, object parametersObject)
 | |
|                     {
 | |
|                         return ((ParametersPartCodeGenerator)parametersObject).NameExists(name);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|             }
 | |
| 
 | |
| 
 | |
|             class TypedMessagePartCodeGenerator : IPartCodeGenerator
 | |
|             {
 | |
|                 CodeTypeDeclaration typeDecl;
 | |
|                 UniqueCodeIdentifierScope memberScope;
 | |
| 
 | |
|                 internal TypedMessagePartCodeGenerator(CodeTypeDeclaration typeDecl)
 | |
|                 {
 | |
|                     this.typeDecl = typeDecl;
 | |
|                     this.memberScope = new UniqueCodeIdentifierScope();
 | |
|                     this.memberScope.AddReserved(typeDecl.Name);
 | |
|                 }
 | |
| 
 | |
|                 CodeAttributeDeclarationCollection IPartCodeGenerator.AddPart(CodeTypeReference type, ref string name)
 | |
|                 {
 | |
|                     CodeMemberField memberDecl = new CodeMemberField();
 | |
|                     memberDecl.Name = name = this.memberScope.AddUnique(name, "member");
 | |
|                     memberDecl.Type = type;
 | |
|                     memberDecl.Attributes = MemberAttributes.Public;
 | |
|                     typeDecl.Members.Add(memberDecl);
 | |
|                     return memberDecl.CustomAttributes;
 | |
|                 }
 | |
| 
 | |
|                 CodeAttributeDeclarationCollection IPartCodeGenerator.MessageLevelAttributes
 | |
|                 {
 | |
|                     get
 | |
|                     {
 | |
|                         return typeDecl.CustomAttributes;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 void IPartCodeGenerator.EndCodeGeneration()
 | |
|                 {
 | |
|                     TypedMessageHelper.GenerateConstructors(typeDecl);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void WrapTypedMessage(CodeNamespace ns, string typeName, MessageDescription messageDescription, bool isReply, bool isInherited, bool hideFromEditor)
 | |
|             {
 | |
|                 Fx.Assert(String.IsNullOrEmpty(typeName) || System.CodeDom.Compiler.CodeGenerator.IsValidLanguageIndependentIdentifier(typeName), String.Format(System.Globalization.CultureInfo.InvariantCulture, "Type name '{0}' is not ValidLanguageIndependentIdentifier!", typeName));
 | |
|                 UniqueCodeNamespaceScope namespaceScope = new UniqueCodeNamespaceScope(ns);
 | |
|                 CodeTypeDeclaration wrapperTypeDecl = Context.Contract.TypeFactory.CreateClassType();
 | |
|                 CodeTypeReference wrapperTypeRef = namespaceScope.AddUnique(wrapperTypeDecl, typeName + "Body", "Body");
 | |
| 
 | |
|                 if (hideFromEditor)
 | |
|                 {
 | |
|                     TypedMessageHelper.AddEditorBrowsableAttribute(wrapperTypeDecl.CustomAttributes);
 | |
|                 }
 | |
| 
 | |
|                 string defaultNS = GetWrapperNamespace(messageDescription);
 | |
|                 string messageName = XmlName.IsNullOrEmpty(messageDescription.MessageName) ? null : messageDescription.MessageName.DecodedName;
 | |
|                 this.WrappedBodyTypeGenerator.AddTypeAttributes(messageName, defaultNS, wrapperTypeDecl.CustomAttributes, this.IsEncoded);
 | |
| 
 | |
|                 IPartCodeGenerator partGenerator = new TypedMessagePartCodeGenerator(wrapperTypeDecl);
 | |
|                 System.Net.Security.ProtectionLevel protectionLevel = System.Net.Security.ProtectionLevel.None;
 | |
|                 bool isProtectionLevelSetExplicitly = false;
 | |
| 
 | |
|                 if (messageDescription.Body.ReturnValue != null)
 | |
|                 {
 | |
|                     AddWrapperPart(messageDescription.MessageName, this.WrappedBodyTypeGenerator, partGenerator, messageDescription.Body.ReturnValue, wrapperTypeDecl.CustomAttributes);
 | |
|                     protectionLevel = ProtectionLevelHelper.Max(protectionLevel, messageDescription.Body.ReturnValue.ProtectionLevel);
 | |
|                     if (messageDescription.Body.ReturnValue.HasProtectionLevel)
 | |
|                         isProtectionLevelSetExplicitly = true;
 | |
|                 }
 | |
| 
 | |
|                 List<CodeTypeReference> wrapperKnownTypes = new List<CodeTypeReference>();
 | |
|                 foreach (MessagePartDescription part in messageDescription.Body.Parts)
 | |
|                 {
 | |
|                     AddWrapperPart(messageDescription.MessageName, this.WrappedBodyTypeGenerator, partGenerator, part, wrapperTypeDecl.CustomAttributes);
 | |
|                     protectionLevel = ProtectionLevelHelper.Max(protectionLevel, part.ProtectionLevel);
 | |
|                     if (part.HasProtectionLevel)
 | |
|                         isProtectionLevelSetExplicitly = true;
 | |
| 
 | |
|                     ICollection<CodeTypeReference> knownTypesForPart = null;
 | |
|                     if (this.KnownTypes != null && this.KnownTypes.TryGetValue(part, out knownTypesForPart))
 | |
|                     {
 | |
|                         foreach (CodeTypeReference typeReference in knownTypesForPart)
 | |
|                             wrapperKnownTypes.Add(typeReference);
 | |
|                     }
 | |
|                 }
 | |
|                 messageDescription.Body.Parts.Clear();
 | |
| 
 | |
|                 MessagePartDescription wrapperPart = new MessagePartDescription(messageDescription.Body.WrapperName, messageDescription.Body.WrapperNamespace);
 | |
|                 if (this.KnownTypes != null)
 | |
|                     this.KnownTypes.Add(wrapperPart, wrapperKnownTypes);
 | |
|                 if (isProtectionLevelSetExplicitly)
 | |
|                     wrapperPart.ProtectionLevel = protectionLevel;
 | |
|                 messageDescription.Body.WrapperName = null;
 | |
|                 messageDescription.Body.WrapperNamespace = null;
 | |
|                 if (isReply)
 | |
|                     messageDescription.Body.ReturnValue = wrapperPart;
 | |
|                 else
 | |
|                     messageDescription.Body.Parts.Add(wrapperPart);
 | |
|                 TypedMessageHelper.GenerateConstructors(wrapperTypeDecl);
 | |
|                 this.Parent.ParameterTypes.Add(wrapperPart, wrapperTypeRef);
 | |
|                 this.Parent.SpecialPartName.Add(wrapperPart, "Body");
 | |
| 
 | |
|             }
 | |
| 
 | |
|             string GetWrapperNamespace(MessageDescription messageDescription)
 | |
|             {
 | |
|                 string defaultNS = this.DefaultNS;
 | |
|                 if (messageDescription.Body.ReturnValue != null)
 | |
|                     defaultNS = messageDescription.Body.ReturnValue.Namespace;
 | |
|                 else if (messageDescription.Body.Parts.Count > 0)
 | |
|                     defaultNS = messageDescription.Body.Parts[0].Namespace;
 | |
|                 return defaultNS;
 | |
|             }
 | |
| 
 | |
|             void GenerateMessageBodyParts(bool generateTypedMessages)
 | |
|             {
 | |
|                 int order = 0;
 | |
|                 if (this.IsNewRequest)
 | |
|                 {
 | |
|                     foreach (MessagePartDescription setting in this.Request.Body.Parts)
 | |
|                         GenerateBodyPart(order++, setting, this.BeginPartCodeGenerator, generateTypedMessages, this.IsEncoded, this.DefaultNS);
 | |
|                 }
 | |
| 
 | |
|                 if (!this.Oneway && IsNewResponse)
 | |
|                 {
 | |
|                     order = this.Response.Body.ReturnValue != null ? 1 : 0;
 | |
|                     foreach (MessagePartDescription setting in this.Response.Body.Parts)
 | |
|                         GenerateBodyPart(order++, setting, this.EndPartCodeGenerator, generateTypedMessages, this.IsEncoded, this.DefaultNS);
 | |
|                 }
 | |
|                 if (IsNewRequest)
 | |
|                 {
 | |
|                     if (this.BeginPartCodeGenerator != null)
 | |
|                         this.BeginPartCodeGenerator.EndCodeGeneration();
 | |
|                 }
 | |
|                 if (IsNewResponse)
 | |
|                 {
 | |
|                     if (EndPartCodeGenerator != null)
 | |
|                         this.EndPartCodeGenerator.EndCodeGeneration();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void AddWrapperPart(XmlName messageName, IWrappedBodyTypeGenerator wrappedBodyTypeGenerator, IPartCodeGenerator partGenerator, MessagePartDescription part, CodeAttributeDeclarationCollection typeAttributes)
 | |
|             {
 | |
|                 string fieldName = part.CodeName;
 | |
|                 CodeTypeReference type;
 | |
|                 if (part.Type == typeof(System.IO.Stream))
 | |
|                     type = Context.ServiceContractGenerator.GetCodeTypeReference(typeof(byte[]));
 | |
|                 else
 | |
|                     type = GetParameterType(part);
 | |
|                 CodeAttributeDeclarationCollection fieldAttributes = partGenerator.AddPart(type, ref fieldName);
 | |
| 
 | |
|                 CodeAttributeDeclarationCollection importedAttributes = null;
 | |
| 
 | |
|                 bool hasAttributes = this.Parent.ParameterAttributes.TryGetValue(part, out importedAttributes);
 | |
| 
 | |
|                 wrappedBodyTypeGenerator.AddMemberAttributes(messageName, part, importedAttributes, typeAttributes, fieldAttributes);
 | |
|                 this.Parent.ParameterTypes.Remove(part);
 | |
|                 if (hasAttributes)
 | |
|                     this.Parent.ParameterAttributes.Remove(part);
 | |
|             }
 | |
| 
 | |
|             void GenerateBodyPart(int order, MessagePartDescription messagePart, IPartCodeGenerator partCodeGenerator, bool generateTypedMessage, bool isEncoded, string defaultNS)
 | |
|             {
 | |
|                 if (!generateTypedMessage) order = -1;
 | |
| 
 | |
|                 string partName;
 | |
|                 if (!this.Parent.SpecialPartName.TryGetValue(messagePart, out partName))
 | |
|                     partName = messagePart.CodeName;
 | |
| 
 | |
|                 CodeTypeReference partType = GetParameterType(messagePart);
 | |
|                 CodeAttributeDeclarationCollection partAttributes = partCodeGenerator.AddPart(partType, ref partName);
 | |
| 
 | |
|                 if (partAttributes == null)
 | |
|                     return;
 | |
| 
 | |
|                 XmlName xmlPartName = new XmlName(partName);
 | |
|                 if (generateTypedMessage)
 | |
|                     TypedMessageHelper.GenerateMessageBodyMemberAttribute(order, messagePart, partAttributes, xmlPartName);
 | |
|                 else
 | |
|                     ParameterizedMessageHelper.GenerateMessageParameterAttribute(messagePart, partAttributes, xmlPartName, defaultNS);
 | |
| 
 | |
|                 AddAdditionalAttributes(messagePart, partAttributes, generateTypedMessage || isEncoded);
 | |
|             }
 | |
| 
 | |
|             void GenerateHeaderPart(MessageHeaderDescription setting, IPartCodeGenerator parts)
 | |
|             {
 | |
|                 string partName;
 | |
|                 if (!this.Parent.SpecialPartName.TryGetValue(setting, out partName))
 | |
|                     partName = setting.CodeName;
 | |
|                 CodeTypeReference partType = GetParameterType(setting);
 | |
|                 CodeAttributeDeclarationCollection partAttributes = parts.AddPart(partType, ref partName);
 | |
|                 TypedMessageHelper.GenerateMessageHeaderAttribute(setting, partAttributes, new XmlName(partName));
 | |
|                 AddAdditionalAttributes(setting, partAttributes, true /*isAdditionalAttributesAllowed*/);
 | |
|             }
 | |
| 
 | |
|             CodeTypeReference GetParameterType(MessagePartDescription setting)
 | |
|             {
 | |
|                 if (setting.Type != null)
 | |
|                     return Context.ServiceContractGenerator.GetCodeTypeReference(setting.Type);
 | |
|                 else if (this.Parent.parameterTypes.ContainsKey(setting))
 | |
|                     return this.Parent.parameterTypes[setting];
 | |
|                 else
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SfxNoTypeSpecifiedForParameter, setting.Name)));
 | |
|             }
 | |
| 
 | |
|             void AddAdditionalAttributes(MessagePartDescription setting, CodeAttributeDeclarationCollection attributes, bool isAdditionalAttributesAllowed)
 | |
|             {
 | |
|                 if (this.Parent.parameterAttributes != null && this.Parent.parameterAttributes.ContainsKey(setting))
 | |
|                 {
 | |
|                     CodeAttributeDeclarationCollection localAttributes = this.Parent.parameterAttributes[setting];
 | |
|                     if (localAttributes != null && localAttributes.Count > 0)
 | |
|                     {
 | |
|                         if (isAdditionalAttributesAllowed)
 | |
|                             attributes.AddRange(localAttributes);
 | |
|                         else
 | |
|                             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SfxUseTypedMessageForCustomAttributes, setting.Name, localAttributes[0].AttributeType.BaseType)));
 | |
| 
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
| 
 | |
|             static class TypedMessageHelper
 | |
|             {
 | |
|                 internal static void GenerateProtectionLevelAttribute(MessageDescription message, IPartCodeGenerator partCodeGenerator)
 | |
|                 {
 | |
|                     CodeAttributeDeclaration messageContractAttr = CustomAttributeHelper.FindOrCreateAttributeDeclaration<MessageContractAttribute>(partCodeGenerator.MessageLevelAttributes);
 | |
|                     if (message.HasProtectionLevel)
 | |
|                     {
 | |
|                         messageContractAttr.Arguments.Add(new CodeAttributeArgument("ProtectionLevel",
 | |
|                             new CodeFieldReferenceExpression(
 | |
|                                 new CodeTypeReferenceExpression(typeof(ProtectionLevel)), message.ProtectionLevel.ToString())));
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 internal static void GenerateWrapperAttribute(MessageDescription message, IPartCodeGenerator partCodeGenerator)
 | |
|                 {
 | |
|                     CodeAttributeDeclaration messageContractAttr = CustomAttributeHelper.FindOrCreateAttributeDeclaration<MessageContractAttribute>(partCodeGenerator.MessageLevelAttributes);
 | |
|                     if (message.Body.WrapperName != null)
 | |
|                     {
 | |
|                         // use encoded name to specify exactly what goes on the wire.
 | |
|                         messageContractAttr.Arguments.Add(new CodeAttributeArgument("WrapperName",
 | |
|                             new CodePrimitiveExpression(NamingHelper.CodeName(message.Body.WrapperName))));
 | |
|                         messageContractAttr.Arguments.Add(new CodeAttributeArgument("WrapperNamespace",
 | |
|                             new CodePrimitiveExpression(message.Body.WrapperNamespace)));
 | |
|                         messageContractAttr.Arguments.Add(new CodeAttributeArgument("IsWrapped",
 | |
|                             new CodePrimitiveExpression(true)));
 | |
|                     }
 | |
|                     else
 | |
|                         messageContractAttr.Arguments.Add(new CodeAttributeArgument("IsWrapped",
 | |
|                             new CodePrimitiveExpression(false)));
 | |
|                 }
 | |
| 
 | |
|                 internal static void AddEditorBrowsableAttribute(CodeAttributeDeclarationCollection attributes)
 | |
|                 {
 | |
|                     attributes.Add(ClientClassGenerator.CreateEditorBrowsableAttribute(EditorBrowsableState.Advanced));
 | |
|                 }
 | |
| 
 | |
|                 internal static void AddGeneratedTypedMessage(ServiceContractGenerationContext contract, MessageDescription message, CodeTypeReference codeTypeReference)
 | |
|                 {
 | |
|                     if (message.XsdTypeName != null && !message.XsdTypeName.IsEmpty)
 | |
|                     {
 | |
|                         contract.ServiceContractGenerator.GeneratedTypedMessages.Add(message, codeTypeReference);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 internal static bool FindGeneratedTypedMessage(ServiceContractGenerationContext contract, MessageDescription message, out CodeTypeReference codeTypeReference)
 | |
|                 {
 | |
|                     if (message.XsdTypeName == null || message.XsdTypeName.IsEmpty)
 | |
|                     {
 | |
|                         codeTypeReference = null;
 | |
|                         return false;
 | |
|                     }
 | |
|                     return contract.ServiceContractGenerator.GeneratedTypedMessages.TryGetValue(message, out codeTypeReference);
 | |
|                 }
 | |
| 
 | |
|                 internal static void GenerateConstructors(CodeTypeDeclaration typeDecl)
 | |
|                 {
 | |
|                     CodeConstructor defaultCtor = new CodeConstructor();
 | |
|                     defaultCtor.Attributes = MemberAttributes.Public;
 | |
|                     typeDecl.Members.Add(defaultCtor);
 | |
|                     CodeConstructor otherCtor = new CodeConstructor();
 | |
|                     otherCtor.Attributes = MemberAttributes.Public;
 | |
|                     foreach (CodeTypeMember member in typeDecl.Members)
 | |
|                     {
 | |
|                         CodeMemberField field = member as CodeMemberField;
 | |
|                         if (field == null)
 | |
|                             continue;
 | |
|                         CodeParameterDeclarationExpression param = new CodeParameterDeclarationExpression(field.Type, field.Name);
 | |
|                         otherCtor.Parameters.Add(param);
 | |
|                         otherCtor.Statements.Add(new CodeAssignStatement(new CodeFieldReferenceExpression(new CodeThisReferenceExpression(), field.Name), new CodeArgumentReferenceExpression(param.Name)));
 | |
|                     }
 | |
|                     if (otherCtor.Parameters.Count > 0)
 | |
|                         typeDecl.Members.Add(otherCtor);
 | |
|                 }
 | |
| 
 | |
|                 internal static void GenerateMessageBodyMemberAttribute(int order, MessagePartDescription setting, CodeAttributeDeclarationCollection attributes, XmlName defaultName)
 | |
|                 {
 | |
|                     GenerateMessageContractMemberAttribute<MessageBodyMemberAttribute>(order, setting, attributes, defaultName);
 | |
|                 }
 | |
| 
 | |
|                 internal static void GenerateMessageHeaderAttribute(MessageHeaderDescription setting, CodeAttributeDeclarationCollection attributes, XmlName defaultName)
 | |
|                 {
 | |
|                     if (setting.Multiple)
 | |
|                         GenerateMessageContractMemberAttribute<MessageHeaderArrayAttribute>(-1, setting, attributes, defaultName);
 | |
|                     else
 | |
|                         GenerateMessageContractMemberAttribute<MessageHeaderAttribute>(-1, setting, attributes, defaultName);
 | |
|                 }
 | |
| 
 | |
|                 static void GenerateMessageContractMemberAttribute<T>(int order, MessagePartDescription setting, CodeAttributeDeclarationCollection attrs, XmlName defaultName)
 | |
|                     where T : Attribute
 | |
|                 {
 | |
|                     CodeAttributeDeclaration decl = CustomAttributeHelper.FindOrCreateAttributeDeclaration<T>(attrs);
 | |
| 
 | |
|                     if (setting.Name != defaultName.EncodedName)
 | |
|                         // override name with encoded value specified in wsdl; this only works beacuse
 | |
|                         // our Encoding algorithm will leave alredy encoded names untouched
 | |
|                         CustomAttributeHelper.CreateOrOverridePropertyDeclaration(decl, MessageContractMemberAttribute.NamePropertyName, setting.Name);
 | |
| 
 | |
|                     CustomAttributeHelper.CreateOrOverridePropertyDeclaration(decl, MessageContractMemberAttribute.NamespacePropertyName, setting.Namespace);
 | |
| 
 | |
|                     if (setting.HasProtectionLevel)
 | |
|                         CustomAttributeHelper.CreateOrOverridePropertyDeclaration(decl, MessageContractMemberAttribute.ProtectionLevelPropertyName, setting.ProtectionLevel);
 | |
| 
 | |
|                     if (order >= 0)
 | |
|                         CustomAttributeHelper.CreateOrOverridePropertyDeclaration(decl, MessageBodyMemberAttribute.OrderPropertyName, order);
 | |
|                 }
 | |
| 
 | |
|             }
 | |
| 
 | |
|             static class ParameterizedMessageHelper
 | |
|             {
 | |
| 
 | |
|                 internal static void GenerateMessageParameterAttribute(MessagePartDescription setting, CodeAttributeDeclarationCollection attributes, XmlName defaultName, string defaultNS)
 | |
|                 {
 | |
|                     if (setting.Name != defaultName.EncodedName)
 | |
|                     {
 | |
|                         // override name with encoded value specified in wsdl; this only works beacuse
 | |
|                         // our Encoding algorithm will leave alredy encoded names untouched
 | |
|                         CustomAttributeHelper.CreateOrOverridePropertyDeclaration(
 | |
|                             CustomAttributeHelper.FindOrCreateAttributeDeclaration<MessageParameterAttribute>(attributes), MessageParameterAttribute.NamePropertyName, setting.Name);
 | |
|                     }
 | |
|                     if (setting.Namespace != defaultNS)
 | |
|                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_NamespaceMismatch, setting.Namespace, defaultNS)));
 | |
|                 }
 | |
| 
 | |
|                 internal static void ValidateProtectionLevel(MethodSignatureGenerator parent)
 | |
|                 {
 | |
|                     if (parent.Request != null && parent.Request.HasProtectionLevel)
 | |
|                     {
 | |
|                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_MessageHasProtectionLevel, parent.Request.Action == null ? "" : parent.Request.Action)));
 | |
|                     }
 | |
|                     if (parent.Response != null && parent.Response.HasProtectionLevel)
 | |
|                     {
 | |
|                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_MessageHasProtectionLevel, parent.Response.Action == null ? "" : parent.Response.Action)));
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 internal static void ValidateWrapperSettings(MethodSignatureGenerator parent)
 | |
|                 {
 | |
|                     if (parent.Request.Body.WrapperName == null || (parent.Response != null && parent.Response.Body.WrapperName == null))
 | |
|                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_Bare, parent.Context.Operation.CodeName)));
 | |
| 
 | |
|                     if (!StringEqualOrNull(parent.Request.Body.WrapperNamespace, parent.ContractNS))
 | |
|                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_DifferentWrapperNs, parent.Request.MessageName, parent.Request.Body.WrapperNamespace, parent.ContractNS)));
 | |
| 
 | |
|                     XmlName defaultName = new XmlName(parent.DefaultName);
 | |
|                     if (!String.Equals(parent.Request.Body.WrapperName, defaultName.EncodedName, StringComparison.Ordinal))
 | |
|                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_DifferentWrapperName, parent.Request.MessageName, parent.Request.Body.WrapperName, defaultName.EncodedName)));
 | |
| 
 | |
|                     if (parent.Response != null)
 | |
|                     {
 | |
|                         if (!StringEqualOrNull(parent.Response.Body.WrapperNamespace, parent.ContractNS))
 | |
|                             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_DifferentWrapperNs, parent.Response.MessageName, parent.Response.Body.WrapperNamespace, parent.ContractNS)));
 | |
| 
 | |
|                         if (!String.Equals(parent.Response.Body.WrapperName, TypeLoader.GetBodyWrapperResponseName(defaultName).EncodedName, StringComparison.Ordinal))
 | |
|                             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_DifferentWrapperName, parent.Response.MessageName, parent.Response.Body.WrapperName, defaultName.EncodedName)));
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 internal static void ValidateNoHeaders(MethodSignatureGenerator parent)
 | |
|                 {
 | |
|                     if (parent.Request.Headers.Count > 0)
 | |
|                     {
 | |
|                         if (parent.IsEncoded)
 | |
|                         {
 | |
|                             parent.Context.Contract.ServiceContractGenerator.Errors.Add(new MetadataConversionError(SR.GetString(SR.SFxCannotImportAsParameters_HeadersAreIgnoredInEncoded, parent.Request.MessageName), true/*isWarning*/));
 | |
|                         }
 | |
|                         else
 | |
|                             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_HeadersAreUnsupported, parent.Request.MessageName)));
 | |
|                     }
 | |
| 
 | |
|                     if (!parent.Oneway && parent.Response.Headers.Count > 0)
 | |
|                     {
 | |
|                         if (parent.IsEncoded)
 | |
|                             parent.Context.Contract.ServiceContractGenerator.Errors.Add(new MetadataConversionError(SR.GetString(SR.SFxCannotImportAsParameters_HeadersAreIgnoredInEncoded, parent.Response.MessageName), true/*isWarning*/));
 | |
|                         else
 | |
|                             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ParameterModeException(SR.GetString(SR.SFxCannotImportAsParameters_HeadersAreUnsupported, parent.Response.MessageName)));
 | |
|                     }
 | |
| 
 | |
|                 }
 | |
| 
 | |
|                 static bool StringEqualOrNull(string overrideValue, string defaultValue)
 | |
|                 {
 | |
|                     return overrideValue == null || String.Equals(overrideValue, defaultValue, StringComparison.Ordinal);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             internal void GenerateTaskSignature(ref OperationFormatStyle style)
 | |
|             {
 | |
|                 this.Method = this.Context.TaskMethod;
 | |
|                 this.EndMethod = this.Context.TaskMethod;
 | |
|                 this.DefaultName = this.Context.SyncMethod.Name;
 | |
|                 GenerateOperationSignatures(ref style);
 | |
| 
 | |
|                 CodeTypeReference resultType = this.Method.ReturnType;
 | |
|                 CodeTypeReference taskReturnType;
 | |
|                 if (resultType.BaseType == ServiceReflector.VoidType.FullName)
 | |
|                 {
 | |
|                     taskReturnType = new CodeTypeReference(ServiceReflector.taskType);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     taskReturnType = new CodeTypeReference(this.Context.ServiceContractGenerator.GetCodeTypeReference(ServiceReflector.taskTResultType).BaseType, resultType);
 | |
|                 }
 | |
|                 
 | |
|                 this.Method.ReturnType = taskReturnType;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         static class CustomAttributeHelper
 | |
|         {
 | |
|             internal static void CreateOrOverridePropertyDeclaration<V>(CodeAttributeDeclaration attribute, string propertyName, V value)
 | |
|             {
 | |
|                 SecurityAttributeGenerationHelper.CreateOrOverridePropertyDeclaration<V>(attribute, propertyName, value);
 | |
|             }
 | |
| 
 | |
|             internal static CodeAttributeDeclaration FindOrCreateAttributeDeclaration<T>(CodeAttributeDeclarationCollection attributes)
 | |
|                 where T : Attribute
 | |
|             {
 | |
|                 return SecurityAttributeGenerationHelper.FindOrCreateAttributeDeclaration<T>(attributes);
 | |
|             }
 | |
| 
 | |
|             internal static CodeAttributeDeclaration GenerateAttributeDeclaration(ServiceContractGenerator generator, Attribute attribute)
 | |
|             {
 | |
|                 Type attributeType = attribute.GetType();
 | |
|                 Attribute defaultAttribute = (Attribute)Activator.CreateInstance(attributeType);
 | |
|                 MemberInfo[] publicMembers = attributeType.GetMembers(BindingFlags.Instance | BindingFlags.Public);
 | |
|                 Array.Sort<MemberInfo>(publicMembers,
 | |
|                                        delegate(MemberInfo a, MemberInfo b)
 | |
|                                        {
 | |
|                                            return String.Compare(a.Name, b.Name, StringComparison.Ordinal);
 | |
|                                        }
 | |
|                 );
 | |
|                 // we should create this reference through ServiceContractGenerator, which tracks referenced assemblies
 | |
|                 CodeAttributeDeclaration attr = new CodeAttributeDeclaration(generator.GetCodeTypeReference(attributeType));
 | |
|                 foreach (MemberInfo member in publicMembers)
 | |
|                 {
 | |
|                     if (member.DeclaringType == typeof(Attribute))
 | |
|                         continue;
 | |
|                     FieldInfo field = member as FieldInfo;
 | |
|                     if (field != null)
 | |
|                     {
 | |
|                         object fieldValue = field.GetValue(attribute);
 | |
|                         object defaultValue = field.GetValue(defaultAttribute);
 | |
| 
 | |
|                         if (!object.Equals(fieldValue, defaultValue))
 | |
|                             attr.Arguments.Add(new CodeAttributeArgument(field.Name, GetArgValue(fieldValue)));
 | |
|                         continue;
 | |
|                     }
 | |
|                     PropertyInfo property = member as PropertyInfo;
 | |
|                     if (property != null)
 | |
|                     {
 | |
|                         object propertyValue = property.GetValue(attribute, null);
 | |
|                         object defaultValue = property.GetValue(defaultAttribute, null);
 | |
|                         if (!object.Equals(propertyValue, defaultValue))
 | |
|                             attr.Arguments.Add(new CodeAttributeArgument(property.Name, GetArgValue(propertyValue)));
 | |
|                         continue;
 | |
|                     }
 | |
|                 }
 | |
|                 return attr;
 | |
|             }
 | |
| 
 | |
|             static CodeExpression GetArgValue(object val)
 | |
|             {
 | |
|                 Type type = val.GetType();
 | |
|                 if (type.IsPrimitive || type == typeof(string))
 | |
|                     return new CodePrimitiveExpression(val);
 | |
|                 if (type.IsEnum)
 | |
|                     return new CodeFieldReferenceExpression(new CodeTypeReferenceExpression(type), Enum.Format(type, val, "G"));
 | |
| 
 | |
|                 Fx.Assert(String.Format(System.Globalization.CultureInfo.InvariantCulture, "Attribute generation is not supported for argument type {0}", type));
 | |
|                 return null;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class ParameterModeException : Exception
 | |
|     {
 | |
|         MessageContractType messageContractType = MessageContractType.WrappedMessageContract;
 | |
|         public ParameterModeException() { }
 | |
|         public ParameterModeException(string message) : base(message) { }
 | |
|         public ParameterModeException(SerializationInfo info, StreamingContext context) : base(info, context) { }
 | |
|         public MessageContractType MessageContractType
 | |
|         {
 | |
|             get { return messageContractType; }
 | |
|             set { messageContractType = value; }
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
| }
 |