我有一系列的通用响应对象,这些对象带有一个抽象类的属性。 NSwag和NJsonSchema生成带有抽象类的架构,这会产生问题。具体的类很容易通过反射来确定,但是,似乎没有一种干净的方法来使NJsonSchema用适当的具体类来代替抽象类型。正确的方法是什么?
public abstract class AppRequest<TData> {
public Guid RequestId { get; set; }
}
public class AppResponse<TData> {
public TData Data { get; set; }
public AppRequest<TData> OriginalRequest { get; set; }
}
public class User {
....
}
public class UserRequest: AppRequest<User> {
public Guid UserId { get; set; }
}
NSwag将响应对象生成为AppResponseOfUser
,这很好,但是它表示OriginalRequest
属性是AppRequestOfUser,并且它是抽象的。我想创建一个SchemaProcessor来将此AppRequestOfUser重新映射到UserRequest。像这样:
public class MySchemaProcessor
: ISchemaProcessor
{
public async Task ProcessAsync(SchemaProcessorContext context)
{
if (context.Type.IsGenericOf(typeof(AppResponse<>)))
{
var modelType = context.Type.GenericTypeArguments[0];
var abstractRequestType = typeof(AppRequest<>).MakeGenericType(modelType);
var actualRequestType = modelType.Assembly.GetTypes()
.Single(t => t.IsClass && t.BaseType == abstractRequestType);
var requestSchema = await JsonSchema4.FromTypeAsync(actualRequestType);
var originalRequestProperty = context.Schema.Properties["originalRequest"];
originalRequestProperty.IsReadOnly = true;
originalRequestProperty.IsAbstract = false;
// CHANGE PROPERTY TYPE HERE!!!
}
}
}
不幸的是,NJsonSchema似乎并不十分灵活,并且没有明确的方法来做到这一点。我不想使用鉴别属性。我想重新映射到适当的具体类型。
答案 0 :(得分:0)
万一有人想知道,这是最终的解决方案:
var classesToMap = typeof(Startup)
.Assembly
.GetTypes()
.Where(t => t.IsClass && t.BaseType.IsGenericOf(typeof(AppRequest<>)));
var settings = new JsonSchemaGeneratorSettings()
{
FlattenInheritanceHierarchy = true,
};
foreach (var type in classesToMap)
{
var actualSchema = JsonSchema4.FromTypeAsync(type,settings).Result;
options.TypeMappers.Add(new ObjectTypeMapper(type.BaseType, actualSchema));
}