如何找出程序集的项目目录?

时间:2013-03-23 16:18:35

标签: c# projects-and-solutions .net-assembly

为了启用精心设计的调试方案,我需要我的程序集来确定源文件在运行时的位置。

Assembly对象不知道此信息,可能不应该知道。 .pdb文件显然知道,但我不知道如何解析它。所以我想也许我可以在构建时用一个属性来标记我的程序集,这有点影响:

[assembly: AssemblyMetadata("ProjectDirectory", $(ProjectDir))]

我无法使用当前目录,因为它在调试期间由IIS设置为某个临时目录。我也不想对目录进行硬编码。

我到目前为止最接近解决此问题的方法是使用Assembly.CodeBase属性。它指向构建IIS解决方案的目录(Solution/IISProject/bin/Debug/,而不是/Solution/source/MyLibrary/),但不是我的项目目录。一个hacky解决方案从那里上升几个级别,然后下载几个级别回到项目文件夹。如果可能的话,我非常想避免这种黑客攻击。

2 个答案:

答案 0 :(得分:0)

要启用这些类型的方案,通常最佳做法是将这些目录放在web.config中的设置中。所以你可以根据给定的情况说,使用这个目录而不是这个目录。通常人们使用if(IsDebugging)使用这个目录,否则使用这个目录。

答案 1 :(得分:0)

经过大量的灵魂搜索,我设法收集了足够的inspiration来撰写以下代码段。我们的想法是使用.pdb文件来发现源代码的位置。您需要引用ISymWrapper.dll并以32位模式进行编译。如果有一个更简单的方法,我会很高兴听到它。

using System;
using System.Diagnostics.SymbolStore;
using System.IO;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Linq;
public static class AssemblyDirectoryInfo
{
    private const string
        ImporterId = "7DAC8207-D3AE-4c75-9B67-92801A497D44",
        DispenserId = "809c652e-7396-11d2-9771-00a0c9b4d50c",
        DispenserClassId = "e5cb7a31-7512-11d2-89ce-0080c792e5d8";

    [Guid(ImporterId), InterfaceType(ComInterfaceType.InterfaceIsIUnknown), ComVisible(true)]
    public interface IMetaDataImport{}

    [Guid(DispenserId), InterfaceType(ComInterfaceType.InterfaceIsIUnknown), ComVisible(true)]
    interface IMetaDataDispenser
    {
        void DefineScope();

        void OpenScope([In, MarshalAs(UnmanagedType.LPWStr)] String szScope, [In] Int32 dwOpenFlags,
                       [In] ref Guid riid, [Out, MarshalAs(UnmanagedType.IUnknown)] out Object punk);
    }

    [DllImport("ole32.dll")]
    private static extern int CoCreateInstance([In] ref Guid guid,
        [In, MarshalAs(UnmanagedType.IUnknown)] Object pUnkOuter,
        [In] uint dwClsContext,
        [In] ref Guid riid,
        [Out, MarshalAs(UnmanagedType.Interface)] out Object ppv);

    public static DirectoryInfo GetAssemblyCodeBase(Assembly assembly)
    {
        var file = new FileInfo(assembly.Location);

        Guid dispenserClassId = new Guid(DispenserClassId),
             importerIid = new Guid(ImporterId),
             dispenserIid = new Guid(DispenserId);

        object dispenser, importer;

        CoCreateInstance(ref dispenserClassId, null, 1, ref dispenserIid, out dispenser);
        ((IMetaDataDispenser)dispenser).OpenScope(file.FullName, 0, ref importerIid, out importer);

        var ptr = Marshal.GetComInterfaceForObject(importer, typeof(IMetaDataImport));
        var reader = new SymBinder().GetReader(ptr, file.FullName, file.DirectoryName);
        return reader.GetDocuments()
                     .Select(d => d.URL)
                     .Where(v => !string.IsNullOrEmpty(v))
                     .OrderBy(v => v.Length)
                     .Select(v => new FileInfo(v).Directory)
                     .First();
    }
}