plugins/languages/csharp/skills/data/SKILL.md
C# 数据访问与 ORM 规范。覆盖 EF Core 10 compiled queries、JSON columns、 complex types、bulk operations (ExecuteUpdate / ExecuteDelete)、SaveChanges interceptors、迁移策略、连接池、TestContainers 集成测试、Dapper micro-ORM 选型、 EF Core 10 Vector / Native AOT。当访问数据库、设计数据模型、调优查询性能、 写迁移脚本、配置 DbContext, 或说 "EF Core"、"Entity Framework"、"DbContext"、 "migration"、"Dapper"、"Npgsql"、"SqlServer"、"compiled query" 时加载。
npx skillsauth add lazygophers/ccplugin csharp-dataInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
EF Core 10 为主线; 高频热点 + 复杂报表用 Dapper。
public class AppDb(DbContextOptions<AppDb> opts) : DbContext(opts)
{
public DbSet<Order> Orders => Set<Order>();
public DbSet<Customer> Customers => Set<Customer>();
protected override void OnModelCreating(ModelBuilder b)
{
b.ApplyConfigurationsFromAssembly(typeof(AppDb).Assembly);
}
}
// Program.cs
builder.Services.AddDbContextPool<AppDb>(o => o
.UseNpgsql(conn, npg => npg.EnableRetryOnFailure())
.EnableSensitiveDataLogging(env.IsDevelopment())
.ConfigureWarnings(w => w.Throw(RelationalEventId.MultipleCollectionIncludeWarning)));
AddDbContextPool 减少分配; ctor 仅接受 DbContextOptions<T>IEntityTypeConfiguration<T> 文件, 不塞 OnModelCreating.UseModel()Guid (UUIDv7) 或 long (Snowflake); 避免暴露自增整数complex type (EF Core 8+) 或 owned entityb.HasQueryFilter(e => !e.IsDeleted)SaveChangesInterceptorpublic sealed class AuditInterceptor : SaveChangesInterceptor
{
public override ValueTask<InterceptionResult<int>> SavingChangesAsync(
DbContextEventData e, InterceptionResult<int> r, CancellationToken ct = default)
{
foreach (var entry in e.Context!.ChangeTracker.Entries<IAuditable>())
{
if (entry.State == EntityState.Added) entry.Entity.CreatedAt = DateTime.UtcNow;
if (entry.State == EntityState.Modified) entry.Entity.UpdatedAt = DateTime.UtcNow;
}
return base.SavingChangesAsync(e, r, ct);
}
}
AsNoTracking(); 写路径走跟踪MultipleCollectionIncludeWarning 为 ThrowAsSplitQuery() 避免笛卡尔积.ToList() 后内存过滤; 下推到 SQLAsAsyncEnumerable() + await foreachvar dtos = await db.Orders.AsNoTracking()
.Where(o => o.CustomerId == cid && o.CreatedAt >= since)
.OrderByDescending(o => o.CreatedAt)
.Select(o => new OrderListDto(o.Id, o.Amount, o.CreatedAt))
.ToListAsync(ct);
热路径用编译查询消除表达式树编译开销:
private static readonly Func<AppDb, long, CancellationToken, Task<Order?>> GetOrderById =
EF.CompileAsyncQuery((AppDb db, long id, CancellationToken ct) =>
db.Orders.AsNoTracking().FirstOrDefault(o => o.Id == id));
EF Core 8+ 内置 JSON 映射, 简化半结构化字段:
b.OwnsOne(c => c.Preferences, p => p.ToJson());
b.OwnsMany(c => c.PhoneNumbers, p => p.ToJson());
可在 LINQ 中直接查询 JSON 内属性 (Provider 支持时下推为 SQL JSON 操作)。
b.Property(d => d.Embedding).HasVector(1536);
var nearest = await db.Documents
.OrderBy(d => d.Embedding.CosineDistance(query))
.Take(10).ToListAsync(ct);
await db.Orders
.Where(o => o.Status == OrderStatus.Cancelled && o.UpdatedAt < cutoff)
.ExecuteDeleteAsync(ct);
await db.Products.Where(p => p.Discontinued)
.ExecuteUpdateAsync(s => s.SetProperty(p => p.Price, p => p.Price * 0.5m), ct);
大量插入用 EFCore.BulkExtensions 或直接 COPY / BULK INSERT。
var strategy = db.Database.CreateExecutionStrategy();
await strategy.ExecuteAsync(async () =>
{
await using var tx = await db.Database.BeginTransactionAsync(ct);
// ...
await db.SaveChangesAsync(ct);
await tx.CommitAsync(ct);
});
ExecutionStrategy 处理 retry 重试; 不要手写循环。
AddOrderStatusIndexdotnet ef migrations script --idempotent 给 DBA 评审if (env.IsDevelopment())
{
using var scope = app.Services.CreateScope();
await scope.ServiceProvider.GetRequiredService<AppDb>().Database.MigrateAsync();
}
appsettings.json + User Secrets / Azure Key Vault; 不入 git| 用 EF Core | 用 Dapper | |-----------|-----------| | CRUD、聚合根 | 复杂报表、多表 join 投影 | | 跟踪变更 | 只读热路径 | | 迁移与建表 | 存储过程调用 |
const string sql = """
SELECT o.id, o.amount, c.name AS customer_name
FROM orders o JOIN customers c ON c.id = o.customer_id
WHERE o.created_at >= @since
ORDER BY o.amount DESC LIMIT 100
""";
await using var conn = await dataSource.OpenConnectionAsync(ct);
var rows = await conn.QueryAsync<OrderReport>(
new CommandDefinition(sql, new { since }, cancellationToken: ct));
Respawn 在每个测试前清表WebApplicationFactory + per-test transaction rollback 也是常见方案var pg = new PostgreSqlBuilder().WithImage("postgres:17").Build();
await pg.StartAsync();
development
Go 数据库规范——GORM Model 命名 ModelXxx、表名单数、枚举 uint8 + 常量、索引 idx_ 前缀 + deleted_at leading column、禁 time.Time 统一 int64 unix、禁指针/nullable 字段、TEXT/BLOB/JSON 禁 default、AutoMigrate 禁改主键。设计 DB model、写 GORM tag、建索引、做 migration 审查时触发。
development
Go HTTP API 规范——响应始终 200 + body code 字段、路由 /api/* 全 POST 单段 <Action><Model>、中间件逐路由注册禁 Group(prefix,mw...)、handler 仅返回 (rsp,error)、认证走 header。设计 HTTP API、写路由/handler/中间件时触发。
development
Go 项目结构规范——三层架构(API → Impl → State)、全局状态模式、internal/ 私有包、cmd/ 仅 main.go、go.work 多模块、禁止 Repository 接口和 DI 容器、struct 公共字段开头全 omitempty、handler var rsp 顶声明、禁 legacy migration。设计项目骨架、新建目录、组织包、做架构评审时触发。
development
Go 命名规范——Id/Uid 字段(非 ID)、IsActive/HasMFA 布尔前缀、CreatedAt 时间字段、接收者统一用 p、包名全小写无下划线、泛型类型参数描述性命名、集合字段 xxx_list 禁 xxxs 复数、Enum 0 值 XxxNil 禁 Unknown、禁 Status 统一 State、Set/Update 语义区分。定义结构体字段、函数、变量、包、接收者名、泛型、枚举时触发。