Files
lijiaoqiao/projects/ai-customer-service/internal/store/postgres/session_store.go
Your Name 687c4535f8 fix: P0-1 RateLimiter并发写安全 + P0-2工单操作错误码区分 + P1 rows.Close修复
P0-1 (limits.go): Allow()方法改为全程使用写锁保护counters map读写,避免RLock写入时的data race
P0-2 (ticket_workflow.go+ticket_handler.go): Assign/Resolve/Close操作先查询ticket存在性和状态,返回明确的CS_TICKET_4001/CS_TKT_4002/CS_TICKET_4092/CS_TICKET_4093错误码,handler根据错误前缀路由HTTP状态码
P1-1 (ticket_store.go): 移除GetStats中3处手动rows.Close(),只保留defer Close()
2026-05-01 20:56:25 +08:00

61 lines
2.4 KiB
Go

package postgres
import (
"context"
"database/sql"
"fmt"
"time"
"github.com/bridge/ai-customer-service/internal/domain/session"
)
type SessionStore struct {
db *sql.DB
}
func NewSessionStore(db *sql.DB) *SessionStore {
return &SessionStore{db: db}
}
func (s *SessionStore) GetOrCreate(ctx context.Context, channel, openID string, now time.Time) (*session.Session, error) {
if s.db == nil {
return nil, fmt.Errorf("db is nil")
}
var sess session.Session
err := s.db.QueryRowContext(ctx, `SELECT id::text, channel, open_id, COALESCE(user_id,''), status, turn_count, last_message_at, created_at, updated_at FROM cs_sessions WHERE channel = $1 AND open_id = $2 AND status != 'closed' ORDER BY updated_at DESC LIMIT 1`, channel, openID).Scan(&sess.ID, &sess.Channel, &sess.OpenID, &sess.UserID, &sess.Status, &sess.TurnCount, &sess.LastMessageAt, new(time.Time), new(time.Time))
if err == nil {
return &sess, nil
}
if err != sql.ErrNoRows {
return nil, err
}
err = s.db.QueryRowContext(ctx, `INSERT INTO cs_sessions(channel, open_id, status, turn_count, last_message_at) VALUES ($1,$2,'idle',0,$3) RETURNING id::text, channel, open_id, COALESCE(user_id,''), status, turn_count, last_message_at, created_at, updated_at`, channel, openID, now).Scan(&sess.ID, &sess.Channel, &sess.OpenID, &sess.UserID, &sess.Status, &sess.TurnCount, &sess.LastMessageAt, new(time.Time), new(time.Time))
if err != nil {
return nil, err
}
return &sess, nil
}
func (s *SessionStore) GetByID(ctx context.Context, id string) (*session.Session, error) {
if s.db == nil {
return nil, fmt.Errorf("db is nil")
}
var sess session.Session
err := s.db.QueryRowContext(ctx,
`SELECT id::text, channel, open_id, COALESCE(user_id,''), status, turn_count, last_message_at, created_at, updated_at FROM cs_sessions WHERE id = $1::uuid`,
id,
).Scan(&sess.ID, &sess.Channel, &sess.OpenID, &sess.UserID, &sess.Status, &sess.TurnCount, &sess.LastMessageAt, new(time.Time), new(time.Time))
if err != nil {
return nil, err
}
return &sess, nil
}
func (s *SessionStore) Save(ctx context.Context, sess *session.Session) error {
if s.db == nil {
return fmt.Errorf("db is nil")
}
_, err := s.db.ExecContext(ctx, `UPDATE cs_sessions SET user_id = NULLIF($2,''), status = $3, turn_count = $4, last_message_at = $5, updated_at = NOW() WHERE id = $1::uuid`, sess.ID, sess.UserID, string(sess.Status), sess.TurnCount, sess.LastMessageAt)
return err
}