欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

.NET9中Swagger平替Scalar使用詳解

 更新時(shí)間:2024年11月28日 09:28:11   作者:IT規(guī)劃師  
本文詳細(xì)介紹了Swagger中常用功能在Scalar中的使用,包括版本說(shuō)明、接口分類、接口及參數(shù)描述、枚舉類型、文件上傳和JWT認(rèn)證等,并提供相關(guān)代碼示例和效果展示,感興趣的朋友跟隨小編一起看看吧

本文分享Swagger中常用功能在Scalar中的使用,包括版本說(shuō)明、接口分類、接口及參數(shù)描述、枚舉類型、文件上傳和JWT認(rèn)證等,并提供相關(guān)代碼示例和效果展示,以及可能遇到的問(wèn)題和解決方案。

書(shū)接上回,上一章介紹了Swagger代替品Scalar,在使用中遇到不少問(wèn)題,今天單獨(dú)分享一下之前Swagger中常用的功能如何在Scalar中使用。

下面我們將圍繞文檔版本說(shuō)明、接口分類、接口描述、參數(shù)描述、枚舉類型、文件上傳、JWT認(rèn)證等方面詳細(xì)講解。

01、版本說(shuō)明

我們先來(lái)看看默認(rèn)添加后是什么樣子的。

public static void Main(string[] args)
{
    var builder = WebApplication.CreateBuilder(args);
    builder.Services.AddControllers();
    builder.Services.AddOpenApi();
    var app = builder.Build();
    app.MapScalarApiReference();
    app.MapOpenApi();
    app.UseAuthorization();
    app.MapControllers();
    app.Run();
}

效果如下:

我們可以直接修改builder.Services.AddOpenApi()這行代碼,修改這塊描述,代碼如下:

builder.Services.AddOpenApi(options =>
{
    options.AddDocumentTransformer((document, context, cancellationToken) =>
    {
        document.Info = new()
        {
            Title = "訂單微服務(wù)",
            Version = "v1",
            Description = "訂單相關(guān)接口"
        };
        return Task.CompletedTask;
    });
});

我們?cè)賮?lái)看看效果。

02、接口分類

通過(guò)上圖可以看到菜單左側(cè)排列著所有接口,現(xiàn)在我們可以通過(guò)Tags特性對(duì)接口進(jìn)行分類,如下圖我們把增刪改查4個(gè)方法分為冪等接口和非冪等接口兩類,如下圖:

然后我們看看效果,如下圖:

03、接口描述

之前使用Swagger我們都是通過(guò)生成的注釋XML來(lái)生成相關(guān)接口描述,現(xiàn)在則是通過(guò)編碼的方式設(shè)置元數(shù)據(jù)來(lái)生成相關(guān)描述。

可以通過(guò)EndpointSummary設(shè)置接口摘要,摘要不設(shè)置默認(rèn)為接口url,通過(guò)EndpointDescription設(shè)置接口描述,代碼如下:

//獲取
[HttpGet(Name = "")]
[Tags("冪等接口")]
[EndpointDescription("獲取訂單列表")]
public IEnumerable<Order> Get()
{
    return null;
}
//刪除
[HttpDelete(Name = "{id}")]
[Tags("冪等接口")]
[EndpointSummary("刪除訂單")]
[EndpointDescription("根據(jù)訂單id,刪除相應(yīng)訂單")]
public bool Delete(string id)
{
    return true;
}

運(yùn)行效果如下:

04、參數(shù)描述

同樣可以通過(guò)Description特性來(lái)設(shè)置參數(shù)的描述,并且此特性可以直接作用于接口中參數(shù)之前,同時(shí)也支持作用于屬性上,可以看看下面示例代碼。

public class Order
{
    [property: Description("創(chuàng)建日期")]
    public DateOnly Date { get; set; }
    [property: Required]
    [property: DefaultValue(120)]
    [property: Description("訂單價(jià)格")]
    public int Price { get; set; }
    [property: Description("訂單折扣價(jià)格")]
    public int PriceF => (int)(Price * 0.5556);
    [property: Description("商品名稱")]
    public string? Name { get; set; }
}
[HttpPut(Name = "{id}")]
[Tags("非冪等接口")]
public bool Put([Description("訂單Id")] string id, Order order)
{
    return true;
}

效果如下圖:

從上圖可以發(fā)現(xiàn)除了描述還有默認(rèn)值、必填項(xiàng)、可空等字樣,這些是通過(guò)其他元數(shù)據(jù)設(shè)置的,對(duì)于屬性還有以下元數(shù)據(jù)可以進(jìn)行設(shè)置。

05、枚舉類型

