WebServiceData.cs source code in C# .NET

Source code for the .NET framework in C#

                        

Code:

/ Dotnetfx_Vista_SP2 / Dotnetfx_Vista_SP2 / 8.0.50727.4016 / DEVDIV / depot / DevDiv / releases / Orcas / QFE / ndp / fx / src / xsp / System / Web / Extensions / Script / Services / WebServiceData.cs / 3 / WebServiceData.cs

                            //------------------------------------------------------------------------------ 
// 
//     Copyright (c) Microsoft Corporation.  All rights reserved.
// 
//----------------------------------------------------------------------------- 

namespace System.Web.Script.Services { 
    using System; 
    using System.Collections;
    using System.Collections.Generic; 
    using System.Diagnostics;
    using System.Globalization;
    using System.IO;
    using System.Reflection; 
    using System.Security;
    using System.ServiceModel.Channels; 
    using System.ServiceModel.Description; 
    using System.ServiceModel.Web;
    using System.Web; 
    using System.Web.Caching;
    using System.Web.Compilation;
    using System.Web.Configuration;
    using System.Web.Hosting; 
    using System.Web.Resources;
    using System.Web.Script.Serialization; 
    using System.Web.Services; 
    using System.Xml;
 
    internal class WebServiceData : JavaScriptTypeResolver {
        private WebServiceTypeData _typeData;
        private bool _pageMethods; // True for page methods(which only look at static methods)
        private Dictionary _methods; 

        // this is used to map __type ids in the JSON string to something other than type.FullName 
        private Dictionary _typeResolverSpecials = new Dictionary(); 
        private Dictionary _clientTypesDictionary;
        private Dictionary _clientTypeNameDictionary; 
        private Dictionary _enumTypesDictionary;
        private Hashtable _processedTypes;
        private bool _clientTypesProcessed;
 
        private JavaScriptSerializer _serializer;
        internal JavaScriptSerializer Serializer { 
            get { 
                return _serializer;
            } 
        }

        internal const string _profileServiceFileName = "Profile_JSON_AppService.axd";
        internal const string _authenticationServiceFileName = "Authentication_JSON_AppService.axd"; 
        internal const string _roleServiceFileName = "Role_JSON_AppService.axd";
 
        private static WebServiceData GetApplicationService(string appRelativePath) { 
            // we only support the application services being accessed at the root level, so that url authorization can be used to control their access.
            // In other words, "~/Profile_JSON_AppService.axd" should work but not "~/SomeSubDir/Profile_JSON_AppService.axd". 
            // AppRelativeCurrentExecutionFilePath looks like "~/path/filename.ext".
            // So we can easily detect if the file requested is in the root by ensuring that the last index of "/" is == 1,
            // as in the path "~/rootfile.ext", where "/" is the second character.
            // Note that the WebServiceData object is cached higher in the stack once calculated 
            int slashIndex = appRelativePath.LastIndexOf('/');
            if (slashIndex == 1) { 
                // it is a root file. Now see if its one of the two built in services 
                string name = Path.GetFileName(appRelativePath);
 
                if (name.Equals(_profileServiceFileName, StringComparison.OrdinalIgnoreCase)) {
                    return new WebServiceData(typeof(System.Web.Profile.ProfileService), false);
                }
                else if (name.Equals(_authenticationServiceFileName, StringComparison.OrdinalIgnoreCase)) { 
                    return new WebServiceData(typeof(System.Web.Security.AuthenticationService), false);
                } 
                else if (name.Equals(_roleServiceFileName, StringComparison.OrdinalIgnoreCase)) { 
                    return new WebServiceData(typeof(System.Web.Security.RoleService), false);
                } 
            }

            return null;
        } 

        internal static WebServiceData GetWebServiceData(HttpContext context, string virtualPath) { 
            return GetWebServiceData(context, virtualPath, true /*failIfNoData*/, false /*pageMethods*/, false/*inlineScript*/); 
        }
 
        private static string GetCacheKey(string virtualPath) {
            return "System.Web.Script.Services.WebServiceData:" + virtualPath;
        }
 
        internal static WebServiceData GetWebServiceData(HttpContext context, string virtualPath, bool failIfNoData, bool pageMethods) {
            return GetWebServiceData(context, virtualPath, failIfNoData, pageMethods, false /*inlineScript*/); 
        } 

