2026-02-05 16:00:34 +08:00
|
|
|
|
package repository
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"context"
|
|
|
|
|
|
"database/sql"
|
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/service"
|
2026-02-28 15:01:20 +08:00
|
|
|
|
"github.com/lib/pq"
|
2026-02-05 16:00:34 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
type userGroupRateRepository struct {
|
|
|
|
|
|
sql sqlExecutor
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// NewUserGroupRateRepository 创建用户专属分组倍率/RPM 仓储
|
2026-02-05 16:00:34 +08:00
|
|
|
|
func NewUserGroupRateRepository(sqlDB *sql.DB) service.UserGroupRateRepository {
|
|
|
|
|
|
return &userGroupRateRepository{sql: sqlDB}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// GetByUserID 获取用户所有专属分组 rate_multiplier(仅返回非 NULL 的条目)
|
2026-02-05 16:00:34 +08:00
|
|
|
|
func (r *userGroupRateRepository) GetByUserID(ctx context.Context, userID int64) (map[int64]float64, error) {
|
2026-04-23 03:33:52 +08:00
|
|
|
|
query := `SELECT group_id, rate_multiplier FROM user_group_rate_multipliers WHERE user_id = $1 AND rate_multiplier IS NOT NULL`
|
2026-02-05 16:00:34 +08:00
|
|
|
|
rows, err := r.sql.QueryContext(ctx, query, userID)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
defer func() { _ = rows.Close() }()
|
|
|
|
|
|
|
|
|
|
|
|
result := make(map[int64]float64)
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
|
|
var groupID int64
|
|
|
|
|
|
var rate float64
|
|
|
|
|
|
if err := rows.Scan(&groupID, &rate); err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
result[groupID] = rate
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := rows.Err(); err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
return result, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// GetByUserIDs 批量获取多个用户的专属分组 rate_multiplier(仅返回非 NULL 的条目)
|
2026-02-28 15:01:20 +08:00
|
|
|
|
func (r *userGroupRateRepository) GetByUserIDs(ctx context.Context, userIDs []int64) (map[int64]map[int64]float64, error) {
|
|
|
|
|
|
result := make(map[int64]map[int64]float64, len(userIDs))
|
|
|
|
|
|
if len(userIDs) == 0 {
|
|
|
|
|
|
return result, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uniqueIDs := make([]int64, 0, len(userIDs))
|
|
|
|
|
|
seen := make(map[int64]struct{}, len(userIDs))
|
|
|
|
|
|
for _, userID := range userIDs {
|
|
|
|
|
|
if userID <= 0 {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
if _, exists := seen[userID]; exists {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
seen[userID] = struct{}{}
|
|
|
|
|
|
uniqueIDs = append(uniqueIDs, userID)
|
|
|
|
|
|
result[userID] = make(map[int64]float64)
|
|
|
|
|
|
}
|
|
|
|
|
|
if len(uniqueIDs) == 0 {
|
|
|
|
|
|
return result, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
rows, err := r.sql.QueryContext(ctx, `
|
|
|
|
|
|
SELECT user_id, group_id, rate_multiplier
|
|
|
|
|
|
FROM user_group_rate_multipliers
|
2026-04-23 03:33:52 +08:00
|
|
|
|
WHERE user_id = ANY($1) AND rate_multiplier IS NOT NULL
|
2026-02-28 15:01:20 +08:00
|
|
|
|
`, pq.Array(uniqueIDs))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
defer func() { _ = rows.Close() }()
|
|
|
|
|
|
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
|
|
var userID int64
|
|
|
|
|
|
var groupID int64
|
|
|
|
|
|
var rate float64
|
|
|
|
|
|
if err := rows.Scan(&userID, &groupID, &rate); err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
if _, ok := result[userID]; !ok {
|
|
|
|
|
|
result[userID] = make(map[int64]float64)
|
|
|
|
|
|
}
|
|
|
|
|
|
result[userID][groupID] = rate
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := rows.Err(); err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
return result, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// GetByGroupID 获取指定分组下所有用户的专属配置(rate 与 rpm_override 任一非 NULL 即返回)
|
2026-03-12 23:37:36 +08:00
|
|
|
|
func (r *userGroupRateRepository) GetByGroupID(ctx context.Context, groupID int64) ([]service.UserGroupRateEntry, error) {
|
|
|
|
|
|
query := `
|
2026-04-23 03:33:52 +08:00
|
|
|
|
SELECT ugr.user_id, u.username, u.email, COALESCE(u.notes, ''), u.status, ugr.rate_multiplier, ugr.rpm_override
|
2026-03-12 23:37:36 +08:00
|
|
|
|
FROM user_group_rate_multipliers ugr
|
2026-04-14 17:35:27 +08:00
|
|
|
|
JOIN users u ON u.id = ugr.user_id AND u.deleted_at IS NULL
|
2026-03-12 23:37:36 +08:00
|
|
|
|
WHERE ugr.group_id = $1
|
|
|
|
|
|
ORDER BY ugr.user_id
|
|
|
|
|
|
`
|
|
|
|
|
|
rows, err := r.sql.QueryContext(ctx, query, groupID)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
defer func() { _ = rows.Close() }()
|
|
|
|
|
|
|
|
|
|
|
|
var result []service.UserGroupRateEntry
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
|
|
var entry service.UserGroupRateEntry
|
2026-04-23 03:33:52 +08:00
|
|
|
|
var rate sql.NullFloat64
|
|
|
|
|
|
var rpm sql.NullInt32
|
|
|
|
|
|
if err := rows.Scan(&entry.UserID, &entry.UserName, &entry.UserEmail, &entry.UserNotes, &entry.UserStatus, &rate, &rpm); err != nil {
|
2026-03-12 23:37:36 +08:00
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
2026-04-23 03:33:52 +08:00
|
|
|
|
if rate.Valid {
|
|
|
|
|
|
v := rate.Float64
|
|
|
|
|
|
entry.RateMultiplier = &v
|
|
|
|
|
|
}
|
|
|
|
|
|
if rpm.Valid {
|
|
|
|
|
|
v := int(rpm.Int32)
|
|
|
|
|
|
entry.RPMOverride = &v
|
|
|
|
|
|
}
|
2026-03-12 23:37:36 +08:00
|
|
|
|
result = append(result, entry)
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := rows.Err(); err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
return result, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// GetByUserAndGroup 获取用户在特定分组的专属 rate_multiplier(NULL 返回 nil)
|
2026-02-05 16:00:34 +08:00
|
|
|
|
func (r *userGroupRateRepository) GetByUserAndGroup(ctx context.Context, userID, groupID int64) (*float64, error) {
|
|
|
|
|
|
query := `SELECT rate_multiplier FROM user_group_rate_multipliers WHERE user_id = $1 AND group_id = $2`
|
2026-04-23 03:33:52 +08:00
|
|
|
|
var rate sql.NullFloat64
|
2026-02-05 16:00:34 +08:00
|
|
|
|
err := scanSingleRow(ctx, r.sql, query, []any{userID, groupID}, &rate)
|
|
|
|
|
|
if err == sql.ErrNoRows {
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
2026-04-23 03:33:52 +08:00
|
|
|
|
if !rate.Valid {
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
v := rate.Float64
|
|
|
|
|
|
return &v, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GetRPMOverrideByUserAndGroup 获取用户在特定分组的 rpm_override(NULL 返回 nil)
|
|
|
|
|
|
func (r *userGroupRateRepository) GetRPMOverrideByUserAndGroup(ctx context.Context, userID, groupID int64) (*int, error) {
|
|
|
|
|
|
query := `SELECT rpm_override FROM user_group_rate_multipliers WHERE user_id = $1 AND group_id = $2`
|
|
|
|
|
|
var rpm sql.NullInt32
|
|
|
|
|
|
err := scanSingleRow(ctx, r.sql, query, []any{userID, groupID}, &rpm)
|
|
|
|
|
|
if err == sql.ErrNoRows {
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
if !rpm.Valid {
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
v := int(rpm.Int32)
|
|
|
|
|
|
return &v, nil
|
2026-02-05 16:00:34 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// SyncUserGroupRates 同步用户的分组专属 rate_multiplier。
|
|
|
|
|
|
// - 传入空 map:清空该用户所有行的 rate_multiplier;若 rpm_override 也为 NULL 则整行删除。
|
|
|
|
|
|
// - 值为 nil:清空对应行的 rate_multiplier(保留 rpm_override)。
|
|
|
|
|
|
// - 值非 nil:upsert rate_multiplier(保留已有 rpm_override)。
|
2026-02-05 16:00:34 +08:00
|
|
|
|
func (r *userGroupRateRepository) SyncUserGroupRates(ctx context.Context, userID int64, rates map[int64]*float64) error {
|
|
|
|
|
|
if len(rates) == 0 {
|
2026-04-23 03:33:52 +08:00
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
UPDATE user_group_rate_multipliers
|
|
|
|
|
|
SET rate_multiplier = NULL, updated_at = NOW()
|
|
|
|
|
|
WHERE user_id = $1
|
|
|
|
|
|
`, userID); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
_, err := r.sql.ExecContext(ctx,
|
|
|
|
|
|
`DELETE FROM user_group_rate_multipliers WHERE user_id = $1 AND rate_multiplier IS NULL AND rpm_override IS NULL`,
|
|
|
|
|
|
userID)
|
2026-02-05 16:00:34 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
var clearGroupIDs []int64
|
2026-02-28 15:01:20 +08:00
|
|
|
|
upsertGroupIDs := make([]int64, 0, len(rates))
|
|
|
|
|
|
upsertRates := make([]float64, 0, len(rates))
|
2026-02-05 16:00:34 +08:00
|
|
|
|
for groupID, rate := range rates {
|
|
|
|
|
|
if rate == nil {
|
2026-04-23 03:33:52 +08:00
|
|
|
|
clearGroupIDs = append(clearGroupIDs, groupID)
|
2026-02-05 16:00:34 +08:00
|
|
|
|
} else {
|
2026-02-28 15:01:20 +08:00
|
|
|
|
upsertGroupIDs = append(upsertGroupIDs, groupID)
|
|
|
|
|
|
upsertRates = append(upsertRates, *rate)
|
2026-02-05 16:00:34 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
if len(clearGroupIDs) > 0 {
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
UPDATE user_group_rate_multipliers
|
|
|
|
|
|
SET rate_multiplier = NULL, updated_at = NOW()
|
|
|
|
|
|
WHERE user_id = $1 AND group_id = ANY($2)
|
|
|
|
|
|
`, userID, pq.Array(clearGroupIDs)); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
2026-02-28 15:01:20 +08:00
|
|
|
|
if _, err := r.sql.ExecContext(ctx,
|
2026-04-23 03:33:52 +08:00
|
|
|
|
`DELETE FROM user_group_rate_multipliers WHERE user_id = $1 AND group_id = ANY($2) AND rate_multiplier IS NULL AND rpm_override IS NULL`,
|
|
|
|
|
|
userID, pq.Array(clearGroupIDs)); err != nil {
|
2026-02-05 16:00:34 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-28 15:01:20 +08:00
|
|
|
|
if len(upsertGroupIDs) > 0 {
|
2026-04-23 03:33:52 +08:00
|
|
|
|
now := time.Now()
|
2026-02-05 16:00:34 +08:00
|
|
|
|
_, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
INSERT INTO user_group_rate_multipliers (user_id, group_id, rate_multiplier, created_at, updated_at)
|
2026-02-28 15:01:20 +08:00
|
|
|
|
SELECT
|
|
|
|
|
|
$1::bigint,
|
|
|
|
|
|
data.group_id,
|
|
|
|
|
|
data.rate_multiplier,
|
|
|
|
|
|
$2::timestamptz,
|
|
|
|
|
|
$2::timestamptz
|
|
|
|
|
|
FROM unnest($3::bigint[], $4::double precision[]) AS data(group_id, rate_multiplier)
|
|
|
|
|
|
ON CONFLICT (user_id, group_id)
|
|
|
|
|
|
DO UPDATE SET
|
|
|
|
|
|
rate_multiplier = EXCLUDED.rate_multiplier,
|
|
|
|
|
|
updated_at = EXCLUDED.updated_at
|
|
|
|
|
|
`, userID, now, pq.Array(upsertGroupIDs), pq.Array(upsertRates))
|
2026-02-05 16:00:34 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// SyncGroupRateMultipliers 同步分组的 rate_multiplier 部分(不触动 rpm_override)。
|
|
|
|
|
|
// 语义:
|
|
|
|
|
|
// - 未出现在 entries 中的用户行:rate_multiplier 归 NULL;若 rpm_override 也为 NULL 则整行删除。
|
|
|
|
|
|
// - 出现的用户行:upsert rate_multiplier。
|
2026-03-12 23:37:36 +08:00
|
|
|
|
func (r *userGroupRateRepository) SyncGroupRateMultipliers(ctx context.Context, groupID int64, entries []service.GroupRateMultiplierInput) error {
|
2026-04-23 03:33:52 +08:00
|
|
|
|
keepUserIDs := make([]int64, 0, len(entries))
|
|
|
|
|
|
for _, e := range entries {
|
|
|
|
|
|
keepUserIDs = append(keepUserIDs, e.UserID)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 未在 entries 列表中的行:清空 rate_multiplier。
|
|
|
|
|
|
if len(keepUserIDs) == 0 {
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
UPDATE user_group_rate_multipliers
|
|
|
|
|
|
SET rate_multiplier = NULL, updated_at = NOW()
|
|
|
|
|
|
WHERE group_id = $1
|
|
|
|
|
|
`, groupID); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
UPDATE user_group_rate_multipliers
|
|
|
|
|
|
SET rate_multiplier = NULL, updated_at = NOW()
|
|
|
|
|
|
WHERE group_id = $1 AND user_id <> ALL($2)
|
|
|
|
|
|
`, groupID, pq.Array(keepUserIDs)); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 清空后若整行 NULL 则删除。
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
DELETE FROM user_group_rate_multipliers
|
|
|
|
|
|
WHERE group_id = $1 AND rate_multiplier IS NULL AND rpm_override IS NULL
|
|
|
|
|
|
`, groupID); err != nil {
|
2026-03-12 23:37:36 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
2026-04-23 03:33:52 +08:00
|
|
|
|
|
2026-03-12 23:37:36 +08:00
|
|
|
|
if len(entries) == 0 {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
2026-04-23 03:33:52 +08:00
|
|
|
|
|
2026-03-12 23:37:36 +08:00
|
|
|
|
userIDs := make([]int64, len(entries))
|
|
|
|
|
|
rates := make([]float64, len(entries))
|
|
|
|
|
|
for i, e := range entries {
|
|
|
|
|
|
userIDs[i] = e.UserID
|
|
|
|
|
|
rates[i] = e.RateMultiplier
|
|
|
|
|
|
}
|
|
|
|
|
|
now := time.Now()
|
|
|
|
|
|
_, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
INSERT INTO user_group_rate_multipliers (user_id, group_id, rate_multiplier, created_at, updated_at)
|
|
|
|
|
|
SELECT data.user_id, $1::bigint, data.rate_multiplier, $2::timestamptz, $2::timestamptz
|
|
|
|
|
|
FROM unnest($3::bigint[], $4::double precision[]) AS data(user_id, rate_multiplier)
|
|
|
|
|
|
ON CONFLICT (user_id, group_id)
|
|
|
|
|
|
DO UPDATE SET rate_multiplier = EXCLUDED.rate_multiplier, updated_at = EXCLUDED.updated_at
|
|
|
|
|
|
`, groupID, now, pq.Array(userIDs), pq.Array(rates))
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// SyncGroupRPMOverrides 同步分组的 rpm_override 部分(不触动 rate_multiplier)。
|
|
|
|
|
|
// 语义:
|
|
|
|
|
|
// - 未出现的用户行:rpm_override 归 NULL;若 rate_multiplier 也为 NULL 则整行删除。
|
|
|
|
|
|
// - 出现的用户行:若 RPMOverride 为 nil 则清空;非 nil 则 upsert。
|
|
|
|
|
|
func (r *userGroupRateRepository) SyncGroupRPMOverrides(ctx context.Context, groupID int64, entries []service.GroupRPMOverrideInput) error {
|
|
|
|
|
|
keepUserIDs := make([]int64, 0, len(entries))
|
|
|
|
|
|
var clearUserIDs []int64
|
|
|
|
|
|
upsertUserIDs := make([]int64, 0, len(entries))
|
|
|
|
|
|
upsertValues := make([]int32, 0, len(entries))
|
|
|
|
|
|
for _, e := range entries {
|
|
|
|
|
|
keepUserIDs = append(keepUserIDs, e.UserID)
|
|
|
|
|
|
if e.RPMOverride == nil {
|
|
|
|
|
|
clearUserIDs = append(clearUserIDs, e.UserID)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
upsertUserIDs = append(upsertUserIDs, e.UserID)
|
|
|
|
|
|
upsertValues = append(upsertValues, int32(*e.RPMOverride))
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 未在 entries 列表中的行:清空 rpm_override。
|
|
|
|
|
|
if len(keepUserIDs) == 0 {
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
UPDATE user_group_rate_multipliers
|
|
|
|
|
|
SET rpm_override = NULL, updated_at = NOW()
|
|
|
|
|
|
WHERE group_id = $1
|
|
|
|
|
|
`, groupID); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
UPDATE user_group_rate_multipliers
|
|
|
|
|
|
SET rpm_override = NULL, updated_at = NOW()
|
|
|
|
|
|
WHERE group_id = $1 AND user_id <> ALL($2)
|
|
|
|
|
|
`, groupID, pq.Array(keepUserIDs)); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 显式 clear 的行。
|
|
|
|
|
|
if len(clearUserIDs) > 0 {
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
UPDATE user_group_rate_multipliers
|
|
|
|
|
|
SET rpm_override = NULL, updated_at = NOW()
|
|
|
|
|
|
WHERE group_id = $1 AND user_id = ANY($2)
|
|
|
|
|
|
`, groupID, pq.Array(clearUserIDs)); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 清空后若整行 NULL 则删除。
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
DELETE FROM user_group_rate_multipliers
|
|
|
|
|
|
WHERE group_id = $1 AND rate_multiplier IS NULL AND rpm_override IS NULL
|
|
|
|
|
|
`, groupID); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if len(upsertUserIDs) > 0 {
|
|
|
|
|
|
now := time.Now()
|
|
|
|
|
|
_, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
INSERT INTO user_group_rate_multipliers (user_id, group_id, rpm_override, created_at, updated_at)
|
|
|
|
|
|
SELECT data.user_id, $1::bigint, data.rpm_override, $2::timestamptz, $2::timestamptz
|
|
|
|
|
|
FROM unnest($3::bigint[], $4::integer[]) AS data(user_id, rpm_override)
|
|
|
|
|
|
ON CONFLICT (user_id, group_id)
|
|
|
|
|
|
DO UPDATE SET rpm_override = EXCLUDED.rpm_override, updated_at = EXCLUDED.updated_at
|
|
|
|
|
|
`, groupID, now, pq.Array(upsertUserIDs), pq.Array(upsertValues))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ClearGroupRPMOverrides 清空指定分组所有行的 rpm_override。
|
|
|
|
|
|
func (r *userGroupRateRepository) ClearGroupRPMOverrides(ctx context.Context, groupID int64) error {
|
|
|
|
|
|
if _, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
UPDATE user_group_rate_multipliers
|
|
|
|
|
|
SET rpm_override = NULL, updated_at = NOW()
|
|
|
|
|
|
WHERE group_id = $1
|
|
|
|
|
|
`, groupID); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
_, err := r.sql.ExecContext(ctx, `
|
|
|
|
|
|
DELETE FROM user_group_rate_multipliers
|
|
|
|
|
|
WHERE group_id = $1 AND rate_multiplier IS NULL AND rpm_override IS NULL
|
|
|
|
|
|
`, groupID)
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// DeleteByGroupID 删除指定分组的所有用户专属条目
|
2026-02-05 16:00:34 +08:00
|
|
|
|
func (r *userGroupRateRepository) DeleteByGroupID(ctx context.Context, groupID int64) error {
|
|
|
|
|
|
_, err := r.sql.ExecContext(ctx, `DELETE FROM user_group_rate_multipliers WHERE group_id = $1`, groupID)
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-23 03:33:52 +08:00
|
|
|
|
// DeleteByUserID 删除指定用户的所有专属条目
|
2026-02-05 16:00:34 +08:00
|
|
|
|
func (r *userGroupRateRepository) DeleteByUserID(ctx context.Context, userID int64) error {
|
|
|
|
|
|
_, err := r.sql.ExecContext(ctx, `DELETE FROM user_group_rate_multipliers WHERE user_id = $1`, userID)
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|