對(duì)于枚舉類型,我們正常關(guān)注兩個(gè)東西,其一為枚舉項(xiàng)以int類型展示還是以字符串展示,其二為枚舉項(xiàng)顯示描述信息。

關(guān)于第一點(diǎn)比較簡(jiǎn)單只要對(duì)枚舉類型使用JsonStringEnumConverter即可,代碼如下:

[JsonConverter(typeof(JsonStringEnumConverter<OrderStatus>))]
public enum OrderStatus
{
    [Description("等待處理")]
    Pending = 1,
    [Description("處理中")]
    Processing = 2,
    [Description("已發(fā)貨")]
    Shipped = 3,
    [Description("已送達(dá)")]
    Delivered = 4,
}

效果如下:

通過(guò)上圖也可以引發(fā)關(guān)于第二點(diǎn)的需求,如何對(duì)每個(gè)枚舉項(xiàng)添加描述信息。

要達(dá)到這個(gè)目標(biāo)需要做兩件事,其一給每個(gè)枚舉項(xiàng)通過(guò)Description添加元數(shù)據(jù)定義,其二我們要修改文檔的數(shù)據(jù)結(jié)構(gòu)Schema。

修改builder.Services.AddOpenApi(),通過(guò)AddSchemaTransformer方法修改文檔數(shù)據(jù)結(jié)構(gòu),代碼如下:

options.AddSchemaTransformer((schema, context, cancellationToken) =>
{
    //找出枚舉類型
    if (context.JsonTypeInfo.Type.BaseType == typeof(Enum))
    {
        var list = new List<IOpenApiAny>();
        //獲取枚舉項(xiàng)
        foreach (var enumValue in schema.Enum.OfType<OpenApiString>())
        {
            //把枚舉項(xiàng)轉(zhuǎn)為枚舉類型
            if (Enum.TryParse(context.JsonTypeInfo.Type, enumValue.Value, out var result))
            {
                //通過(guò)枚舉擴(kuò)展方法獲取枚舉描述
                var description = ((Enum)result).ToDescription();
                //重新組織枚舉值展示結(jié)構(gòu)
                list.Add(new OpenApiString($"{enumValue.Value} - {description}"));
            }
            else
            {
                list.Add(enumValue);
            }
        }
        schema.Enum = list;
    }
    return Task.CompletedTask;
});

我們?cè)賮?lái)看看結(jié)果。

但是這也帶來(lái)了一個(gè)問(wèn)題,就是參數(shù)的默認(rèn)值也是添加描述的格式,顯然這樣的數(shù)據(jù)格式作為參數(shù)肯定是錯(cuò)誤的,因此我們需要自己注意,如下圖。目前我也沒(méi)有發(fā)現(xiàn)更好的方式即可以把每項(xiàng)枚舉描述加上,又不影響參數(shù)默認(rèn)值,有解決方案的希望可以不吝賜教。

06、文件上傳

下面我們來(lái)看看文件上傳怎么用,直接上代碼:

[HttpPost("upload/image")]
[EndpointDescription("圖片上傳接口")]
[DisableRequestSizeLimit]
public bool UploadImgageAsync(IFormFile file)
{
    return true;
}

然后我們測(cè)試一下效果。

首先我們可以看到請(qǐng)求示例中相關(guān)信息,這個(gè)相當(dāng)于告訴我們后面要怎么選擇文件上傳,我們繼續(xù)點(diǎn)擊Test Request。

首先請(qǐng)求體需要選擇multipart/form-data,上圖請(qǐng)求示例中已經(jīng)給出提示。

然后設(shè)置key為file,上圖請(qǐng)求示例中已經(jīng)給出提示,然后點(diǎn)擊File上傳圖片,最后點(diǎn)擊Send即可。

07、JWT認(rèn)證

最后我們來(lái)看看如何使用JWT認(rèn)證,

首先我們需要注入AddAuthentication及AddJwtBearer,具體代碼如下:

public class JwtSettingOption
{
    //這個(gè)字符數(shù)量有要求,不能隨便寫(xiě),否則會(huì)報(bào)錯(cuò)
    public static string Secret { get; set; } = "123456789qwertyuiopasdfghjklzxcb";
    public static string Issuer { get; set; } = "asdfghjkkl";
    public static string Audience { get; set; } = "zxcvbnm";
    public static int Expires { get; set; } = 120;
    public static string RefreshAudience { get; set; } = "zxcvbnm.2024.refresh";
    public static int RefreshExpires { get; set; } = 10080;
}
builder.Services.AddAuthentication(options =>
{
    options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
    options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
}).AddJwtBearer(options =>
{
    //取出私鑰
    var secretByte = Encoding.UTF8.GetBytes(JwtSettingOption.Secret);
    options.TokenValidationParameters = new TokenValidationParameters()
    {
        //驗(yàn)證發(fā)布者
        ValidateIssuer = true,
        ValidIssuer = JwtSettingOption.Issuer,
        //驗(yàn)證接收者
        ValidateAudience = true,
        ValidAudiences = new List<string> { JwtSettingOption.Audience, JwtSettingOption.Audience },
        //驗(yàn)證是否過(guò)期
        ValidateLifetime = true,
        //驗(yàn)證私鑰
        IssuerSigningKey = new SymmetricSecurityKey(secretByte),
        ClockSkew = TimeSpan.FromHours(1), //過(guò)期時(shí)間容錯(cuò)值,解決服務(wù)器端時(shí)間不同步問(wèn)題(秒)
        RequireExpirationTime = true,
    };
});