        internal static WebServiceData GetWebServiceData(HttpContext context, string virtualPath, bool failIfNoData, bool pageMethods, bool inlineScript) { 
            // Make sure the path is cannonical to avoid doing more work than necessary
            virtualPath = VirtualPathUtility.ToAbsolute(virtualPath);

            string cacheKey = GetCacheKey(virtualPath); 
            WebServiceData data = context.Cache[cacheKey] as WebServiceData;
 
            // Handle the case where the virtualPath exists, for example a real asmx page. 
            if (data == null) {
                if (HostingEnvironment.VirtualPathProvider.FileExists(virtualPath)) { 
                    Type compiledType = null;
                    try {
                        compiledType = BuildManager.GetCompiledType(virtualPath);
 
                        // If we can't get the compiled type, try creating an instance (i.e. for no compile pages)
                        if (compiledType == null) { 
                            object page = BuildManager.CreateInstanceFromVirtualPath(virtualPath, typeof(System.Web.UI.Page)); 
                            if (page != null) {
                                compiledType = page.GetType(); 
                            }
                        }

                    } 
                    catch (SecurityException) {
                        // DevDiv 33708: BuildManager requires Medium trust, so we need to no-op rather than 
                        // destroying the page. 
                    }
 
                    if (compiledType != null) {
                        data = new WebServiceData(compiledType, pageMethods);
                        BuildDependencySet deps = BuildManager.GetCachedBuildDependencySet(context, virtualPath);
                        CacheDependency cd = HostingEnvironment.VirtualPathProvider.GetCacheDependency(virtualPath, deps.VirtualPaths, DateTime.Now); 
                        context.Cache.Insert(cacheKey, data, cd);
                    } 
                } 
                else if (virtualPath.EndsWith("_AppService.axd", StringComparison.OrdinalIgnoreCase)) {
                    // File does not exist, but the url may be a request for one of the three built-in services: ProfileService, AuthenticationService, RoleService 
                    data = WebServiceData.GetApplicationService(context.Request.AppRelativeCurrentExecutionFilePath);
                    if (data != null) {
                        context.Cache.Insert(cacheKey, data);
                    } 
                }
            } 
 
            if (data == null) {
                if (failIfNoData) { 
                    if (inlineScript) {
                        //DevDiv 74432: InlineScript = true fails, for WCF serviceReferences: Need an appropriate error message
                        throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, AtlasWeb.WebService_NoWebServiceDataInlineScript, virtualPath));
                    } 
                    else {
                        throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, AtlasWeb.WebService_NoWebServiceData, virtualPath)); 
                    } 
                }
                else { 
                    return null;
                }
            }
 
            return data;
        } 
 
        internal static WebServiceData GetWebServiceData(ContractDescription contract) {
            WebServiceData serviceData = new WebServiceData(); 

            // set service type
            serviceData._typeData = new WebServiceTypeData(XmlConvert.DecodeName(contract.Name), XmlConvert.DecodeName(contract.Namespace), contract.ContractType);
 
            // build list of types used in service
            Dictionary clientTypeDictionary = new Dictionary(); 
            serviceData._clientTypesDictionary = clientTypeDictionary; 
            Dictionary enumTypeDictionary = new Dictionary();
            serviceData._enumTypesDictionary = enumTypeDictionary; 
            serviceData._processedTypes = new Hashtable();
            serviceData._clientTypesProcessed = true;
            serviceData._clientTypeNameDictionary = new Dictionary();
 
            //build method dictionary
            Dictionary methodDataDictionary = new Dictionary(); 
            serviceData._methods = methodDataDictionary; 
            foreach (OperationDescription operation in contract.Operations) {
                Dictionary parameterDataDictionary = new Dictionary(); 
                bool useHttpGet = operation.Behaviors.Find() != null;
                WebServiceMethodData methodData = new WebServiceMethodData(serviceData, XmlConvert.DecodeName(operation.Name), parameterDataDictionary, useHttpGet);
                // build parameter dictionary
                MessageDescription requestMessage = operation.Messages[0]; 
                if (requestMessage != null) {
                    int numMessageParts = requestMessage.Body.Parts.Count; 
                    for (int p = 0; p < numMessageParts; p++) { 
                        MessagePartDescription messagePart = requestMessage.Body.Parts[p];
                        // DevDiv 129964:JS proxy generation fails for a WCF service that uses an untyped message 
                        // Message or its derived class are special, used for untyped operation contracts.
                        // As per the WCF team proxy generated for them should treat Message equivalent to Object type.
                        Type paramType = ReplaceMessageWithObject(messagePart.Type);
                        WebServiceParameterData parameterData = new WebServiceParameterData(XmlConvert.DecodeName(messagePart.Name), paramType, p); 
                        parameterDataDictionary[parameterData.ParameterName] = parameterData;
                        serviceData.ProcessClientType(paramType, false, true); 
                    } 
                }
                if (operation.Messages.Count > 1) { 
                    // its a two way operation, get type information from return message
                    MessageDescription responseMessage = operation.Messages[1];
                    if (responseMessage != null) {
                        if (responseMessage.Body.ReturnValue != null && responseMessage.Body.ReturnValue.Type != null) { 
                            // operation has a return type, add type to list of type proxy to generate
                            serviceData.ProcessClientType(ReplaceMessageWithObject(responseMessage.Body.ReturnValue.Type), false, true); 
                        } 
                    }
                } 

                //add known types at operation level
                for (int t = 0; t < operation.KnownTypes.Count; t++) {
                    serviceData.ProcessClientType(operation.KnownTypes[t], false, true); 
                }
 
                methodDataDictionary[methodData.MethodName] = methodData; 
            }
            serviceData._processedTypes = null; 
            return serviceData;

        }
 
        private static Type ReplaceMessageWithObject(Type t) {
            return (typeof(Message).IsAssignableFrom(t)) ? typeof(object) : t; 
        } 

        private WebServiceData() { 
        }

        private WebServiceData(WebServiceTypeData typeData) {
            _typeData = typeData; 
            _serializer = new JavaScriptSerializer(this);
#pragma warning disable 0436 
            ScriptingJsonSerializationSection.ApplicationSettings settings = new ScriptingJsonSerializationSection.ApplicationSettings(); 
#pragma warning restore 0436
            _serializer.MaxJsonLength = settings.MaxJsonLimit; 
            _serializer.RecursionLimit = settings.RecursionLimit;
            _serializer.RegisterConverters(settings.Converters);
        }
 
        // Normal ASMX Atlas codepath for creating webservice data
        internal WebServiceData(Type type, bool pageMethods) 
            : this(new WebServiceTypeData(type.Name, type.Namespace, type)) { 
            _pageMethods = pageMethods;
            // Pages don't need to have script service attribute 
            if (!_pageMethods) {
                object[] attribs = type.GetCustomAttributes(typeof(ScriptServiceAttribute), true);
                if (attribs.Length == 0) {
                    throw new InvalidOperationException(AtlasWeb.WebService_NoScriptServiceAttribute); 
                }
            } 
        } 

        // Indigo entry point for creating WebServiceData 
        internal WebServiceData(WebServiceTypeData typeData, Dictionary methods)
            : this(typeData) {
            _methods = methods;
        } 

        private void AddMethod(Dictionary methods, MethodInfo method) { 
            object[] wmAttribs = method.GetCustomAttributes(typeof(WebMethodAttribute), true); 

            // Skip it if it doesn't have the WebMethod attribute 
            if (wmAttribs.Length == 0)
                return;

            ScriptMethodAttribute sm = null; 
            object[] responseAttribs = method.GetCustomAttributes(typeof(ScriptMethodAttribute), true);
            if (responseAttribs.Length > 0) { 
                sm = (ScriptMethodAttribute)responseAttribs[0]; 
            }
 
            // Create an object to keep track of this method's data
            WebServiceMethodData wmd = new WebServiceMethodData(this, method, (WebMethodAttribute)wmAttribs[0], sm);
            methods[wmd.MethodName] = wmd;
        } 

        private void EnsureMethods() { 
            // Type will only be null for the Indigo code path 
            if (_methods != null || _typeData.Type == null)
                return; 

            // Build the method collection on demand
            lock (this) {
 
                // Need to add the methods of each type in reverse order
                List typeList = new List(); 
                Type current = _typeData.Type; 
                typeList.Add(current);
                while (current.BaseType != null) { 
                    current = current.BaseType;
                    typeList.Add(current);
                }
                Dictionary methods = new Dictionary(StringComparer.OrdinalIgnoreCase); 
                BindingFlags flags = BindingFlags.Public | BindingFlags.DeclaredOnly;
                if (_pageMethods) flags |= BindingFlags.Static; 
                else flags |= BindingFlags.Instance; 

                // Add the methods in reverse order from base to derived 
                for (int i = typeList.Count - 1; i >= 0; --i) {
                    MethodInfo[] methodInfos = typeList[i].GetMethods(flags);
                    foreach (MethodInfo method in methodInfos) {
                        AddMethod(methods, method); 
                    }
                } 
                _methods = methods; 
            }
        } 

        internal WebServiceTypeData TypeData {
            get { return _typeData; }
        } 

        internal ICollection MethodDatas { 
            get { 
                EnsureMethods();
                return _methods.Values; 
            }
        }

        internal WebServiceMethodData GetMethodData(string methodName) { 
            EnsureMethods();
 
            // Fail if the web method doesn't exist 
            WebServiceMethodData methodData = null;
            if (!_methods.TryGetValue(methodName, out methodData)) { 
                throw new ArgumentException(
                    String.Format(CultureInfo.CurrentCulture, AtlasWeb.WebService_UnknownWebMethod, methodName), "methodName");
            }
 
            EnsureClientTypesProcessed();
            return methodData; 
        } 

        private void EnsureClientTypesProcessed() { 
            if (_clientTypesProcessed)
                return;

            lock (this) { 
                if (_clientTypesProcessed)
                    return; 
                ProcessClientTypes(); 
            }
        } 

        private void ProcessClientTypes() {
            Debug.Assert(!_clientTypesProcessed, "ProcessClientTypes shouldn't be called after it has already been successfully run.");
 
            // List of types that can be instantiated on the client
            _clientTypesDictionary = new Dictionary(); 
            _enumTypesDictionary = new Dictionary(); 

            _clientTypeNameDictionary = new Dictionary(); 

            try {
                // _processedTypes is used to avoid processing a Type more than once
                _processedTypes = new Hashtable(); 

                // Process any GenerateScriptTypes on the Service type 
                ProcessIncludeAttributes((GenerateScriptTypeAttribute[])_typeData.Type.GetCustomAttributes(typeof(GenerateScriptTypeAttribute), true)); 

                foreach (WebServiceMethodData methodData in MethodDatas) { 

                    // Process any GenerateScriptTypes on the method
                    ProcessIncludeAttributes((GenerateScriptTypeAttribute[])methodData.MethodInfo.GetCustomAttributes(typeof(GenerateScriptTypeAttribute), true));
 
                    // Also add any input parameters
                    foreach (WebServiceParameterData paramData in methodData.ParameterDatas) { 
                        ProcessClientType(paramData.ParameterInfo.ParameterType); 
                    }
 
                    // Ignore return type if it uses XML instead of JSON
                    if (methodData.UseXmlResponse) continue;
                    ProcessClientType(methodData.ReturnType);
                } 

                // DevDiv 60672: Only set to true if the proxies were SUCCESSFULLY processed 
                // Only setting _clientTypesProcessed=true on success will cause us to retry creating the proxies each 
                // request when there is an exception, and so the same exception will be thrown each time.
                _clientTypesProcessed = true; 
            }
            catch {
                // If we have any exception we have to null out our caches
                _clientTypesDictionary = null; 
                _enumTypesDictionary = null;
                _clientTypeNameDictionary = null; 
                throw; 
            }
            finally { 
                _processedTypes = null;
            }
        }
 
        private void ProcessIncludeAttributes(GenerateScriptTypeAttribute[] attributes) {
            foreach (GenerateScriptTypeAttribute attribute in attributes) { 
                if (!String.IsNullOrEmpty(attribute.ScriptTypeId)) 
                    _typeResolverSpecials[attribute.Type.FullName] = attribute.ScriptTypeId;
 
                Type t = attribute.Type;
                if (t.IsPrimitive || t == typeof(object) || t == typeof(string) ||
                    t == typeof(DateTime) || t == typeof(Guid) ||
                    typeof(IEnumerable).IsAssignableFrom(t) || typeof(IDictionary).IsAssignableFrom(t) || 
                    (t.IsGenericType && t.GetGenericArguments().Length > 1) ||
                    !ObjectConverter.IsClientInstantiatableType(t, _serializer)) 
                    throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, 
                        AtlasWeb.WebService_InvalidGenerateScriptType, t.FullName));
 
                ProcessClientType(t, true);
            }
        }
 
        private void ProcessClientType(Type t) {
            ProcessClientType(t, false, false); 
        } 

 
        private void ProcessClientType(Type t, bool force) {
            ProcessClientType(t, force, false);
        }
 
        // Force = true is required for when we detect a GenerateScriptType which may supply a new ScriptTypeID
        private void ProcessClientType(Type t, bool force, bool isWCF) { 
            if (!force && _processedTypes.Contains(t)) 
                return;
            _processedTypes[t] = null; 

            // Keep track of all enum Types
            if (t.IsEnum) {
                WebServiceEnumData enumData = null; 
                if (isWCF) {
                    enumData = (WebServiceEnumData)WebServiceTypeData.GetWebServiceTypeData(t); 
                } 
                else {
                    enumData = new WebServiceEnumData(t.Name, t.Namespace, t, Enum.GetNames(t), Enum.GetValues(t), Enum.GetUnderlyingType(t) == typeof(ulong)); 
                }
                _enumTypesDictionary[GetTypeStringRepresentation(enumData.TypeName, false)] = enumData;
                return;
            } 

            // For generics, we only allow generic types with one parameter, which we will try to process 
            if (t.IsGenericType) { 
                if (isWCF) {
                     ProcessKnownTypes(t); 
                }
                else {
                    Type[] genericArgs = t.GetGenericArguments();
                    if (genericArgs.Length > 1) { 
                        return;
                    } 
                    ProcessClientType(genericArgs[0], false, isWCF); 
                }
            } 
            // Support arrays explicitly
            else if (t.IsArray) {
                ProcessClientType(t.GetElementType(), false, isWCF);
            } 
            else {
                // Ignore primitive types 
                // Ignore DateTime, since we have special serialization handling for it in the JavaScriptSerializer 
                // Ignore IDctionary and IEnumerables as well
                if (t.IsPrimitive || t == typeof(object) || t == typeof(string) || t == typeof(DateTime) || 
                    t == typeof(void) || t == typeof(System.Decimal) || t == typeof(Guid) ||
                    typeof(IEnumerable).IsAssignableFrom(t) || typeof(IDictionary).IsAssignableFrom(t) ||
                    (!isWCF && !ObjectConverter.IsClientInstantiatableType(t, _serializer)))
                    return; 

                // Only add it to the list of client types if it can be instantiated. 
                // pass false to skip the lock 
                if (isWCF) {
                    ProcessKnownTypes(t); 
                }
                else {
                    string typeStringRepresentation = GetTypeStringRepresentation(t.FullName, false);
                    _clientTypesDictionary[typeStringRepresentation] = new WebServiceTypeData(t.Name, t.Namespace, t); 
                    _clientTypeNameDictionary[t] = typeStringRepresentation;
 
                } 
            }
        } 

        private void ProcessKnownTypes(Type t) {
            WebServiceTypeData typeData = WebServiceTypeData.GetWebServiceTypeData(t);
            bool alreadyProcessed = false; 
            if (typeData == null) {
                // indicates a type was used that is a built-in type 
                return; 
            }
 
            // if T implments IEnumerable or IDictionary, do not include type proxy for it
            // but still continue to get known types. I.e List should ignore List
            // but process MyType
            if (!(typeof(IEnumerable).IsAssignableFrom(t) || typeof(IDictionary).IsAssignableFrom(t))) { 
                 _clientTypeNameDictionary[t] = GetTypeStringRepresentation(typeData.TypeName);
                 alreadyProcessed = ProcessTypeData(typeData); 
            } 

            if (!alreadyProcessed) { 
                IList knownTypes = WebServiceTypeData.GetKnownTypes(t, typeData);
                foreach (WebServiceTypeData knownType in knownTypes) {
                    ProcessTypeData(knownType);
                } 
            }
 
        } 

        // returns true if typeData already exists in typeDictionary 
        private bool ProcessTypeData(WebServiceTypeData typeData) {
            string typeString = GetTypeStringRepresentation(typeData.TypeName);
            bool retval = true;
            if (typeData is WebServiceEnumData) { 
                if (!_enumTypesDictionary.ContainsKey(typeString)) {
                    _enumTypesDictionary[typeString] = (WebServiceEnumData)typeData; 
                    retval = false; 
                }
            } 
            else {
                if (!_clientTypesDictionary.ContainsKey(typeString)) {
                    _clientTypesDictionary[typeString] = typeData;
                    retval = false; 
                }
            } 
            return retval; 
        }
 

        internal IEnumerable ClientTypes {
            get {
                return ClientTypeDictionary.Values; 
            }
        } 
 
        internal Dictionary ClientTypeDictionary {
            get { 
                EnsureClientTypesProcessed();
                return _clientTypesDictionary;
            }
            set { 
                _clientTypesDictionary = value;
            } 
        } 

        internal Dictionary ClientTypeNameDictionary { 
            get {
                EnsureClientTypesProcessed();
                return _clientTypeNameDictionary;
            } 
        }
 
        internal IEnumerable EnumTypes { 
            get {
                EnsureClientTypesProcessed(); 
                return _enumTypesDictionary.Values;
            }
        }
 
        internal Dictionary EnumTypeDictionary {
            get { 
                EnsureClientTypesProcessed(); 
                return _enumTypesDictionary;
            } 
            set {
                _enumTypesDictionary = value;
            }
        } 

        public override Type ResolveType(string id) { 
            WebServiceTypeData type = null; 
            if (ClientTypeDictionary.TryGetValue(id, out type)) {
                if (type != null) { 
                    return type.Type;
                }
            }
            return null; 
        }
 
        public override string ResolveTypeId(Type type) { 
            string typeString = GetTypeStringRepresentation(type.FullName);
 
            // If this type is not in the dictionary
            if (!ClientTypeDictionary.ContainsKey(typeString))
                return null;
 
            return typeString;
        } 
 
        internal string GetTypeStringRepresentation(string typeName) {
            return GetTypeStringRepresentation(typeName, true); 
        }

        internal string GetTypeStringRepresentation(string typeName, bool ensure) {
            if (ensure) { 
                EnsureClientTypesProcessed();
            } 
 
            // Handle special cases from GenerateScriptType first
            string typeString; 
            if (_typeResolverSpecials.TryGetValue(typeName, out typeString)) {
                return typeString;
            }
            return typeName; 
        }
 
        internal string GetTypeStringRepresentation(WebServiceTypeData typeData) { 
            //First check if typeData provides its string representaiton ( for WCF case)
            string typeString = typeData.StringRepresentation; 
            if (typeString == null) {
                typeString = GetTypeStringRepresentation(typeData.TypeName, true);
            }
            return typeString; 
        }
    } 
} 

