//---------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//
// @owner [....]
// @backupOwner [....]
//---------------------------------------------------------------------
namespace System.Data.Metadata.Edm
{
using System.Collections.Generic;
using System.Data.EntityModel.SchemaObjectModel;
using System.Diagnostics;
using System.IO;
using System.Reflection;
using System.Xml;
///
/// This class represents one resource item to be loaded from an assembly.
///
internal class MetadataArtifactLoaderResource : MetadataArtifactLoader, IComparable
{
private readonly bool _alreadyLoaded = false;
private readonly Assembly _assembly;
private readonly string _resourceName;
///
/// Constructor - loads the resource stream
///
/// The path to the resource to load
/// The global registry of URIs
internal MetadataArtifactLoaderResource(Assembly assembly, string resourceName, ICollection uriRegistry)
{
Debug.Assert(assembly != null);
Debug.Assert(resourceName != null);
_assembly = assembly;
_resourceName = resourceName;
string tempPath = MetadataArtifactLoaderCompositeResource.CreateResPath(_assembly, _resourceName);
_alreadyLoaded = uriRegistry.Contains(tempPath);
if (!_alreadyLoaded)
{
uriRegistry.Add(tempPath);
// '_alreadyLoaded' is not set because while we would like to prevent
// other instances of MetadataArtifactLoaderFile that wrap the same
// _path from being added to the list of paths/readers, we do want to
// include this particular instance.
}
}
public override string Path
{
get
{
return MetadataArtifactLoaderCompositeResource.CreateResPath(_assembly, _resourceName);
}
}
///
/// Implementation of IComparable.CompareTo()
///
/// The object to compare to
/// 0 if the loaders are "equal" (i.e., have the same _path value)
public int CompareTo(object obj)
{
MetadataArtifactLoaderResource loader = obj as MetadataArtifactLoaderResource;
if (loader != null)
{
return string.Compare(Path, loader.Path, StringComparison.OrdinalIgnoreCase);
}
Debug.Assert(false, "object is not a MetadataArtifactLoaderResource");
return -1;
}
///
/// Equals() returns true if the objects have the same _path value
///
/// The object to compare to
/// true if the objects have the same _path value
public override bool Equals(object obj)
{
return this.CompareTo(obj) == 0;
}
///
/// GetHashCode override that defers the result to the _path member variable.
///
///
public override int GetHashCode()
{
return Path.GetHashCode();
}
public override void CollectFilePermissionPaths(List paths, DataSpace spaceToGet)
{
// does not apply
}
///
/// Get paths to artifacts for a specific DataSpace.
///
/// The DataSpace for the artifacts of interest
/// A List of strings identifying paths to all artifacts for a specific DataSpace
public override List GetPaths(DataSpace spaceToGet)
{
List list = new List();
if (!_alreadyLoaded && MetadataArtifactLoader.IsArtifactOfDataSpace(Path, spaceToGet))
{
list.Add(Path);
}
return list;
}
///
/// Get paths to all artifacts
///
/// A List of strings identifying paths to all resources
public override List GetPaths()
{
List list = new List();
if (!_alreadyLoaded)
{
list.Add(Path);
}
return list;
}
///
/// Create and return an XmlReader around the resource represented by this instance.
///
/// A List of XmlReaders for all resources
public override List GetReaders(Dictionary sourceDictionary)
{
List list = new List();
if (!_alreadyLoaded)
{
XmlReader reader = CreateReader();
list.Add(reader);
if (sourceDictionary != null)
{
sourceDictionary.Add(this, reader);
}
}
return list;
}
private XmlReader CreateReader()
{
Stream stream = LoadResource();
XmlReaderSettings readerSettings = Schema.CreateEdmStandardXmlReaderSettings();
// close the stream when the xmlreader is closed
// now the reader owns the stream
readerSettings.CloseInput = true;
// we know that we aren't reading a fragment
readerSettings.ConformanceLevel = ConformanceLevel.Document;
XmlReader reader = XmlReader.Create(stream, readerSettings);
// cannot set the base URI because res:// URIs cause the schema parser
// to choke
return reader;
}
///
/// Create and return an XmlReader around the resource represented by this instance
/// if it is of the requested DataSpace type.
///
/// The DataSpace corresponding to the requested artifacts
/// A List of XmlReader objects
public override List CreateReaders(DataSpace spaceToGet)
{
List list = new List();
if (!_alreadyLoaded)
{
if (MetadataArtifactLoader.IsArtifactOfDataSpace(Path, spaceToGet))
{
XmlReader reader = CreateReader();
list.Add(reader);
}
}
return list;
}
///
/// This method parses the path to the resource and attempts to load it.
/// The method also accounts for the wildcard assembly name.
///
private Stream LoadResource()
{
Stream resourceStream;
if (TryCreateResourceStream(out resourceStream))
{
return resourceStream;
}
throw EntityUtil.Metadata(System.Data.Entity.Strings.UnableToLoadResource);
}
private bool TryCreateResourceStream(out Stream resourceStream)
{
resourceStream = _assembly.GetManifestResourceStream(_resourceName);
return resourceStream != null;
}
}
}