forked from tangled.org/core
Monorepo for Tangled — https://tangled.org
1package db 2 3import ( 4 "database/sql" 5 "fmt" 6 "strings" 7 8 "github.com/bluesky-social/indigo/atproto/syntax" 9 "tangled.org/core/api/tangled" 10 "tangled.org/core/appview/models" 11) 12 13// ValidateReferenceLinks resolves refLinks to Issue/PR/IssueComment/PullComment ATURIs. 14// It will ignore missing refLinks. 15func ValidateReferenceLinks(e Execer, refLinks []models.ReferenceLink) ([]syntax.ATURI, error) { 16 var ( 17 issueRefs []models.ReferenceLink 18 pullRefs []models.ReferenceLink 19 ) 20 for _, ref := range refLinks { 21 switch ref.Kind { 22 case models.RefKindIssue: 23 issueRefs = append(issueRefs, ref) 24 case models.RefKindPull: 25 pullRefs = append(pullRefs, ref) 26 } 27 } 28 issueUris, err := findIssueReferences(e, issueRefs) 29 if err != nil { 30 return nil, fmt.Errorf("find issue references: %w", err) 31 } 32 pullUris, err := findPullReferences(e, pullRefs) 33 if err != nil { 34 return nil, fmt.Errorf("find pull references: %w", err) 35 } 36 37 return append(issueUris, pullUris...), nil 38} 39 40func findIssueReferences(e Execer, refLinks []models.ReferenceLink) ([]syntax.ATURI, error) { 41 if len(refLinks) == 0 { 42 return nil, nil 43 } 44 vals := make([]string, len(refLinks)) 45 args := make([]any, 0, len(refLinks)*4) 46 for i, ref := range refLinks { 47 vals[i] = "(?, ?, ?, ?)" 48 args = append(args, ref.Handle, ref.Repo, ref.SubjectId, ref.CommentId) 49 } 50 query := fmt.Sprintf( 51 `with input(owner_did, name, issue_id, comment_id) as ( 52 values %s 53 ) 54 select 55 i.did, i.rkey, 56 c.did, c.rkey 57 from input inp 58 join repos r 59 on r.did = inp.owner_did 60 and r.name = inp.name 61 join issues i 62 on i.repo_at = r.at_uri 63 and i.issue_id = inp.issue_id 64 left join issue_comments c 65 on inp.comment_id is not null 66 and c.issue_at = i.at_uri 67 and c.id = inp.comment_id 68 `, 69 strings.Join(vals, ","), 70 ) 71 rows, err := e.Query(query, args...) 72 if err != nil { 73 return nil, err 74 } 75 defer rows.Close() 76 77 var uris []syntax.ATURI 78 79 for rows.Next() { 80 // Scan rows 81 var issueOwner, issueRkey string 82 var commentOwner, commentRkey sql.NullString 83 var uri syntax.ATURI 84 if err := rows.Scan(&issueOwner, &issueRkey, &commentOwner, &commentRkey); err != nil { 85 return nil, err 86 } 87 if commentOwner.Valid && commentRkey.Valid { 88 uri = syntax.ATURI(fmt.Sprintf( 89 "at://%s/%s/%s", 90 commentOwner.String, 91 tangled.RepoIssueCommentNSID, 92 commentRkey.String, 93 )) 94 } else { 95 uri = syntax.ATURI(fmt.Sprintf( 96 "at://%s/%s/%s", 97 issueOwner, 98 tangled.RepoIssueNSID, 99 issueRkey, 100 )) 101 } 102 uris = append(uris, uri) 103 } 104 if err := rows.Err(); err != nil { 105 return nil, fmt.Errorf("iterate rows: %w", err) 106 } 107 108 return uris, nil 109} 110 111func findPullReferences(e Execer, refLinks []models.ReferenceLink) ([]syntax.ATURI, error) { 112 if len(refLinks) == 0 { 113 return nil, nil 114 } 115 vals := make([]string, len(refLinks)) 116 args := make([]any, 0, len(refLinks)*4) 117 for i, ref := range refLinks { 118 vals[i] = "(?, ?, ?, ?)" 119 args = append(args, ref.Handle, ref.Repo, ref.SubjectId, ref.CommentId) 120 } 121 query := fmt.Sprintf( 122 `with input(owner_did, name, pull_id, comment_id) as ( 123 values %s 124 ) 125 select 126 p.owner_did, p.rkey, 127 c.comment_at 128 from input inp 129 join repos r 130 on r.did = inp.owner_did 131 and r.name = inp.name 132 join pulls p 133 on p.repo_at = r.at_uri 134 and p.pull_id = inp.pull_id 135 left join pull_comments c 136 on inp.comment_id is not null 137 and c.repo_at = r.at_uri and c.pull_id = p.pull_id 138 and c.id = inp.comment_id 139 `, 140 strings.Join(vals, ","), 141 ) 142 rows, err := e.Query(query, args...) 143 if err != nil { 144 return nil, err 145 } 146 defer rows.Close() 147 148 var uris []syntax.ATURI 149 150 for rows.Next() { 151 // Scan rows 152 var pullOwner, pullRkey string 153 var commentUri sql.NullString 154 var uri syntax.ATURI 155 if err := rows.Scan(&pullOwner, &pullRkey, &commentUri); err != nil { 156 return nil, err 157 } 158 if commentUri.Valid { 159 // no-op 160 uri = syntax.ATURI(commentUri.String) 161 } else { 162 uri = syntax.ATURI(fmt.Sprintf( 163 "at://%s/%s/%s", 164 pullOwner, 165 tangled.RepoPullNSID, 166 pullRkey, 167 )) 168 } 169 uris = append(uris, uri) 170 } 171 return uris, nil 172} 173 174func putReferences(tx *sql.Tx, fromAt syntax.ATURI, references []syntax.ATURI) error { 175 err := deleteReferences(tx, fromAt) 176 if err != nil { 177 return fmt.Errorf("delete old reference_links: %w", err) 178 } 179 if len(references) == 0 { 180 return nil 181 } 182 183 values := make([]string, 0, len(references)) 184 args := make([]any, 0, len(references)*2) 185 for _, ref := range references { 186 values = append(values, "(?, ?)") 187 args = append(args, fromAt, ref) 188 } 189 _, err = tx.Exec( 190 fmt.Sprintf( 191 `insert into reference_links (from_at, to_at) 192 values %s`, 193 strings.Join(values, ","), 194 ), 195 args..., 196 ) 197 if err != nil { 198 return fmt.Errorf("insert new reference_links: %w", err) 199 } 200 return nil 201} 202 203func deleteReferences(tx *sql.Tx, fromAt syntax.ATURI) error { 204 _, err := tx.Exec(`delete from reference_links where from_at = ?`, fromAt) 205 return err 206} 207 208func GetReferencesAll(e Execer, filters ...filter) (map[syntax.ATURI][]syntax.ATURI, error) { 209 var ( 210 conditions []string 211 args []any 212 ) 213 for _, filter := range filters { 214 conditions = append(conditions, filter.Condition()) 215 args = append(args, filter.Arg()...) 216 } 217 218 whereClause := "" 219 if conditions != nil { 220 whereClause = " where " + strings.Join(conditions, " and ") 221 } 222 223 rows, err := e.Query( 224 fmt.Sprintf( 225 `select from_at, to_at from reference_links %s`, 226 whereClause, 227 ), 228 args..., 229 ) 230 if err != nil { 231 return nil, fmt.Errorf("query reference_links: %w", err) 232 } 233 defer rows.Close() 234 235 result := make(map[syntax.ATURI][]syntax.ATURI) 236 237 for rows.Next() { 238 var from, to syntax.ATURI 239 if err := rows.Scan(&from, &to); err != nil { 240 return nil, fmt.Errorf("scan row: %w", err) 241 } 242 243 result[from] = append(result[from], to) 244 } 245 if err := rows.Err(); err != nil { 246 return nil, fmt.Errorf("iterate rows: %w", err) 247 } 248 249 return result, nil 250} 251 252func GetBacklinks(e Execer, target syntax.ATURI) ([]models.RichReferenceLink, error) { 253 rows, err := e.Query( 254 `select from_at from reference_links 255 where to_at = ?`, 256 target, 257 ) 258 if err != nil { 259 return nil, fmt.Errorf("query backlinks: %w", err) 260 } 261 defer rows.Close() 262 263 var ( 264 backlinks []models.RichReferenceLink 265 backlinksMap = make(map[string][]syntax.ATURI) 266 ) 267 for rows.Next() { 268 var from syntax.ATURI 269 if err := rows.Scan(&from); err != nil { 270 return nil, fmt.Errorf("scan row: %w", err) 271 } 272 nsid := from.Collection().String() 273 backlinksMap[nsid] = append(backlinksMap[nsid], from) 274 } 275 if err := rows.Err(); err != nil { 276 return nil, fmt.Errorf("iterate rows: %w", err) 277 } 278 279 var ls []models.RichReferenceLink 280 ls, err = getIssueBacklinks(e, backlinksMap[tangled.RepoIssueNSID]) 281 if err != nil { 282 return nil, fmt.Errorf("get issue backlinks: %w", err) 283 } 284 backlinks = append(backlinks, ls...) 285 ls, err = getIssueCommentBacklinks(e, backlinksMap[tangled.RepoIssueCommentNSID]) 286 if err != nil { 287 return nil, fmt.Errorf("get issue_comment backlinks: %w", err) 288 } 289 backlinks = append(backlinks, ls...) 290 ls, err = getPullBacklinks(e, backlinksMap[tangled.RepoPullNSID]) 291 if err != nil { 292 return nil, fmt.Errorf("get pull backlinks: %w", err) 293 } 294 backlinks = append(backlinks, ls...) 295 ls, err = getPullCommentBacklinks(e, backlinksMap[tangled.RepoPullCommentNSID]) 296 if err != nil { 297 return nil, fmt.Errorf("get pull_comment backlinks: %w", err) 298 } 299 backlinks = append(backlinks, ls...) 300 301 return backlinks, nil 302} 303 304func getIssueBacklinks(e Execer, aturis []syntax.ATURI) ([]models.RichReferenceLink, error) { 305 if len(aturis) == 0 { 306 return nil, nil 307 } 308 vals := make([]string, len(aturis)) 309 args := make([]any, 0, len(aturis)*2) 310 for i, aturi := range aturis { 311 vals[i] = "(?, ?)" 312 did := aturi.Authority().String() 313 rkey := aturi.RecordKey().String() 314 args = append(args, did, rkey) 315 } 316 rows, err := e.Query( 317 fmt.Sprintf( 318 `select r.did, r.name, i.issue_id, i.title, i.open 319 from issues i 320 join repos r 321 on r.at_uri = i.repo_at 322 where (i.did, i.rkey) in (%s)`, 323 strings.Join(vals, ","), 324 ), 325 args..., 326 ) 327 if err != nil { 328 return nil, err 329 } 330 defer rows.Close() 331 var refLinks []models.RichReferenceLink 332 for rows.Next() { 333 var l models.RichReferenceLink 334 l.Kind = models.RefKindIssue 335 if err := rows.Scan(&l.Handle, &l.Repo, &l.SubjectId, &l.Title, &l.State); err != nil { 336 return nil, err 337 } 338 refLinks = append(refLinks, l) 339 } 340 if err := rows.Err(); err != nil { 341 return nil, fmt.Errorf("iterate rows: %w", err) 342 } 343 return refLinks, nil 344} 345 346func getIssueCommentBacklinks(e Execer, aturis []syntax.ATURI) ([]models.RichReferenceLink, error) { 347 if len(aturis) == 0 { 348 return nil, nil 349 } 350 filter := FilterIn("c.at_uri", aturis) 351 rows, err := e.Query( 352 fmt.Sprintf( 353 `select r.did, r.name, i.issue_id, c.id, i.title, i.open 354 from issue_comments c 355 join issues i 356 on i.at_uri = c.issue_at 357 join repos r 358 on r.at_uri = i.repo_at 359 where %s`, 360 filter.Condition(), 361 ), 362 filter.Arg()..., 363 ) 364 if err != nil { 365 return nil, err 366 } 367 defer rows.Close() 368 var refLinks []models.RichReferenceLink 369 for rows.Next() { 370 var l models.RichReferenceLink 371 l.Kind = models.RefKindIssue 372 l.CommentId = new(int) 373 if err := rows.Scan(&l.Handle, &l.Repo, &l.SubjectId, l.CommentId, &l.Title, &l.State); err != nil { 374 return nil, err 375 } 376 refLinks = append(refLinks, l) 377 } 378 if err := rows.Err(); err != nil { 379 return nil, fmt.Errorf("iterate rows: %w", err) 380 } 381 return refLinks, nil 382} 383 384func getPullBacklinks(e Execer, aturis []syntax.ATURI) ([]models.RichReferenceLink, error) { 385 if len(aturis) == 0 { 386 return nil, nil 387 } 388 vals := make([]string, len(aturis)) 389 args := make([]any, 0, len(aturis)*2) 390 for i, aturi := range aturis { 391 vals[i] = "(?, ?)" 392 did := aturi.Authority().String() 393 rkey := aturi.RecordKey().String() 394 args = append(args, did, rkey) 395 } 396 rows, err := e.Query( 397 fmt.Sprintf( 398 `select r.did, r.name, p.pull_id, p.title, p.state 399 from pulls p 400 join repos r 401 on r.at_uri = p.repo_at 402 where (p.owner_did, p.rkey) in (%s)`, 403 strings.Join(vals, ","), 404 ), 405 args..., 406 ) 407 if err != nil { 408 return nil, err 409 } 410 defer rows.Close() 411 var refLinks []models.RichReferenceLink 412 for rows.Next() { 413 var l models.RichReferenceLink 414 l.Kind = models.RefKindPull 415 if err := rows.Scan(&l.Handle, &l.Repo, &l.SubjectId, &l.Title, &l.State); err != nil { 416 return nil, err 417 } 418 refLinks = append(refLinks, l) 419 } 420 if err := rows.Err(); err != nil { 421 return nil, fmt.Errorf("iterate rows: %w", err) 422 } 423 return refLinks, nil 424} 425 426func getPullCommentBacklinks(e Execer, aturis []syntax.ATURI) ([]models.RichReferenceLink, error) { 427 if len(aturis) == 0 { 428 return nil, nil 429 } 430 filter := FilterIn("c.comment_at", aturis) 431 rows, err := e.Query( 432 fmt.Sprintf( 433 `select r.did, r.name, p.pull_id, c.id, p.title, p.state 434 from repos r 435 join pulls p 436 on r.at_uri = p.repo_at 437 join pull_comments c 438 on r.at_uri = c.repo_at and p.pull_id = c.pull_id 439 where %s`, 440 filter.Condition(), 441 ), 442 filter.Arg()..., 443 ) 444 if err != nil { 445 return nil, err 446 } 447 defer rows.Close() 448 var refLinks []models.RichReferenceLink 449 for rows.Next() { 450 var l models.RichReferenceLink 451 l.Kind = models.RefKindPull 452 l.CommentId = new(int) 453 if err := rows.Scan(&l.Handle, &l.Repo, &l.SubjectId, l.CommentId, &l.Title, &l.State); err != nil { 454 return nil, err 455 } 456 refLinks = append(refLinks, l) 457 } 458 if err := rows.Err(); err != nil { 459 return nil, fmt.Errorf("iterate rows: %w", err) 460 } 461 return refLinks, nil 462}