然后我們需要繼續(xù)修改builder.Services.AddOpenApi()這行代碼,在里面加上如下代碼:

其中BearerSecuritySchemeTransformer實(shí)現(xiàn)如下:

public sealed class BearerSecuritySchemeTransformer(IAuthenticationSchemeProvider authenticationSchemeProvider) : IOpenApiDocumentTransformer
{
    public async Task TransformAsync(OpenApiDocument document, OpenApiDocumentTransformerContext context, CancellationToken cancellationToken)
    {
        var authenticationSchemes = await authenticationSchemeProvider.GetAllSchemesAsync();
        if (authenticationSchemes.Any(authScheme => authScheme.Name == JwtBearerDefaults.AuthenticationScheme))
        {
            var requirements = new Dictionary<string, OpenApiSecurityScheme>
            {
                [JwtBearerDefaults.AuthenticationScheme] = new OpenApiSecurityScheme
                {
                    Type = SecuritySchemeType.Http,
                    Scheme = JwtBearerDefaults.AuthenticationScheme.ToLower(),
                    In = ParameterLocation.Header,
                    BearerFormat = "Json Web Token"
                }
            };
            document.Components ??= new OpenApiComponents();
            document.Components.SecuritySchemes = requirements;
            foreach (var operation in document.Paths.Values.SelectMany(path => path.Operations))
            {
                operation.Value.Security.Add(new OpenApiSecurityRequirement
                {
                    [new OpenApiSecurityScheme
                    {
                        Reference = new OpenApiReference
                        {
                            Id = JwtBearerDefaults.AuthenticationScheme,
                            Type = ReferenceType.SecurityScheme
                        }
                    }] = Array.Empty<string>()
                });
            }
        }
    }
}

下面就可以通過(guò)[Authorize]開(kāi)啟接口認(rèn)證,并實(shí)現(xiàn)一個(gè)登錄接口獲取token用來(lái)測(cè)試。

[HttpPost("login")]
[EndpointDescription("登錄成功后生成token")]
[AllowAnonymous]
public string  Login()
{
    //登錄成功返回一個(gè)token
    // 1.定義需要使用到的Claims
    var claims = new[] { new Claim("UserId", "test") };
    // 2.從 appsettings.json 中讀取SecretKey
    var secretKey = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(JwtSettingOption.Secret));
    // 3.選擇加密算法
    var algorithm = SecurityAlgorithms.HmacSha256;
    // 4.生成Credentials
    var signingCredentials = new SigningCredentials(secretKey, algorithm);
    var now = DateTime.Now;
    var expires = now.AddMinutes(JwtSettingOption.Expires);
    // 5.根據(jù)以上,生成token
    var jwtSecurityToken = new JwtSecurityToken(
        JwtSettingOption.Issuer,         //Issuer
        JwtSettingOption.Audience,       //Audience
        claims,                          //Claims,
        now,                             //notBefore
        expires,                         //expires
        signingCredentials               //Credentials
    );
    // 6.將token變?yōu)閟tring
    var token = new JwtSecurityTokenHandler().WriteToken(jwtSecurityToken);
    return token;
}

下面我們先用登錄接口獲取一個(gè)token。

我們先用token調(diào)用接口,可以發(fā)現(xiàn)返回401。

然后我們把上面獲取的token放進(jìn)去,請(qǐng)求成功。

在這個(gè)過(guò)程中有可能會(huì)遇到一種情況:Auth Type后面的下拉框不可選,如下圖。

可能因以下原因?qū)е?,缺少[builder.Services.AddAuthentication().AddJwtBearer();]或[options.AddDocumentTransformer();]任意一行代碼。

注:測(cè)試方法代碼以及示例源碼都已經(jīng)上傳至代碼庫(kù),有興趣的可以看看。https://gitee.com/hugogoos/Planner

到此這篇關(guān)于.NET9中Swagger平替Scalar詳解(四)的文章就介紹到這了,更多相關(guān).NET9 Swagger平替Scalar內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論