// File provided for Reference Use Only by Microsoft Corporation (c) 2007.
//------------------------------------------------------------------------------ 
// 
//     Copyright (c) Microsoft Corporation.  All rights reserved.
// 
//----------------------------------------------------------------------------- 

namespace System.Web.Script.Services { 
    using System; 
    using System.Collections;
    using System.Collections.Generic; 
    using System.Diagnostics;
    using System.Globalization;
    using System.IO;
    using System.Reflection; 
    using System.Security;
    using System.ServiceModel.Channels; 
    using System.ServiceModel.Description; 
    using System.ServiceModel.Web;
    using System.Web; 
    using System.Web.Caching;
    using System.Web.Compilation;
    using System.Web.Configuration;
    using System.Web.Hosting; 
    using System.Web.Resources;
    using System.Web.Script.Serialization; 
    using System.Web.Services; 
    using System.Xml;
 
    internal class WebServiceData : JavaScriptTypeResolver {
        private WebServiceTypeData _typeData;
        private bool _pageMethods; // True for page methods(which only look at static methods)
        private Dictionary _methods; 

        // this is used to map __type ids in the JSON string to something other than type.FullName 
        private Dictionary _typeResolverSpecials = new Dictionary(); 
        private Dictionary _clientTypesDictionary;
        private Dictionary _clientTypeNameDictionary; 
        private Dictionary _enumTypesDictionary;
        private Hashtable _processedTypes;
        private bool _clientTypesProcessed;
 
