我提前为一个简单问题的长篇描述道歉,但我想确保人们正确理解我想要做的事情。
我正在编写一个工具,可以读取由SqlMetal生成的文件,并创建一个包含简单插入,更新,删除和选择方法的类,然后可以将这些方法公开给Web服务。这里的主要优点是,如果表格发生变化,我只需重新运行该工具,并且与数据库相关的代码会自动更新,并且使用它的任何地方都会产生编译错误,从而可以轻松追踪手动更改所需的位置。做成。例如,如果我有一个Customer表,其中包含以下字段:
我希望能够生成Insert和Delete方法,如下所示:
// I only want non-database-generated fields to be parameters here.
public static Customer InsertCustomer(String firstName, String lastName)
{
...
}
// I only want the primary key fields to be parameters here.
public static int DeleteCustomer(int customerId)
{
...
}
我正在使用SqlMetal生成Customer类。现在我要做的是将.cs文件读入我的新工具,以便使用上述方法创建另一个类。然后,可以将此新类公开给Web服务,以授予对此功能的访问权限,而无需公开底层数据库。我正在使用NRefactory读取SqlMetal生成的文件,到目前为止,它进展顺利,但我遇到了试图读取Customer类的属性属性的障碍。
SqlMetal使用ColumnAttribute生成其类,以标识从数据库列派生的每个属性。 ColumnAttribute将有许多参数来描述数据库列的属性。在上面的例子中,它会生成如下内容:
...
[Column(Name="customerId", Storage="_CustomerId, DbType="INT NOT NULL IDENTITY", IsPrimaryKey=true, IsDbGenerated=true)]
public int CustomerId
{
...
}
[Column(Name="firstName", Storage="_FirstName", DbType="NVarChar(100) NOT NULL", CanBeNull=false)]
public String FirstName
{
...
}
[Column(Name="lastName", Storage="_LastName", DbType="NVarChar(100) NOT NULL", CanBeNull=false)]
public String LastName
{
...
}
...
正如您所看到的,SqlMetal为我提供了我需要的属性,以便识别哪些列是数据库生成的,哪些列是主键的一部分。因此,当我将此文件读入NRefactory并解析类型时,我希望能够获得所有这些信息。但是,我发现虽然我可以访问ColumnAttribute,但它上面的所有参数都未解析,因此无法通过NamedArguments或PositionalArguments属性访问。
这是我的代码:
SyntaxTree syntaxTree = ...;
foreach(AstNode tableNode in syntaxTree.Children)
{
ResolveResult result = resolver.Resolve(tableNode);
var properties = result.Type.GetProperties();
foreach (IProperty p in properties)
{
var attributes = p.Attributes;
bool isPrimaryKeyField = false;
bool isDbGenerated = false;
bool isColumn = false;
foreach (IAttribute attr in attributes)
{
if (attr.AttributeType.Name == "Column")
{
isColumn = true;
foreach (var arg in attr.NamedArguments) // NamedArguments contains no items.
{
if (arg.Key.Name == "IsPrimaryKey")
{
isPrimaryKeyField = (bool)arg.Value.ConstantValue == true;
}
if (arg.Key.Name == "IsDbGenerated")
{
isDbGenerated = (bool)arg.Value.ConstantValue == true;
}
}
}
}
if (isColumn)
{
... // Create a parameter as appropriate.
}
}
}
这一切都有效,直到我尝试遍历IAttribute.NamedArguments,因为该集合不包含任何元素。但是,当我通过调试器并检查'attr'的值时,我可以看到有一个名为'unresolved'的私有变量,它包含我想要的所有参数的列表但是我找不到访问它的方法通过代码。
如何获取此“未解析”变量的内容?我是否需要使用旋转变压器做更多事情?这是我第一次使用NRefactory,所以我还不太熟悉所有的细微差别。我一直很难找到一个在Google上达到这种深度的例子,而我在NRefactory上看到的文档似乎并没有涵盖它。任何帮助将不胜感激。
答案 0 :(得分:2)
我明白了。在解析SyntaxTree之前,我需要将System.Data.Linq的程序集加载到IProjectContent中。
...
CecilLoader loader = new CecilLoader();
Assembly[] assembliesToLoad = {
...
typeof(System.Data.Linq.Mapping.ColumnAttribute).Assembly
...};
IUnresolvedAssembly[] projectAssemblies = new IUnresolvedAssembly[assembliesToLoad.Length];
for(int i = 0; i < assembliesToLoad.Length; i++)
{
projectAssemblies[i] = loader.LoadAssemblyFile(assembliesToLoad[i].Location);
}
IProjectContent project = new CSharpProjectContent();
project = project.AddAssemblyReferences(projectAssemblies);
...