Files
stream.api/internal/video/runtime/module.go
claude e7fdd0e1ab feat: Add player_configs feature and migrate user preferences
- Implemented player_configs table to store multiple player configurations per user.
- Migrated existing player settings from user_preferences to player_configs.
- Removed player-related columns from user_preferences.
- Added referral state fields to user for tracking referral rewards.
- Created migration scripts for database changes and data migration.
- Added test cases for app services and usage helpers.
- Introduced video job service interfaces and implementations.
2026-03-24 16:08:36 +00:00

76 lines
2.5 KiB
Go

package runtime
import (
"context"
"fmt"
"net"
grpcpkg "google.golang.org/grpc"
"gorm.io/gorm"
"stream.api/internal/config"
apprpc "stream.api/internal/rpc/app"
"stream.api/internal/video"
redisadapter "stream.api/internal/video/runtime/adapters/queue/redis"
runtimegrpc "stream.api/internal/video/runtime/grpc"
"stream.api/internal/video/runtime/services"
"stream.api/pkg/cache"
"stream.api/pkg/logger"
"stream.api/pkg/token"
)
type Module struct {
ctx context.Context
cfg *config.Config
jobService *services.JobService
healthService *services.HealthService
grpcServer *runtimegrpc.Server
mqttPublisher *mqttPublisher
grpcRaw *grpcpkg.Server
}
func NewModule(ctx context.Context, cfg *config.Config, db *gorm.DB, cacheClient cache.Cache, tokenProvider token.Provider, appLogger logger.Logger) (*Module, error) {
adapter, err := redisadapter.NewAdapter(cfg.Redis.Addr, cfg.Redis.Password, cfg.Redis.DB)
if err != nil {
return nil, err
}
jobService := services.NewJobService(adapter, adapter)
healthService := services.NewHealthService(db, adapter.Client(), cfg.Render.ServiceName)
grpcServer := runtimegrpc.NewServer(jobService, cfg.Render.AgentSecret)
module := &Module{ctx: ctx, cfg: cfg, jobService: jobService, healthService: healthService, grpcServer: grpcServer, grpcRaw: grpcpkg.NewServer()}
if publisher, err := NewMQTTBootstrap(jobService, grpcServer, appLogger); err != nil {
appLogger.Error("Failed to initialize MQTT publisher", "error", err)
} else {
module.mqttPublisher = publisher.mqttPublisher
grpcServer.SetAgentEventHandler(func(eventType string, agent *services.AgentWithStats) {
PublishAgentMQTTEvent(publisher.Client(), appLogger, eventType, agent)
})
}
videoService := video.NewService(db, jobService)
grpcServer.Register(module.grpcRaw)
apprpc.Register(module.grpcRaw, apprpc.NewServices(cacheClient, tokenProvider, db, appLogger, cfg, videoService, grpcServer))
if module.mqttPublisher != nil {
module.mqttPublisher.start(ctx)
}
return module, nil
}
func (m *Module) JobService() *services.JobService { return m.jobService }
func (m *Module) AgentRuntime() *runtimegrpc.Server { return m.grpcServer }
func (m *Module) GRPCServer() *grpcpkg.Server { return m.grpcRaw }
func (m *Module) GRPCAddress() string {
return fmt.Sprintf(":%s", m.cfg.Server.GRPCPort)
}
func (m *Module) ServeGRPC(listener net.Listener) error {
return m.grpcRaw.Serve(listener)
}
func (m *Module) Shutdown() {
if m.grpcRaw != nil {
m.grpcRaw.GracefulStop()
}
}