using System.Net; using System.Net.Http.Headers; using System.Net.Http.Json; using CampusErrand.Data; using CampusErrand.Models; using CampusErrand.Models.Dtos; using FsCheck; using FsCheck.Xunit; using Microsoft.AspNetCore.Mvc.Testing; using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.DependencyInjection; namespace CampusErrand.Tests; /// /// Property 2: 跑腿佣金输入校验 /// 对任意数值输入,当该值小于 1.0 时,订单创建请求应被拒绝。 /// 当该值大于等于 1.0 且小数位不超过 1 位时,订单创建请求应被接受。 /// **Feature: login-and-homepage, Property 2: 跑腿佣金输入校验** /// /// public class CommissionValidationPropertyTests : IDisposable { private const string JwtSecret = "YourSuperSecretKeyForJwtTokenGeneration_AtLeast32Chars!"; private const string JwtIssuer = "CampusErrand"; private const string JwtAudience = "CampusErrandApp"; private readonly List> _factories = []; private WebApplicationFactory CreateFactory(string dbName) { var factory = new WebApplicationFactory() .WithWebHostBuilder(builder => { builder.ConfigureServices(services => { var efDescriptors = services .Where(d => d.ServiceType.FullName?.Contains("EntityFrameworkCore") == true || d.ServiceType == typeof(DbContextOptions) || d.ServiceType == typeof(DbContextOptions) || d.ImplementationType?.FullName?.Contains("EntityFrameworkCore") == true) .ToList(); foreach (var d in efDescriptors) services.Remove(d); services.AddDbContext(options => options.UseInMemoryDatabase(dbName)); }); }); _factories.Add(factory); return factory; } private static string GenerateUserToken(int userId = 1) { var key = new Microsoft.IdentityModel.Tokens.SymmetricSecurityKey( System.Text.Encoding.UTF8.GetBytes(JwtSecret)); var credentials = new Microsoft.IdentityModel.Tokens.SigningCredentials( key, Microsoft.IdentityModel.Tokens.SecurityAlgorithms.HmacSha256); var claims = new[] { new System.Security.Claims.Claim(System.Security.Claims.ClaimTypes.NameIdentifier, userId.ToString()), new System.Security.Claims.Claim(System.Security.Claims.ClaimTypes.Role, "User") }; var token = new System.IdentityModel.Tokens.Jwt.JwtSecurityToken( issuer: JwtIssuer, audience: JwtAudience, claims: claims, expires: DateTime.UtcNow.AddHours(1), signingCredentials: credentials); return new System.IdentityModel.Tokens.Jwt.JwtSecurityTokenHandler().WriteToken(token); } private static CreateOrderRequest MakePickupOrder(decimal commission) => new() { OrderType = "Pickup", ItemName = "测试物品", DeliveryLocation = "测试地点", Phone = "13800138000", Commission = commission }; /// /// 属性:佣金小于 1.0 时,订单创建应被拒绝(返回 400) /// [Property(MaxTest = 20)] public bool 佣金低于1元时订单创建被拒绝(PositiveInt seed) { // 生成 0.0 ~ 0.9 之间的佣金值 var commission = Math.Round((seed.Get % 10) * 0.1m, 1); if (commission >= 1.0m) return true; // 跳过不符合条件的值 var dbName = $"comm_low_{Guid.NewGuid()}"; using var factory = CreateFactory(dbName); var client = factory.CreateClient(); client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", GenerateUserToken()); var request = MakePickupOrder(commission); var response = client.PostAsJsonAsync("/api/orders", request).Result; return response.StatusCode == HttpStatusCode.BadRequest; } /// /// 属性:佣金大于等于 1.0 且小数位不超过 1 位时,订单创建应成功(返回 201) /// [Property(MaxTest = 20)] public bool 合法佣金时订单创建成功(PositiveInt seed) { // 生成 1.0 ~ 100.0 之间的合法佣金值(1 位小数) var commission = Math.Round(1.0m + (seed.Get % 990) * 0.1m, 1); var dbName = $"comm_ok_{Guid.NewGuid()}"; using var factory = CreateFactory(dbName); var client = factory.CreateClient(); client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", GenerateUserToken()); var request = MakePickupOrder(commission); var response = client.PostAsJsonAsync("/api/orders", request).Result; return response.StatusCode == HttpStatusCode.Created; } /// /// 属性:佣金小数位超过 1 位时,订单创建应被拒绝(返回 400) /// [Property(MaxTest = 20)] public bool 佣金小数位超过1位时订单创建被拒绝(PositiveInt seed) { // 生成小数位为 2 位的佣金值,如 1.23, 2.56 等 var commission = 1.0m + (seed.Get % 900) * 0.01m; // 确保确实有超过 1 位小数 if (commission == Math.Round(commission, 1)) commission += 0.01m; var dbName = $"comm_dec_{Guid.NewGuid()}"; using var factory = CreateFactory(dbName); var client = factory.CreateClient(); client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", GenerateUserToken()); var request = MakePickupOrder(commission); var response = client.PostAsJsonAsync("/api/orders", request).Result; return response.StatusCode == HttpStatusCode.BadRequest; } public void Dispose() { foreach (var f in _factories) f.Dispose(); _factories.Clear(); } }