        private JavaScriptSerializer _serializer;
        internal JavaScriptSerializer Serializer { 
            get { 
                return _serializer;
            } 
        }

        internal const string _profileServiceFileName = "Profile_JSON_AppService.axd";
        internal const string _authenticationServiceFileName = "Authentication_JSON_AppService.axd"; 
        internal const string _roleServiceFileName = "Role_JSON_AppService.axd";
 
        private static WebServiceData GetApplicationService(string appRelativePath) { 
            // we only support the application services being accessed at the root level, so that url authorization can be used to control their access.
            // In other words, "~/Profile_JSON_AppService.axd" should work but not "~/SomeSubDir/Profile_JSON_AppService.axd". 
            // AppRelativeCurrentExecutionFilePath looks like "~/path/filename.ext".
            // So we can easily detect if the file requested is in the root by ensuring that the last index of "/" is == 1,
            // as in the path "~/rootfile.ext", where "/" is the second character.
            // Note that the WebServiceData object is cached higher in the stack once calculated 
            int slashIndex = appRelativePath.LastIndexOf('/');
            if (slashIndex == 1) { 
                // it is a root file. Now see if its one of the two built in services 
                string name = Path.GetFileName(appRelativePath);
 
                if (name.Equals(_profileServiceFileName, StringComparison.OrdinalIgnoreCase)) {
                    return new WebServiceData(typeof(System.Web.Profile.ProfileService), false);
                }
                else if (name.Equals(_authenticationServiceFileName, StringComparison.OrdinalIgnoreCase)) { 
                    return new WebServiceData(typeof(System.Web.Security.AuthenticationService), false);
                } 
                else if (name.Equals(_roleServiceFileName, StringComparison.OrdinalIgnoreCase)) { 
                    return new WebServiceData(typeof(System.Web.Security.RoleService), false);
                } 
            }

            return null;
        } 

