Live video on the AT Protocol
at eli/optional-convergence 191 lines 6.8 kB view raw
1package model 2 3import ( 4 "context" 5 "fmt" 6 "os" 7 "path/filepath" 8 "time" 9 10 comatproto "github.com/bluesky-social/indigo/api/atproto" 11 "github.com/bluesky-social/indigo/api/bsky" 12 "github.com/bluesky-social/indigo/atproto/syntax" 13 "gorm.io/driver/sqlite" 14 "gorm.io/gorm" 15 "gorm.io/plugin/prometheus" 16 "stream.place/streamplace/pkg/config" 17 "stream.place/streamplace/pkg/log" 18 "stream.place/streamplace/pkg/streamplace" 19) 20 21type DBModel struct { 22 DB *gorm.DB 23} 24 25type Model interface { 26 CreatePlayerEvent(event PlayerEventAPI) error 27 ListPlayerEvents(playerID string) ([]PlayerEvent, error) 28 PlayerReport(playerID string) (map[string]any, error) 29 ClearPlayerEvents() error 30 31 CreateSegment(segment *Segment) error 32 MostRecentSegments() ([]Segment, error) 33 LatestSegmentForUser(user string) (*Segment, error) 34 LatestSegmentsForUser(user string, limit int, before *time.Time, after *time.Time) ([]Segment, error) 35 FilterLiveRepoDIDs(repoDIDs []string) ([]string, error) 36 CreateThumbnail(thumb *Thumbnail) error 37 LatestThumbnailForUser(user string) (*Thumbnail, error) 38 GetSegment(id string) (*Segment, error) 39 GetExpiredSegments(ctx context.Context) ([]Segment, error) 40 DeleteSegment(ctx context.Context, id string) error 41 StartSegmentCleaner(ctx context.Context) error 42 SegmentCleaner(ctx context.Context) error 43 44 GetIdentity(id string) (*Identity, error) 45 UpdateIdentity(ident *Identity) error 46 47 GetRepo(did string) (*Repo, error) 48 GetRepoByHandle(handle string) (*Repo, error) 49 GetRepoByHandleOrDID(arg string) (*Repo, error) 50 GetRepoBySigningKey(signingKey string) (*Repo, error) 51 GetAllRepos() ([]Repo, error) 52 SearchReposByHandle(query string, limit int) ([]Repo, error) 53 UpdateRepo(repo *Repo) error 54 55 UpdateSigningKey(key *SigningKey) error 56 GetSigningKey(ctx context.Context, did, repoDID string) (*SigningKey, error) 57 GetSigningKeyByRKey(ctx context.Context, rkey string) (*SigningKey, error) 58 GetSigningKeysForRepo(repoDID string) ([]SigningKey, error) 59 60 CreateFollow(ctx context.Context, userDID, rev string, follow *bsky.GraphFollow) error 61 GetUserFollowing(ctx context.Context, userDID string) ([]Follow, error) 62 GetUserFollowers(ctx context.Context, userDID string) ([]Follow, error) 63 GetUserFollowingUser(ctx context.Context, userDID, subjectDID string) (*Follow, error) 64 DeleteFollow(ctx context.Context, userDID, rev string) error 65 66 CreateFeedPost(ctx context.Context, post *FeedPost) error 67 ListFeedPosts() ([]FeedPost, error) 68 ListFeedPostsByType(feedType string, limit int, after int64) ([]FeedPost, error) 69 GetFeedPost(uri string) (*FeedPost, error) 70 GetReplies(repoDID string) ([]*bsky.FeedDefs_PostView, error) 71 72 CreateLivestream(ctx context.Context, ls *Livestream) error 73 GetLatestLivestreamForRepo(repoDID string) (*Livestream, error) 74 GetLivestreamByPostURI(postURI string) (*Livestream, error) 75 GetLatestLivestreams(limit int, before *time.Time) ([]Livestream, error) 76 77 CreateBlock(ctx context.Context, block *Block) error 78 GetBlock(ctx context.Context, rkey string) (*Block, error) 79 GetUserBlock(ctx context.Context, userDID, subjectDID string) (*Block, error) 80 DeleteBlock(ctx context.Context, rkey string) error 81 82 CreateChatMessage(ctx context.Context, message *ChatMessage) error 83 MostRecentChatMessages(repoDID string) ([]*streamplace.ChatDefs_MessageView, error) 84 GetChatMessage(uri string) (*ChatMessage, error) 85 DeleteChatMessage(ctx context.Context, uri string, deletedAt *time.Time) error 86 87 CreateGate(ctx context.Context, gate *Gate) error 88 DeleteGate(ctx context.Context, rkey string) error 89 GetGate(ctx context.Context, rkey string) (*Gate, error) 90 GetUserGates(ctx context.Context, userDID string) ([]*Gate, error) 91 92 CreateChatProfile(ctx context.Context, profile *ChatProfile) error 93 GetChatProfile(ctx context.Context, repoDID string) (*ChatProfile, error) 94 95 UpdateServerSettings(ctx context.Context, settings *ServerSettings) error 96 GetServerSettings(ctx context.Context, server string, repoDID string) (*ServerSettings, error) 97 DeleteServerSettings(ctx context.Context, server string, repoDID string) error 98 99 CreateLabeler(did string) (*Labeler, error) 100 GetLabeler(did string) (*Labeler, error) 101 UpdateLabelerCursor(did string, cursor int64) error 102 103 CreateLabel(label *Label) error 104 GetActiveLabels(uri string) ([]*comatproto.LabelDefs_Label, error) 105 106 UpdateBroadcastOrigin(ctx context.Context, origin *streamplace.BroadcastOrigin, aturi syntax.ATURI) error 107 GetRecentBroadcastOrigins(ctx context.Context) ([]*streamplace.BroadcastDefs_BroadcastOriginView, error) 108 109 CreateMetadataConfiguration(ctx context.Context, metadata *MetadataConfiguration) error 110 GetMetadataConfiguration(ctx context.Context, repoDID string) (*MetadataConfiguration, error) 111 DeleteMetadataConfiguration(ctx context.Context, repoDID string) error 112 113 GetRecommendation(userDID string) (*Recommendation, error) 114 UpsertRecommendation(rec *Recommendation) error 115} 116 117var DBRevision = 2 118 119func MakeDB(dbURL string) (Model, error) { 120 sqliteSuffix := dbURL 121 if dbURL != ":memory:" { 122 // Ensure dbURL exists as a directory on the filesystem 123 if err := os.MkdirAll(dbURL, os.ModePerm); err != nil { 124 return nil, fmt.Errorf("error creating database directory: %w", err) 125 } 126 dbPath := filepath.Join(dbURL, fmt.Sprintf("index_%d.sqlite", DBRevision)) 127 sqliteSuffix = dbPath 128 // if this isn't ":memory:", ensure that directory exists (eg, if db 129 // file is being initialized) 130 if err := os.MkdirAll(filepath.Dir(sqliteSuffix), os.ModePerm); err != nil { 131 return nil, fmt.Errorf("error creating database path: %w", err) 132 } 133 } 134 log.Log(context.Background(), "starting database", "dbURL", sqliteSuffix) 135 dial := sqlite.Open(sqliteSuffix) 136 137 db, err := gorm.Open(dial, &gorm.Config{ 138 SkipDefaultTransaction: true, 139 TranslateError: true, 140 Logger: config.GormLogger, 141 }) 142 if err != nil { 143 return nil, fmt.Errorf("error starting database: %w", err) 144 } 145 err = db.Exec("PRAGMA journal_mode=WAL;").Error 146 if err != nil { 147 return nil, fmt.Errorf("error setting journal mode: %w", err) 148 } 149 150 err = db.Use(prometheus.New(prometheus.Config{ 151 DBName: "index", 152 RefreshInterval: 10, 153 StartServer: false, 154 })) 155 if err != nil { 156 return nil, fmt.Errorf("error using prometheus plugin: %w", err) 157 } 158 159 sqlDB, err := db.DB() 160 if err != nil { 161 return nil, fmt.Errorf("error getting database: %w", err) 162 } 163 sqlDB.SetMaxOpenConns(1) 164 for _, model := range []any{ 165 PlayerEvent{}, 166 Segment{}, 167 Thumbnail{}, 168 Identity{}, 169 Repo{}, 170 SigningKey{}, 171 Follow{}, 172 FeedPost{}, 173 Livestream{}, 174 Block{}, 175 ChatMessage{}, 176 ChatProfile{}, 177 Gate{}, 178 ServerSettings{}, 179 Labeler{}, 180 Label{}, 181 BroadcastOrigin{}, 182 MetadataConfiguration{}, 183 Recommendation{}, 184 } { 185 err = db.AutoMigrate(model) 186 if err != nil { 187 return nil, err 188 } 189 } 190 return &DBModel{DB: db}, nil 191}