ModuleScope
Summary description for ModuleScope.
            
                using Castle.Core.Internal;
using Castle.DynamicProxy.Generators;
using System;
using System.Collections.Generic;
using System.IO;
using System.Reflection;
using System.Reflection.Emit;
using System.Resources;
namespace Castle.DynamicProxy
{
    public class ModuleScope
    {
        public static readonly string DEFAULT_FILE_NAME = "CastleDynProxy2.dll";
        public static readonly string DEFAULT_ASSEMBLY_NAME = "DynamicProxyGenAssembly2";
        private ModuleBuilder moduleBuilderWithStrongName;
        private ModuleBuilder moduleBuilder;
        private readonly string strongAssemblyName;
        private readonly string weakAssemblyName;
        private readonly string strongModulePath;
        private readonly string weakModulePath;
        private readonly Dictionary<CacheKey, Type> typeCache = new Dictionary<CacheKey, Type>();
        private readonly Lock cacheLock = Lock.Create();
        private readonly object moduleLocker = new object();
        private readonly bool savePhysicalAssembly;
        private readonly bool disableSignedModule;
        private readonly INamingScope namingScope;
        public INamingScope NamingScope => namingScope;
        public Lock Lock => cacheLock;
        public ModuleBuilder StrongNamedModule => moduleBuilderWithStrongName;
        public string StrongNamedModuleName => Path.GetFileName(strongModulePath);
        public ModuleBuilder WeakNamedModule => moduleBuilder;
        public string WeakNamedModuleName => Path.GetFileName(weakModulePath);
        public ModuleScope()
            : this(false, false)
        {
        }
        public ModuleScope(bool savePhysicalAssembly)
            : this(savePhysicalAssembly, false)
        {
        }
        public ModuleScope(bool savePhysicalAssembly, bool disableSignedModule)
            : this(savePhysicalAssembly, disableSignedModule, DEFAULT_ASSEMBLY_NAME, DEFAULT_FILE_NAME, DEFAULT_ASSEMBLY_NAME, DEFAULT_FILE_NAME)
        {
        }
        public ModuleScope(bool savePhysicalAssembly, bool disableSignedModule, string strongAssemblyName, string strongModulePath, string weakAssemblyName, string weakModulePath)
            : this(savePhysicalAssembly, disableSignedModule, new NamingScope(), strongAssemblyName, strongModulePath, weakAssemblyName, weakModulePath)
        {
        }
        public ModuleScope(bool savePhysicalAssembly, bool disableSignedModule, INamingScope namingScope, string strongAssemblyName, string strongModulePath, string weakAssemblyName, string weakModulePath)
        {
            this.savePhysicalAssembly = savePhysicalAssembly;
            this.disableSignedModule = disableSignedModule;
            this.namingScope = namingScope;
            this.strongAssemblyName = strongAssemblyName;
            this.strongModulePath = strongModulePath;
            this.weakAssemblyName = weakAssemblyName;
            this.weakModulePath = weakModulePath;
        }
        public Type GetFromCache(CacheKey key)
        {
            typeCache.TryGetValue(key, out Type value);
            return value;
        }
        public void RegisterInCache(CacheKey key, Type type)
        {
            typeCache[key] = type;
        }
        public static byte[] GetKeyPair()
        {
            using (Stream stream = typeof(ModuleScope).GetTypeInfo().get_Assembly().GetManifestResourceStream("Castle.DynamicProxy.DynProxy.snk")) {
                if (stream == null)
                    throw new MissingManifestResourceException("Should have a Castle.DynamicProxy.DynProxy.snk as an embedded resource, so Dynamic Proxy could sign generated assembly");
                int num = (int)stream.Length;
                byte[] array = new byte[num];
                stream.Read(array, 0, num);
                return array;
            }
        }
        public ModuleBuilder ObtainDynamicModule(bool isStrongNamed)
        {
            if (isStrongNamed)
                return ObtainDynamicModuleWithStrongName();
            return ObtainDynamicModuleWithWeakName();
        }
        public ModuleBuilder ObtainDynamicModuleWithStrongName()
        {
            if (disableSignedModule)
                throw new InvalidOperationException("Usage of signed module has been disabled. Use unsigned module or enable signed module.");
            lock (moduleLocker) {
                if ((object)moduleBuilderWithStrongName == null)
                    moduleBuilderWithStrongName = CreateModule(true);
                return moduleBuilderWithStrongName;
            }
        }
        public ModuleBuilder ObtainDynamicModuleWithWeakName()
        {
            lock (moduleLocker) {
                if ((object)moduleBuilder == null)
                    moduleBuilder = CreateModule(false);
                return moduleBuilder;
            }
        }
        private ModuleBuilder CreateModule(bool signStrongName)
        {
            AssemblyName assemblyName = GetAssemblyName(signStrongName);
            string name = signStrongName ? StrongNamedModuleName : WeakNamedModuleName;
            return AssemblyBuilder.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Run).DefineDynamicModule(name);
        }
        private AssemblyName GetAssemblyName(bool signStrongName)
        {
            AssemblyName assemblyName = new AssemblyName {
                Name = (signStrongName ? strongAssemblyName : weakAssemblyName)
            };
            if (signStrongName)
                assemblyName.SetPublicKey(InternalsVisible.DynamicProxyGenAssembly2PublicKey);
            return assemblyName;
        }
        public TypeBuilder DefineType(bool inSignedModulePreferably, string name, TypeAttributes flags)
        {
            return ObtainDynamicModule(!disableSignedModule & inSignedModulePreferably).DefineType(name, flags);
        }
    }
}