You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			390 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			390 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //---------------------------------------------------------------------
 | |
| // <copyright file="MetadataArtifactLoaderCompositeResource.cs" company="Microsoft">
 | |
| //      Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // </copyright>
 | |
| //
 | |
| // @owner       Microsoft
 | |
| // @backupOwner Microsoft
 | |
| //---------------------------------------------------------------------
 | |
| 
 | |
| namespace System.Data.Metadata.Edm
 | |
| {
 | |
|     using System.Collections.Generic;
 | |
|     using System.Collections.ObjectModel;
 | |
|     using System.Data.Entity;
 | |
|     using System.Diagnostics;
 | |
|     using System.Globalization;
 | |
|     using System.IO;
 | |
|     using System.Reflection;
 | |
|     using System.Xml;
 | |
| 
 | |
|     /// <summary>
 | |
|     /// This class represents a collection of resources to be loaded from one
 | |
|     /// or more assemblies.
 | |
|     /// </summary>
 | |
|     internal class MetadataArtifactLoaderCompositeResource : MetadataArtifactLoader
 | |
|     {
 | |
|         /// <summary>
 | |
|         /// The list of metadata artifacts encapsulated by the composite.
 | |
|         /// </summary>
 | |
|         private readonly ReadOnlyCollection<MetadataArtifactLoaderResource> _children;
 | |
|         private readonly string _originalPath;
 | |
|         
 | |
|         /// <summary>
 | |
|         /// This constructor expects to get the paths that have potential to turn into multiple 
 | |
|         /// artifacts like
 | |
|         /// 
 | |
|         /// res://*/foo.csdl   -- could be multiple assemblies
 | |
|         /// res://MyAssembly/  -- could be multiple artifacts in the one assembly
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="path">The path to the (collection of) resources</param>
 | |
|         /// <param name="uriRegistry">The global registry of URIs</param>
 | |
|         /// <param name="resolveAssembly"></param>
 | |
|         internal MetadataArtifactLoaderCompositeResource(string originalPath, string assemblyName, string resourceName, ICollection<string> uriRegistry, MetadataArtifactAssemblyResolver resolver)
 | |
|         {
 | |
|             Debug.Assert(resolver != null);
 | |
| 
 | |
|             _originalPath = originalPath;
 | |
|             _children = LoadResources(assemblyName, resourceName, uriRegistry, resolver).AsReadOnly();
 | |
|         }
 | |
| 
 | |
|         public override string Path
 | |
|         {
 | |
|             get { return _originalPath; }
 | |
|         }
 | |
| 
 | |
|         public override bool IsComposite
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override void CollectFilePermissionPaths(List<string> paths, DataSpace spaceToGet)
 | |
|         {
 | |
|             foreach (var loader in _children)
 | |
|             {
 | |
|                 loader.CollectFilePermissionPaths(paths, spaceToGet);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Get paths to artifacts for a specific DataSpace, in the original, unexpanded 
 | |
|         /// form.
 | |
|         /// </summary>
 | |
|         /// <remarks>An assembly can embed any kind of artifact as a resource, so we simply
 | |
|         /// ignore the parameter and return the original assembly name in the URI.</remarks>
 | |
|         /// <param name="spaceToGet">The DataSpace for the artifacts of interest</param>
 | |
|         /// <returns>A List of strings identifying paths to all artifacts for a specific DataSpace</returns>
 | |
|         public override List<string> GetOriginalPaths(DataSpace spaceToGet)
 | |
|         {
 | |
|             return GetOriginalPaths();
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Get paths to artifacts for a specific DataSpace.
 | |
|         /// </summary>
 | |
|         /// <param name="spaceToGet">The DataSpace for the artifacts of interest</param>
 | |
|         /// <returns>A List of strings identifying paths to all artifacts for a specific DataSpace</returns>
 | |
|         public override List<string> GetPaths(DataSpace spaceToGet)
 | |
|         {
 | |
|             List<string> list = new List<string>();
 | |
| 
 | |
|             foreach (MetadataArtifactLoaderResource resource in _children)
 | |
|             {
 | |
|                 list.AddRange(resource.GetPaths(spaceToGet));
 | |
|             }
 | |
| 
 | |
|             return list;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Get paths to all artifacts
 | |
|         /// </summary>
 | |
|         /// <returns>A List of strings identifying paths to all resources</returns>
 | |
|         public override List<string> GetPaths()
 | |
|         {
 | |
|             List<string> list = new List<string>();
 | |
| 
 | |
|             foreach (MetadataArtifactLoaderResource resource in _children)
 | |
|             {
 | |
|                 list.AddRange(resource.GetPaths());
 | |
|             }
 | |
| 
 | |
|             return list;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Aggregates all resource streams from the _children collection
 | |
|         /// </summary>
 | |
|         /// <returns>A List of XmlReader objects; cannot be null</returns>
 | |
|         public override List<XmlReader> GetReaders(Dictionary<MetadataArtifactLoader, XmlReader> sourceDictionary)
 | |
|         {
 | |
|             List<XmlReader> list = new List<XmlReader>();
 | |
| 
 | |
|             foreach (MetadataArtifactLoaderResource resource in _children)
 | |
|             {
 | |
|                 list.AddRange(resource.GetReaders(sourceDictionary));
 | |
|             }
 | |
| 
 | |
|             return list;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Get XmlReaders for a specific DataSpace.
 | |
|         /// </summary>
 | |
|         /// <param name="spaceToGet">The DataSpace corresponding to the requested artifacts</param>
 | |
|         /// <returns>A List of XmlReader objects</returns>
 | |
|         public override List<XmlReader> CreateReaders(DataSpace spaceToGet)
 | |
|         {
 | |
|             List<XmlReader> list = new List<XmlReader>();
 | |
| 
 | |
|             foreach (MetadataArtifactLoaderResource resource in _children)
 | |
|             {
 | |
|                 list.AddRange(resource.CreateReaders(spaceToGet));
 | |
|             }
 | |
| 
 | |
|             return list;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Load all resources from the assembly/assemblies identified in the resource path.
 | |
|         /// </summary>
 | |
|         /// <param name="uriRegistry">The global registry of URIs</param>
 | |
|         /// <param name="resolveAssembly"></param>
 | |
|         /// <returns></returns>
 | |
|         private static List<MetadataArtifactLoaderResource> LoadResources(string assemblyName, string resourceName, ICollection<string> uriRegistry, MetadataArtifactAssemblyResolver resolver)
 | |
|         {
 | |
|             Debug.Assert(resolver != null);
 | |
| 
 | |
|             List<MetadataArtifactLoaderResource> loaders = new List<MetadataArtifactLoaderResource>();
 | |
|             Debug.Assert(!string.IsNullOrEmpty(assemblyName));
 | |
| 
 | |
|             if (assemblyName == MetadataArtifactLoader.wildcard)
 | |
|             {
 | |
|                 foreach (Assembly assembly in resolver.GetWildcardAssemblies())
 | |
|                 {
 | |
|                     if (AssemblyContainsResource(assembly, ref resourceName))
 | |
|                     {
 | |
|                         LoadResourcesFromAssembly(assembly, resourceName, uriRegistry, loaders, resolver);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 Assembly assembly = ResolveAssemblyName(assemblyName, resolver);
 | |
|                 LoadResourcesFromAssembly(assembly, resourceName, uriRegistry, loaders, resolver);
 | |
|             }
 | |
| 
 | |
|             if (resourceName != null && loaders.Count == 0)
 | |
|             {
 | |
|                 // they were asking for a specific resource name, and we didn't find it
 | |
|                 throw EntityUtil.Metadata(System.Data.Entity.Strings.UnableToLoadResource);
 | |
|             }
 | |
| 
 | |
|             return loaders;
 | |
|         }
 | |
| 
 | |
|         private static bool AssemblyContainsResource(Assembly assembly, ref string resourceName)
 | |
|         {
 | |
|             if (resourceName == null)
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             string[] allresources = GetManifestResourceNamesForAssembly(assembly);
 | |
|             foreach (string current in allresources)
 | |
|             {
 | |
|                 if (string.Equals(resourceName, current, StringComparison.OrdinalIgnoreCase))
 | |
|                 {
 | |
|                     resourceName = current;
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
| 
 | |
|         }
 | |
| 
 | |
|         private static void LoadResourcesFromAssembly(Assembly assembly, string resourceName, ICollection<string> uriRegistry, List<MetadataArtifactLoaderResource> loaders, MetadataArtifactAssemblyResolver resolver)
 | |
|         {
 | |
|             if (resourceName == null)
 | |
|             {
 | |
|                 LoadAllResourcesFromAssembly(assembly, uriRegistry, loaders, resolver);
 | |
|             }
 | |
|             else if (AssemblyContainsResource(assembly, ref resourceName))
 | |
|             {
 | |
|                 CreateAndAddSingleResourceLoader(assembly, resourceName, uriRegistry, loaders);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 throw EntityUtil.Metadata(System.Data.Entity.Strings.UnableToLoadResource);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static void LoadAllResourcesFromAssembly(Assembly assembly, ICollection<string> uriRegistry, List<MetadataArtifactLoaderResource> loaders, MetadataArtifactAssemblyResolver resolver)
 | |
|         {
 | |
|             Debug.Assert(assembly != null);
 | |
|             string[] allresources = GetManifestResourceNamesForAssembly(assembly);
 | |
| 
 | |
|             foreach (string resourceName in allresources)
 | |
|             {
 | |
|                 CreateAndAddSingleResourceLoader(assembly, resourceName, uriRegistry, loaders);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static void CreateAndAddSingleResourceLoader(Assembly assembly, string resourceName, ICollection<string> uriRegistry, List<MetadataArtifactLoaderResource> loaders)
 | |
|         {
 | |
|             Debug.Assert(resourceName != null);
 | |
|             Debug.Assert(assembly != null);
 | |
| 
 | |
|             string resourceUri = CreateResPath(assembly, resourceName);
 | |
|             if (!uriRegistry.Contains(resourceUri))
 | |
|             {
 | |
|                 loaders.Add(new MetadataArtifactLoaderResource(assembly, resourceName, uriRegistry));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal static string CreateResPath(Assembly assembly, string resourceName)
 | |
|         {
 | |
|             string resourceUri = string.Format(CultureInfo.InvariantCulture,
 | |
|                                                "{0}{1}{2}{3}",
 | |
|                                                resPathPrefix,
 | |
|                                                assembly.FullName,
 | |
|                                                resPathSeparator,
 | |
|                                                resourceName);
 | |
| 
 | |
|             return resourceUri;
 | |
|         }
 | |
| 
 | |
|         internal static string[] GetManifestResourceNamesForAssembly(Assembly assembly)
 | |
|         {
 | |
|             Debug.Assert(assembly != null);
 | |
| 
 | |
|             return !assembly.IsDynamic ? assembly.GetManifestResourceNames() : new string[0];
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Load all resources from a specific assembly.
 | |
|         /// </summary>
 | |
|         /// <param name="fullName">The full name identifying the assembly to
 | |
|         /// load resources from</param>
 | |
|         /// <param name="uriRegistry">The global registry of URIs</param>
 | |
|         /// <param name="resolveAssembly">delegate for resolve the assembly</param>
 | |
|         private static Assembly ResolveAssemblyName(string assemblyName, MetadataArtifactAssemblyResolver resolver)
 | |
|         {
 | |
|             Debug.Assert(resolver != null);
 | |
| 
 | |
|             AssemblyName referenceName = new AssemblyName(assemblyName);
 | |
|             Assembly assembly;
 | |
|             if(!resolver.TryResolveAssemblyReference(referenceName, out assembly))
 | |
|             {
 | |
|                 throw new FileNotFoundException(Strings.UnableToResolveAssembly(assemblyName));
 | |
|             }
 | |
| 
 | |
|             return assembly;
 | |
|         }
 | |
| 
 | |
| 
 | |
|         internal static MetadataArtifactLoader CreateResourceLoader(string path, ExtensionCheck extensionCheck, string validExtension, ICollection<string> uriRegistry, MetadataArtifactAssemblyResolver resolver)
 | |
|         {
 | |
|             Debug.Assert(path != null);
 | |
|             Debug.Assert(MetadataArtifactLoader.PathStartsWithResPrefix(path));
 | |
| 
 | |
|             // if the supplied path ends with a separator, or contains only one
 | |
|             // segment (i.e., the name of an assembly, or the wildcard character),
 | |
|             // create a composite loader that can extract resources from one or 
 | |
|             // more assemblies
 | |
|             //
 | |
|             bool createCompositeResLoader = false;
 | |
|             string assemblyName = null;
 | |
|             string resourceName = null;
 | |
|             ParseResourcePath(path, out assemblyName, out resourceName);
 | |
|             createCompositeResLoader = (assemblyName != null) && (resourceName == null || assemblyName.Trim() == wildcard);
 | |
| 
 | |
|             ValidateExtension(extensionCheck, validExtension, resourceName);
 | |
| 
 | |
|             if (createCompositeResLoader)
 | |
|             {
 | |
|                 return new MetadataArtifactLoaderCompositeResource(path, assemblyName, resourceName, uriRegistry, resolver);
 | |
|             }
 | |
| 
 | |
|             Debug.Assert(!string.IsNullOrEmpty(resourceName), "we should not get here is the resourceName is null");
 | |
|             Assembly assembly = ResolveAssemblyName(assemblyName, resolver);
 | |
|             return new MetadataArtifactLoaderResource(assembly, resourceName, uriRegistry);
 | |
|         }
 | |
| 
 | |
|         private static void ValidateExtension(ExtensionCheck extensionCheck, string validExtension, string resourceName)
 | |
|         {
 | |
|             if (resourceName == null)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             // the supplied path represents a single resource
 | |
|             //
 | |
|             switch (extensionCheck)
 | |
|             {
 | |
|                 case ExtensionCheck.Specific:
 | |
|                     MetadataArtifactLoader.CheckArtifactExtension(resourceName, validExtension);
 | |
|                     break;
 | |
| 
 | |
|                 case ExtensionCheck.All:
 | |
|                     if (!MetadataArtifactLoader.IsValidArtifact(resourceName))
 | |
|                     {
 | |
|                         throw EntityUtil.Metadata(Strings.InvalidMetadataPath);
 | |
|                     }
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Splits the supplied path into the assembly portion and the resource
 | |
|         /// part (if any)
 | |
|         /// </summary>
 | |
|         /// <param name="path">The resource path to parse</param>
 | |
|         /// <returns>An array of (two) strings containing the assembly name
 | |
|         /// and the resource name</returns>
 | |
|         private static void ParseResourcePath(string path, out string assemblyName, out string resourceName)
 | |
|         {
 | |
|             // Extract the components from the path
 | |
|             int prefixLength = MetadataArtifactLoader.resPathPrefix.Length;
 | |
| 
 | |
|             string[] result = path.Substring(prefixLength).Split(
 | |
|                                                 new string[] { 
 | |
|                                                     MetadataArtifactLoader.resPathSeparator,
 | |
|                                                     MetadataArtifactLoader.altPathSeparator 
 | |
|                                                 },
 | |
|                                                 StringSplitOptions.RemoveEmptyEntries
 | |
|                                             );
 | |
| 
 | |
|             if (result.Length == 0 || result.Length > 2)
 | |
|             {
 | |
|                 throw EntityUtil.Metadata(Strings.InvalidMetadataPath);
 | |
|             }
 | |
| 
 | |
|             if (result.Length >= 1)
 | |
|             {
 | |
|                 assemblyName = result[0];
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 assemblyName = null;
 | |
|             }
 | |
| 
 | |
|             if (result.Length == 2)
 | |
|             {
 | |
|                 resourceName = result[1];
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 resourceName = null;
 | |
|             }
 | |
| 
 | |
|         }
 | |
|     }
 | |
| }
 |