我有一系列通用响应对象,这些对象返回的属性是一个抽象类。 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似乎不是很灵活,也没有明确的方法来做到这一点。 我不想使用鉴别器属性。 我想重新映射到适当的具体类型。
如果有人想知道,这是最终的解决方案:
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));
}