        internal static WebServiceData GetWebServiceData(HttpContext context, string virtualPath) { 
            return GetWebServiceData(context, virtualPath, true /*failIfNoData*/, false /*pageMethods*/, false/*inlineScript*/); 
        }
 
        private static string GetCacheKey(string virtualPath) {
            return "System.Web.Script.Services.WebServiceData:" + virtualPath;
        }
 
        internal static WebServiceData GetWebServiceData(HttpContext context, string virtualPath, bool failIfNoData, bool pageMethods) {
            return GetWebServiceData(context, virtualPath, failIfNoData, pageMethods, false /*inlineScript*/); 
        } 

        internal static WebServiceData GetWebServiceData(HttpContext context, string virtualPath, bool failIfNoData, bool pageMethods, bool inlineScript) { 
            // Make sure the path is cannonical to avoid doing more work than necessary
            virtualPath = VirtualPathUtility.ToAbsolute(virtualPath);

            string cacheKey = GetCacheKey(virtualPath); 
            WebServiceData data = context.Cache[cacheKey] as WebServiceData;
 
            // Handle the case where the virtualPath exists, for example a real asmx page. 
            if (data == null) {
                if (HostingEnvironment.VirtualPathProvider.FileExists(virtualPath)) { 
                    Type compiledType = null;
                    try {
                        compiledType = BuildManager.GetCompiledType(virtualPath);
 
                        // If we can't get the compiled type, try creating an instance (i.e. for no compile pages)
                        if (compiledType == null) { 
                            object page = BuildManager.CreateInstanceFromVirtualPath(virtualPath, typeof(System.Web.UI.Page)); 
                            if (page != null) {
                                compiledType = page.GetType(); 
                            }
                        }

                    } 
                    catch (SecurityException) {
                        // DevDiv 33708: BuildManager requires Medium trust, so we need to no-op rather than 
                        // destroying the page. 
                    }
 
                    if (compiledType != null) {
                        data = new WebServiceData(compiledType, pageMethods);
                        BuildDependencySet deps = BuildManager.GetCachedBuildDependencySet(context, virtualPath);
                        CacheDependency cd = HostingEnvironment.VirtualPathProvider.GetCacheDependency(virtualPath, deps.VirtualPaths, DateTime.Now); 
                        context.Cache.Insert(cacheKey, data, cd);
                    } 
                } 
                else if (virtualPath.EndsWith("_AppService.axd", StringComparison.OrdinalIgnoreCase)) {
                    // File does not exist, but the url may be a request for one of the three built-in services: ProfileService, AuthenticationService, RoleService 
                    data = WebServiceData.GetApplicationService(context.Request.AppRelativeCurrentExecutionFilePath);
                    if (data != null) {
                        context.Cache.Insert(cacheKey, data);
                    } 
                }
            } 
 
            if (data == null) {
                if (failIfNoData) { 
                    if (inlineScript) {
                        //DevDiv 74432: InlineScript = true fails, for WCF serviceReferences: Need an appropriate error message
                        throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, AtlasWeb.WebService_NoWebServiceDataInlineScript, virtualPath));
                    } 
                    else {
                        throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, AtlasWeb.WebService_NoWebServiceData, virtualPath)); 
                    } 
                }
                else { 
                    return null;
                }
            }
 
            return data;
        } 
 
        internal static WebServiceData GetWebServiceData(ContractDescription contract) {
            WebServiceData serviceData = new WebServiceData(); 

            // set service type
            serviceData._typeData = new WebServiceTypeData(XmlConvert.DecodeName(contract.Name), XmlConvert.DecodeName(contract.Namespace), contract.ContractType);
 
            // build list of types used in service
            Dictionary clientTypeDictionary = new Dictionary(); 
            serviceData._clientTypesDictionary = clientTypeDictionary; 
            Dictionary enumTypeDictionary = new Dictionary();
            serviceData._enumTypesDictionary = enumTypeDictionary; 
            serviceData._processedTypes = new Hashtable();
            serviceData._clientTypesProcessed = true;
            serviceData._clientTypeNameDictionary = new Dictionary();
 
            //build method dictionary
            Dictionary methodDataDictionary = new Dictionary(); 
            serviceData._methods = methodDataDictionary; 
            foreach (OperationDescription operation in contract.Operations) {
                Dictionary parameterDataDictionary = new Dictionary(); 
                bool useHttpGet = operation.Behaviors.Find() != null;
                WebServiceMethodData methodData = new WebServiceMethodData(serviceData, XmlConvert.DecodeName(operation.Name), parameterDataDictionary, useHttpGet);
                // build parameter dictionary
                MessageDescription requestMessage = operation.Messages[0]; 
                if (requestMessage != null) {
                    int numMessageParts = requestMessage.Body.Parts.Count; 
                    for (int p = 0; p < numMessageParts; p++) { 
                        MessagePartDescription messagePart = requestMessage.Body.Parts[p];
                        // DevDiv 129964:JS proxy generation fails for a WCF service that uses an untyped message 
                        // Message or its derived class are special, used for untyped operation contracts.
                        // As per the WCF team proxy generated for them should treat Message equivalent to Object type.
                        Type paramType = ReplaceMessageWithObject(messagePart.Type);
                        WebServiceParameterData parameterData = new WebServiceParameterData(XmlConvert.DecodeName(messagePart.Name), paramType, p); 
                        parameterDataDictionary[parameterData.ParameterName] = parameterData;
                        serviceData.ProcessClientType(paramType, false, true); 
                    } 
                }
                if (operation.Messages.Count > 1) { 
                    // its a two way operation, get type information from return message
                    MessageDescription responseMessage = operation.Messages[1];
                    if (responseMessage != null) {
                        if (responseMessage.Body.ReturnValue != null && responseMessage.Body.ReturnValue.Type != null) { 
                            // operation has a return type, add type to list of type proxy to generate
                            serviceData.ProcessClientType(ReplaceMessageWithObject(responseMessage.Body.ReturnValue.Type), false, true); 
                        } 
                    }
                } 

                //add known types at operation level
                for (int t = 0; t < operation.KnownTypes.Count; t++) {
                    serviceData.ProcessClientType(operation.KnownTypes[t], false, true); 
                }
 
                methodDataDictionary[methodData.MethodName] = methodData; 
            }
            serviceData._processedTypes = null; 
            return serviceData;

        }
 
        private static Type ReplaceMessageWithObject(Type t) {
            return (typeof(Message).IsAssignableFrom(t)) ? typeof(object) : t; 
        } 

        private WebServiceData() { 
        }

        private WebServiceData(WebServiceTypeData typeData) {
            _typeData = typeData; 
            _serializer = new JavaScriptSerializer(this);
#pragma warning disable 0436 
            ScriptingJsonSerializationSection.ApplicationSettings settings = new ScriptingJsonSerializationSection.ApplicationSettings(); 
#pragma warning restore 0436
            _serializer.MaxJsonLength = settings.MaxJsonLimit; 
            _serializer.RecursionLimit = settings.RecursionLimit;
            _serializer.RegisterConverters(settings.Converters);
        }
 
        // Normal ASMX Atlas codepath for creating webservice data
        internal WebServiceData(Type type, bool pageMethods) 
            : this(new WebServiceTypeData(type.Name, type.Namespace, type)) { 
            _pageMethods = pageMethods;
            // Pages don't need to have script service attribute 
            if (!_pageMethods) {
                object[] attribs = type.GetCustomAttributes(typeof(ScriptServiceAttribute), true);
                if (attribs.Length == 0) {
                    throw new InvalidOperationException(AtlasWeb.WebService_NoScriptServiceAttribute); 
                }
            } 
        } 

        // Indigo entry point for creating WebServiceData 
        internal WebServiceData(WebServiceTypeData typeData, Dictionary methods)
            : this(typeData) {
            _methods = methods;
        } 

        private void AddMethod(Dictionary methods, MethodInfo method) { 
            object[] wmAttribs = method.GetCustomAttributes(typeof(WebMethodAttribute), true); 

            // Skip it if it doesn't have the WebMethod attribute 
            if (wmAttribs.Length == 0)
                return;

            ScriptMethodAttribute sm = null; 
            object[] responseAttribs = method.GetCustomAttributes(typeof(ScriptMethodAttribute), true);
            if (responseAttribs.Length > 0) { 
                sm = (ScriptMethodAttribute)responseAttribs[0]; 
            }
 
            // Create an object to keep track of this method's data
            WebServiceMethodData wmd = new WebServiceMethodData(this, method, (WebMethodAttribute)wmAttribs[0], sm);
            methods[wmd.MethodName] = wmd;
        } 

        private void EnsureMethods() { 
            // Type will only be null for the Indigo code path 
            if (_methods != null || _typeData.Type == null)
                return; 

            // Build the method collection on demand
            lock (this) {
 
                // Need to add the methods of each type in reverse order
                List typeList = new List(); 
                Type current = _typeData.Type; 
                typeList.Add(current);
                while (current.BaseType != null) { 
                    current = current.BaseType;
                    typeList.Add(current);
                }
                Dictionary methods = new Dictionary(StringComparer.OrdinalIgnoreCase); 
                BindingFlags flags = BindingFlags.Public | BindingFlags.DeclaredOnly;
                if (_pageMethods) flags |= BindingFlags.Static; 
                else flags |= BindingFlags.Instance; 

                // Add the methods in reverse order from base to derived 
                for (int i = typeList.Count - 1; i >= 0; --i) {
                    MethodInfo[] methodInfos = typeList[i].GetMethods(flags);
                    foreach (MethodInfo method in methodInfos) {
                        AddMethod(methods, method); 
                    }
                } 
                _methods = methods; 
            }
        } 

        internal WebServiceTypeData TypeData {
            get { return _typeData; }
        } 

        internal ICollection MethodDatas { 
            get { 
                EnsureMethods();
                return _methods.Values; 
            }
        }

        internal WebServiceMethodData GetMethodData(string methodName) { 
            EnsureMethods();
 
            // Fail if the web method doesn't exist 
            WebServiceMethodData methodData = null;
            if (!_methods.TryGetValue(methodName, out methodData)) { 
                throw new ArgumentException(
                    String.Format(CultureInfo.CurrentCulture, AtlasWeb.WebService_UnknownWebMethod, methodName), "methodName");
            }
 
            EnsureClientTypesProcessed();
            return methodData; 
        } 

        private void EnsureClientTypesProcessed() { 
            if (_clientTypesProcessed)
                return;

            lock (this) { 
                if (_clientTypesProcessed)
                    return; 
                ProcessClientTypes(); 
            }
        } 

        private void ProcessClientTypes() {
            Debug.Assert(!_clientTypesProcessed, "ProcessClientTypes shouldn't be called after it has already been successfully run.");
 
            // List of types that can be instantiated on the client
            _clientTypesDictionary = new Dictionary(); 
            _enumTypesDictionary = new Dictionary(); 

            _clientTypeNameDictionary = new Dictionary(); 

            try {
                // _processedTypes is used to avoid processing a Type more than once
                _processedTypes = new Hashtable(); 

                // Process any GenerateScriptTypes on the Service type 
                ProcessIncludeAttributes((GenerateScriptTypeAttribute[])_typeData.Type.GetCustomAttributes(typeof(GenerateScriptTypeAttribute), true)); 

                foreach (WebServiceMethodData methodData in MethodDatas) { 

                    // Process any GenerateScriptTypes on the method
                    ProcessIncludeAttributes((GenerateScriptTypeAttribute[])methodData.MethodInfo.GetCustomAttributes(typeof(GenerateScriptTypeAttribute), true));
 
                    // Also add any input parameters
                    foreach (WebServiceParameterData paramData in methodData.ParameterDatas) { 
                        ProcessClientType(paramData.ParameterInfo.ParameterType); 
                    }
 
                    // Ignore return type if it uses XML instead of JSON
                    if (methodData.UseXmlResponse) continue;
                    ProcessClientType(methodData.ReturnType);
                } 

                // DevDiv 60672: Only set to true if the proxies were SUCCESSFULLY processed 
                // Only setting _clientTypesProcessed=true on success will cause us to retry creating the proxies each 
                // request when there is an exception, and so the same exception will be thrown each time.
                _clientTypesProcessed = true; 
            }
            catch {
                // If we have any exception we have to null out our caches
                _clientTypesDictionary = null; 
                _enumTypesDictionary = null;
                _clientTypeNameDictionary = null; 
                throw; 
            }
            finally { 
                _processedTypes = null;
            }
        }
 
        private void ProcessIncludeAttributes(GenerateScriptTypeAttribute[] attributes) {
            foreach (GenerateScriptTypeAttribute attribute in attributes) { 
                if (!String.IsNullOrEmpty(attribute.ScriptTypeId)) 
                    _typeResolverSpecials[attribute.Type.FullName] = attribute.ScriptTypeId;
 
                Type t = attribute.Type;
                if (t.IsPrimitive || t == typeof(object) || t == typeof(string) ||
                    t == typeof(DateTime) || t == typeof(Guid) ||
                    typeof(IEnumerable).IsAssignableFrom(t) || typeof(IDictionary).IsAssignableFrom(t) || 
                    (t.IsGenericType && t.GetGenericArguments().Length > 1) ||
                    !ObjectConverter.IsClientInstantiatableType(t, _serializer)) 
                    throw new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, 
                        AtlasWeb.WebService_InvalidGenerateScriptType, t.FullName));
 
                ProcessClientType(t, true);
            }
        }
 
        private void ProcessClientType(Type t) {
            ProcessClientType(t, false, false); 
        } 

 
        private void ProcessClientType(Type t, bool force) {
            ProcessClientType(t, force, false);
        }
 
        // Force = true is required for when we detect a GenerateScriptType which may supply a new ScriptTypeID
        private void ProcessClientType(Type t, bool force, bool isWCF) { 
            if (!force && _processedTypes.Contains(t)) 
                return;
            _processedTypes[t] = null; 

            // Keep track of all enum Types
            if (t.IsEnum) {
                WebServiceEnumData enumData = null; 
                if (isWCF) {
                    enumData = (WebServiceEnumData)WebServiceTypeData.GetWebServiceTypeData(t); 
                } 
                else {
                    enumData = new WebServiceEnumData(t.Name, t.Namespace, t, Enum.GetNames(t), Enum.GetValues(t), Enum.GetUnderlyingType(t) == typeof(ulong)); 
                }
                _enumTypesDictionary[GetTypeStringRepresentation(enumData.TypeName, false)] = enumData;
                return;
            } 

            // For generics, we only allow generic types with one parameter, which we will try to process 
            if (t.IsGenericType) { 
                if (isWCF) {
                     ProcessKnownTypes(t); 
                }
                else {
                    Type[] genericArgs = t.GetGenericArguments();
                    if (genericArgs.Length > 1) { 
                        return;
                    } 
                    ProcessClientType(genericArgs[0], false, isWCF); 
                }
            } 
            // Support arrays explicitly
            else if (t.IsArray) {
                ProcessClientType(t.GetElementType(), false, isWCF);
            } 
            else {
                // Ignore primitive types 
                // Ignore DateTime, since we have special serialization handling for it in the JavaScriptSerializer 
                // Ignore IDctionary and IEnumerables as well
                if (t.IsPrimitive || t == typeof(object) || t == typeof(string) || t == typeof(DateTime) || 
                    t == typeof(void) || t == typeof(System.Decimal) || t == typeof(Guid) ||
                    typeof(IEnumerable).IsAssignableFrom(t) || typeof(IDictionary).IsAssignableFrom(t) ||
                    (!isWCF && !ObjectConverter.IsClientInstantiatableType(t, _serializer)))
                    return; 

                // Only add it to the list of client types if it can be instantiated. 
                // pass false to skip the lock 
                if (isWCF) {
                    ProcessKnownTypes(t); 
                }
                else {
                    string typeStringRepresentation = GetTypeStringRepresentation(t.FullName, false);
                    _clientTypesDictionary[typeStringRepresentation] = new WebServiceTypeData(t.Name, t.Namespace, t); 
                    _clientTypeNameDictionary[t] = typeStringRepresentation;
 
                } 
            }
        } 

        private void ProcessKnownTypes(Type t) {
            WebServiceTypeData typeData = WebServiceTypeData.GetWebServiceTypeData(t);
            bool alreadyProcessed = false; 
            if (typeData == null) {
                // indicates a type was used that is a built-in type 
                return; 
            }
 
            // if T implments IEnumerable or IDictionary, do not include type proxy for it
            // but still continue to get known types. I.e List should ignore List
            // but process MyType
            if (!(typeof(IEnumerable).IsAssignableFrom(t) || typeof(IDictionary).IsAssignableFrom(t))) { 
                 _clientTypeNameDictionary[t] = GetTypeStringRepresentation(typeData.TypeName);
                 alreadyProcessed = ProcessTypeData(typeData); 
            } 

            if (!alreadyProcessed) { 
                IList knownTypes = WebServiceTypeData.GetKnownTypes(t, typeData);
                foreach (WebServiceTypeData knownType in knownTypes) {
                    ProcessTypeData(knownType);
                } 
            }
 
        } 

        // returns true if typeData already exists in typeDictionary 
        private bool ProcessTypeData(WebServiceTypeData typeData) {
            string typeString = GetTypeStringRepresentation(typeData.TypeName);
            bool retval = true;
            if (typeData is WebServiceEnumData) { 
                if (!_enumTypesDictionary.ContainsKey(typeString)) {
                    _enumTypesDictionary[typeString] = (WebServiceEnumData)typeData; 
                    retval = false; 
                }
            } 
            else {
                if (!_clientTypesDictionary.ContainsKey(typeString)) {
                    _clientTypesDictionary[typeString] = typeData;
                    retval = false; 
                }
            } 
            return retval; 
        }
 

        internal IEnumerable ClientTypes {
            get {
                return ClientTypeDictionary.Values; 
            }
        } 
 
        internal Dictionary ClientTypeDictionary {
            get { 
                EnsureClientTypesProcessed();
                return _clientTypesDictionary;
            }
            set { 
                _clientTypesDictionary = value;
            } 
        } 

        internal Dictionary ClientTypeNameDictionary { 
            get {
                EnsureClientTypesProcessed();
                return _clientTypeNameDictionary;
            } 
        }
 
        internal IEnumerable EnumTypes { 
            get {
                EnsureClientTypesProcessed(); 
                return _enumTypesDictionary.Values;
            }
        }
 
        internal Dictionary EnumTypeDictionary {
            get { 
                EnsureClientTypesProcessed(); 
                return _enumTypesDictionary;
            } 
            set {
                _enumTypesDictionary = value;
            }
        } 

        public override Type ResolveType(string id) { 
            WebServiceTypeData type = null; 
            if (ClientTypeDictionary.TryGetValue(id, out type)) {
                if (type != null) { 
                    return type.Type;
                }
            }
            return null; 
        }
 
        public override string ResolveTypeId(Type type) { 
            string typeString = GetTypeStringRepresentation(type.FullName);
 
            // If this type is not in the dictionary
            if (!ClientTypeDictionary.ContainsKey(typeString))
                return null;
 
            return typeString;
        } 
 
        internal string GetTypeStringRepresentation(string typeName) {
            return GetTypeStringRepresentation(typeName, true); 
        }

        internal string GetTypeStringRepresentation(string typeName, bool ensure) {
            if (ensure) { 
                EnsureClientTypesProcessed();
            } 
 
            // Handle special cases from GenerateScriptType first
            string typeString; 
            if (_typeResolverSpecials.TryGetValue(typeName, out typeString)) {
                return typeString;
            }
            return typeName; 
        }
 
        internal string GetTypeStringRepresentation(WebServiceTypeData typeData) { 
            //First check if typeData provides its string representaiton ( for WCF case)
            string typeString = typeData.StringRepresentation; 
            if (typeString == null) {
                typeString = GetTypeStringRepresentation(typeData.TypeName, true);
            }
            return typeString; 
        }
    } 
} 

// File provided for Reference Use Only by Microsoft Corporation (c) 2007.

                        

Link Menu

Network programming in C#, Network Programming in VB.NET, Network Programming in .NET
This book is available now!
Buy at Amazon US or
Buy at Amazon UK