| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | /* | 
					
						
							|  |  |  |    GoToSocial | 
					
						
							| 
									
										
										
										
											2021-12-20 18:42:19 +01:00
										 |  |  |    Copyright (C) 2021-2022 GoToSocial Authors admin@gotosocial.org | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |    This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |    it under the terms of the GNU Affero General Public License as published by | 
					
						
							|  |  |  |    the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |    (at your option) any later version. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |    GNU Affero General Public License for more details. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  |    along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | package bundb | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2022-10-29 17:10:28 +02:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/db" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" | 
					
						
							| 
									
										
										
										
											2022-10-29 17:10:28 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/id" | 
					
						
							| 
									
										
										
										
											2022-07-19 09:47:55 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/log" | 
					
						
							| 
									
										
										
										
											2022-12-08 17:35:14 +00:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/state" | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 	"github.com/uptrace/bun" | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	"golang.org/x/exp/slices" | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | type timelineDB struct { | 
					
						
							| 
									
										
										
										
											2022-12-08 17:35:14 +00:00
										 |  |  | 	conn  *DBConn | 
					
						
							|  |  |  | 	state *state.State | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | func (t *timelineDB) GetHomeTimeline(ctx context.Context, accountID string, maxID string, sinceID string, minID string, limit int, local bool) ([]*gtsmodel.Status, db.Error) { | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 	// Ensure reasonable | 
					
						
							|  |  |  | 	if limit < 0 { | 
					
						
							|  |  |  | 		limit = 0 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 	// Make educated guess for slice size | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	statusIDs := make([]string, 0, limit) | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 	q := t.conn. | 
					
						
							|  |  |  | 		NewSelect(). | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		TableExpr("? AS ?", bun.Ident("statuses"), bun.Ident("status")). | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 		// Select only IDs from table | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		Column("status.id"). | 
					
						
							| 
									
										
										
										
											2021-08-15 18:43:08 +02:00
										 |  |  | 		// Find out who accountID follows. | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		Join("LEFT JOIN ? AS ? ON ? = ? AND ? = ?", | 
					
						
							|  |  |  | 			bun.Ident("follows"), | 
					
						
							|  |  |  | 			bun.Ident("follow"), | 
					
						
							|  |  |  | 			bun.Ident("follow.target_account_id"), | 
					
						
							|  |  |  | 			bun.Ident("status.account_id"), | 
					
						
							|  |  |  | 			bun.Ident("follow.account_id"), | 
					
						
							|  |  |  | 			accountID). | 
					
						
							| 
									
										
										
										
											2021-08-15 18:43:08 +02:00
										 |  |  | 		// Sort by highest ID (newest) to lowest ID (oldest) | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		Order("status.id DESC") | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 17:10:28 +02:00
										 |  |  | 	if maxID == "" { | 
					
						
							|  |  |  | 		var err error | 
					
						
							|  |  |  | 		// don't return statuses more than five minutes in the future | 
					
						
							|  |  |  | 		maxID, err = id.NewULIDFromTime(time.Now().Add(5 * time.Minute)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 17:10:28 +02:00
										 |  |  | 	// return only statuses LOWER (ie., older) than maxID | 
					
						
							|  |  |  | 	q = q.Where("? < ?", bun.Ident("status.id"), maxID) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	if sinceID != "" { | 
					
						
							| 
									
										
										
										
											2021-08-15 18:43:08 +02:00
										 |  |  | 		// return only statuses HIGHER (ie., newer) than sinceID | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		q = q.Where("? > ?", bun.Ident("status.id"), sinceID) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if minID != "" { | 
					
						
							| 
									
										
										
										
											2021-08-15 18:43:08 +02:00
										 |  |  | 		// return only statuses HIGHER (ie., newer) than minID | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		q = q.Where("? > ?", bun.Ident("status.id"), minID) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if local { | 
					
						
							| 
									
										
										
										
											2021-08-15 18:43:08 +02:00
										 |  |  | 		// return only statuses posted by local account havers | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		q = q.Where("? = ?", bun.Ident("status.local"), local) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if limit > 0 { | 
					
						
							| 
									
										
										
										
											2021-08-15 18:43:08 +02:00
										 |  |  | 		// limit amount of statuses returned | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 		q = q.Limit(limit) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 	// Use a WhereGroup here to specify that we want EITHER statuses posted by accounts that accountID follows, | 
					
						
							|  |  |  | 	// OR statuses posted by accountID itself (since a user should be able to see their own statuses). | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// This is equivalent to something like WHERE ... AND (... OR ...) | 
					
						
							|  |  |  | 	// See: https://bun.uptrace.dev/guide/queries.html#select | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 	q = q.WhereGroup(" AND ", func(*bun.SelectQuery) *bun.SelectQuery { | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 		return q. | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 			WhereOr("? = ?", bun.Ident("follow.account_id"), accountID). | 
					
						
							|  |  |  | 			WhereOr("? = ?", bun.Ident("status.account_id"), accountID) | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	if err := q.Scan(ctx, &statusIDs); err != nil { | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 		return nil, t.conn.ProcessError(err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	statuses := make([]*gtsmodel.Status, 0, len(statusIDs)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, id := range statusIDs { | 
					
						
							|  |  |  | 		// Fetch status from db for ID | 
					
						
							| 
									
										
										
										
											2022-12-08 17:35:14 +00:00
										 |  |  | 		status, err := t.state.DB.GetStatusByID(ctx, id) | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2022-07-19 09:47:55 +01:00
										 |  |  | 			log.Errorf("GetHomeTimeline: error fetching status %q: %v", id, err) | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Append status to slice | 
					
						
							|  |  |  | 		statuses = append(statuses, status) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 	return statuses, nil | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-14 09:30:01 +00:00
										 |  |  | func (t *timelineDB) GetPublicTimeline(ctx context.Context, maxID string, sinceID string, minID string, limit int, local bool) ([]*gtsmodel.Status, db.Error) { | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 	// Ensure reasonable | 
					
						
							|  |  |  | 	if limit < 0 { | 
					
						
							|  |  |  | 		limit = 0 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Make educated guess for slice size | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	statusIDs := make([]string, 0, limit) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 	q := t.conn. | 
					
						
							|  |  |  | 		NewSelect(). | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		TableExpr("? AS ?", bun.Ident("statuses"), bun.Ident("status")). | 
					
						
							|  |  |  | 		Column("status.id"). | 
					
						
							|  |  |  | 		Where("? = ?", bun.Ident("status.visibility"), gtsmodel.VisibilityPublic). | 
					
						
							|  |  |  | 		WhereGroup(" AND ", whereEmptyOrNull("status.in_reply_to_id")). | 
					
						
							|  |  |  | 		WhereGroup(" AND ", whereEmptyOrNull("status.in_reply_to_uri")). | 
					
						
							|  |  |  | 		WhereGroup(" AND ", whereEmptyOrNull("status.boost_of_id")). | 
					
						
							|  |  |  | 		Order("status.id DESC") | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 17:10:28 +02:00
										 |  |  | 	if maxID == "" { | 
					
						
							|  |  |  | 		var err error | 
					
						
							|  |  |  | 		// don't return statuses more than five minutes in the future | 
					
						
							|  |  |  | 		maxID, err = id.NewULIDFromTime(time.Now().Add(5 * time.Minute)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-29 17:10:28 +02:00
										 |  |  | 	// return only statuses LOWER (ie., older) than maxID | 
					
						
							|  |  |  | 	q = q.Where("? < ?", bun.Ident("status.id"), maxID) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	if sinceID != "" { | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		q = q.Where("? > ?", bun.Ident("status.id"), sinceID) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if minID != "" { | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		q = q.Where("? > ?", bun.Ident("status.id"), minID) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if local { | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		q = q.Where("? = ?", bun.Ident("status.local"), local) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if limit > 0 { | 
					
						
							|  |  |  | 		q = q.Limit(limit) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	if err := q.Scan(ctx, &statusIDs); err != nil { | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 		return nil, t.conn.ProcessError(err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	statuses := make([]*gtsmodel.Status, 0, len(statusIDs)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, id := range statusIDs { | 
					
						
							|  |  |  | 		// Fetch status from db for ID | 
					
						
							| 
									
										
										
										
											2022-12-08 17:35:14 +00:00
										 |  |  | 		status, err := t.state.DB.GetStatusByID(ctx, id) | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2022-07-19 09:47:55 +01:00
										 |  |  | 			log.Errorf("GetPublicTimeline: error fetching status %q: %v", id, err) | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Append status to slice | 
					
						
							|  |  |  | 		statuses = append(statuses, status) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 	return statuses, nil | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TODO optimize this query and the logic here, because it's slow as balls -- it takes like a literal second to return with a limit of 20! | 
					
						
							|  |  |  | // It might be worth serving it through a timeline instead of raw DB queries, like we do for Home feeds. | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | func (t *timelineDB) GetFavedTimeline(ctx context.Context, accountID string, maxID string, minID string, limit int) ([]*gtsmodel.Status, string, string, db.Error) { | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 	// Ensure reasonable | 
					
						
							|  |  |  | 	if limit < 0 { | 
					
						
							|  |  |  | 		limit = 0 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 	// Make educated guess for slice size | 
					
						
							|  |  |  | 	faves := make([]*gtsmodel.StatusFave, 0, limit) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 	fq := t.conn. | 
					
						
							|  |  |  | 		NewSelect(). | 
					
						
							|  |  |  | 		Model(&faves). | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		Where("? = ?", bun.Ident("status_fave.account_id"), accountID). | 
					
						
							|  |  |  | 		Order("status_fave.id DESC") | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if maxID != "" { | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		fq = fq.Where("? < ?", bun.Ident("status_fave.id"), maxID) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if minID != "" { | 
					
						
							| 
									
										
										
										
											2022-10-08 13:50:48 +02:00
										 |  |  | 		fq = fq.Where("? > ?", bun.Ident("status_fave.id"), minID) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if limit > 0 { | 
					
						
							|  |  |  | 		fq = fq.Limit(limit) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 	err := fq.Scan(ctx) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 		return nil, "", "", t.conn.ProcessError(err) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if len(faves) == 0 { | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 		return nil, "", "", db.ErrNoEntries | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	// Sort by favourite ID rather than status ID | 
					
						
							|  |  |  | 	slices.SortFunc(faves, func(a, b *gtsmodel.StatusFave) bool { | 
					
						
							| 
									
										
										
										
											2022-12-09 11:46:52 +01:00
										 |  |  | 		return b.CreatedAt.Before(a.CreatedAt) | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	statuses := make([]*gtsmodel.Status, 0, len(faves)) | 
					
						
							| 
									
										
										
										
											2021-08-29 15:41:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 	for _, fave := range faves { | 
					
						
							|  |  |  | 		// Fetch status from db for corresponding favourite | 
					
						
							| 
									
										
										
										
											2022-12-08 17:35:14 +00:00
										 |  |  | 		status, err := t.state.DB.GetStatusByID(ctx, fave.StatusID) | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2022-07-19 09:47:55 +01:00
										 |  |  | 			log.Errorf("GetFavedTimeline: error fetching status for fave %q: %v", fave.ID, err) | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 16:18:21 +01:00
										 |  |  | 		// Append status to slice | 
					
						
							|  |  |  | 		statuses = append(statuses, status) | 
					
						
							| 
									
										
										
										
											2021-07-09 18:32:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	nextMaxID := faves[len(faves)-1].ID | 
					
						
							|  |  |  | 	prevMinID := faves[0].ID | 
					
						
							|  |  |  | 	return statuses, nextMaxID, prevMinID, nil | 
					
						
							|  |  |  | } |