diff options
Diffstat (limited to 'internal/processing/timeline')
| -rw-r--r-- | internal/processing/timeline/common.go | 71 | ||||
| -rw-r--r-- | internal/processing/timeline/faved.go | 73 | ||||
| -rw-r--r-- | internal/processing/timeline/home.go | 133 | ||||
| -rw-r--r-- | internal/processing/timeline/list.go | 157 | ||||
| -rw-r--r-- | internal/processing/timeline/notification.go | 144 | ||||
| -rw-r--r-- | internal/processing/timeline/public.go | 88 | ||||
| -rw-r--r-- | internal/processing/timeline/timeline.go | 38 | 
7 files changed, 704 insertions, 0 deletions
diff --git a/internal/processing/timeline/common.go b/internal/processing/timeline/common.go new file mode 100644 index 000000000..6d29d81d6 --- /dev/null +++ b/internal/processing/timeline/common.go @@ -0,0 +1,71 @@ +// GoToSocial +// Copyright (C) GoToSocial Authors admin@gotosocial.org +// SPDX-License-Identifier: AGPL-3.0-or-later +// +// 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/>. + +package timeline + +import ( +	"context" + +	"github.com/superseriousbusiness/gotosocial/internal/timeline" +) + +// SkipInsert returns a function that satisifes SkipInsertFunction. +func SkipInsert() timeline.SkipInsertFunction { +	// Gap to allow between a status or boost of status, +	// and reinsertion of a new boost of that status. +	// This is useful to avoid a heavily boosted status +	// showing up way too often in a user's timeline. +	const boostReinsertionDepth = 50 + +	return func( +		ctx context.Context, +		newItemID string, +		newItemAccountID string, +		newItemBoostOfID string, +		newItemBoostOfAccountID string, +		nextItemID string, +		nextItemAccountID string, +		nextItemBoostOfID string, +		nextItemBoostOfAccountID string, +		depth int, +	) (bool, error) { +		if newItemID == nextItemID { +			// Don't insert duplicates. +			return true, nil +		} + +		if newItemBoostOfID != "" { +			if newItemBoostOfID == nextItemBoostOfID && +				depth < boostReinsertionDepth { +				// Don't insert boosts of items +				// we've seen boosted recently. +				return true, nil +			} + +			if newItemBoostOfID == nextItemID && +				depth < boostReinsertionDepth { +				// Don't insert boosts of items when +				// we've seen the original recently. +				return true, nil +			} +		} + +		// Proceed with insertion +		// (that's what she said!). +		return false, nil +	} +} diff --git a/internal/processing/timeline/faved.go b/internal/processing/timeline/faved.go new file mode 100644 index 000000000..0fc92d8fa --- /dev/null +++ b/internal/processing/timeline/faved.go @@ -0,0 +1,73 @@ +// GoToSocial +// Copyright (C) GoToSocial Authors admin@gotosocial.org +// SPDX-License-Identifier: AGPL-3.0-or-later +// +// 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/>. + +package timeline + +import ( +	"context" +	"errors" +	"fmt" + +	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" +	"github.com/superseriousbusiness/gotosocial/internal/db" +	"github.com/superseriousbusiness/gotosocial/internal/gtserror" +	"github.com/superseriousbusiness/gotosocial/internal/log" +	"github.com/superseriousbusiness/gotosocial/internal/oauth" +	"github.com/superseriousbusiness/gotosocial/internal/util" +) + +func (p *Processor) FavedTimelineGet(ctx context.Context, authed *oauth.Auth, maxID string, minID string, limit int) (*apimodel.PageableResponse, gtserror.WithCode) { +	statuses, nextMaxID, prevMinID, err := p.state.DB.GetFavedTimeline(ctx, authed.Account.ID, maxID, minID, limit) +	if err != nil && !errors.Is(err, db.ErrNoEntries) { +		err = fmt.Errorf("FavedTimelineGet: db error getting statuses: %w", err) +		return nil, gtserror.NewErrorInternalError(err) +	} + +	count := len(statuses) +	if count == 0 { +		return util.EmptyPageableResponse(), nil +	} + +	items := make([]interface{}, 0, count) +	for _, s := range statuses { +		visible, err := p.filter.StatusVisible(ctx, authed.Account, s) +		if err != nil { +			log.Debugf(ctx, "skipping status %s because of an error checking status visibility: %s", s.ID, err) +			continue +		} + +		if !visible { +			continue +		} + +		apiStatus, err := p.tc.StatusToAPIStatus(ctx, s, authed.Account) +		if err != nil { +			log.Debugf(ctx, "skipping status %s because it couldn't be converted to its api representation: %s", s.ID, err) +			continue +		} + +		items = append(items, apiStatus) +	} + +	return util.PackagePageableResponse(util.PageableResponseParams{ +		Items:          items, +		Path:           "api/v1/favourites", +		NextMaxIDValue: nextMaxID, +		PrevMinIDValue: prevMinID, +		Limit:          limit, +	}) +} diff --git a/internal/processing/timeline/home.go b/internal/processing/timeline/home.go new file mode 100644 index 000000000..e65f12e17 --- /dev/null +++ b/internal/processing/timeline/home.go @@ -0,0 +1,133 @@ +// GoToSocial +// Copyright (C) GoToSocial Authors admin@gotosocial.org +// SPDX-License-Identifier: AGPL-3.0-or-later +// +// 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/>. + +package timeline + +import ( +	"context" +	"errors" +	"fmt" + +	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" +	"github.com/superseriousbusiness/gotosocial/internal/db" +	"github.com/superseriousbusiness/gotosocial/internal/gtserror" +	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" +	"github.com/superseriousbusiness/gotosocial/internal/oauth" +	"github.com/superseriousbusiness/gotosocial/internal/state" +	"github.com/superseriousbusiness/gotosocial/internal/timeline" +	"github.com/superseriousbusiness/gotosocial/internal/typeutils" +	"github.com/superseriousbusiness/gotosocial/internal/util" +	"github.com/superseriousbusiness/gotosocial/internal/visibility" +) + +// HomeTimelineGrab returns a function that satisfies GrabFunction for home timelines. +func HomeTimelineGrab(state *state.State) timeline.GrabFunction { +	return func(ctx context.Context, accountID string, maxID string, sinceID string, minID string, limit int) ([]timeline.Timelineable, bool, error) { +		statuses, err := state.DB.GetHomeTimeline(ctx, accountID, maxID, sinceID, minID, limit, false) +		if err != nil { +			if errors.Is(err, db.ErrNoEntries) { +				return nil, true, nil // we just don't have enough statuses left in the db so return stop = true +			} +			return nil, false, fmt.Errorf("HomeTimelineGrab: error getting statuses from db: %w", err) +		} + +		items := make([]timeline.Timelineable, len(statuses)) +		for i, s := range statuses { +			items[i] = s +		} + +		return items, false, nil +	} +} + +// HomeTimelineFilter returns a function that satisfies FilterFunction for home timelines. +func HomeTimelineFilter(state *state.State, filter *visibility.Filter) timeline.FilterFunction { +	return func(ctx context.Context, accountID string, item timeline.Timelineable) (shouldIndex bool, err error) { +		status, ok := item.(*gtsmodel.Status) +		if !ok { +			return false, errors.New("HomeTimelineFilter: could not convert item to *gtsmodel.Status") +		} + +		requestingAccount, err := state.DB.GetAccountByID(ctx, accountID) +		if err != nil { +			return false, fmt.Errorf("HomeTimelineFilter: error getting account with id %s: %w", accountID, err) +		} + +		timelineable, err := filter.StatusHomeTimelineable(ctx, requestingAccount, status) +		if err != nil { +			return false, fmt.Errorf("HomeTimelineFilter: error checking hometimelineability of status %s for account %s: %w", status.ID, accountID, err) +		} + +		return timelineable, nil +	} +} + +// HomeTimelineStatusPrepare returns a function that satisfies PrepareFunction for home timelines. +func HomeTimelineStatusPrepare(state *state.State, tc typeutils.TypeConverter) timeline.PrepareFunction { +	return func(ctx context.Context, accountID string, itemID string) (timeline.Preparable, error) { +		status, err := state.DB.GetStatusByID(ctx, itemID) +		if err != nil { +			return nil, fmt.Errorf("StatusPrepare: error getting status with id %s: %w", itemID, err) +		} + +		requestingAccount, err := state.DB.GetAccountByID(ctx, accountID) +		if err != nil { +			return nil, fmt.Errorf("StatusPrepare: error getting account with id %s: %w", accountID, err) +		} + +		return tc.StatusToAPIStatus(ctx, status, requestingAccount) +	} +} + +func (p *Processor) HomeTimelineGet(ctx context.Context, authed *oauth.Auth, maxID string, sinceID string, minID string, limit int, local bool) (*apimodel.PageableResponse, gtserror.WithCode) { +	statuses, err := p.state.Timelines.Home.GetTimeline(ctx, authed.Account.ID, maxID, sinceID, minID, limit, local) +	if err != nil { +		err = fmt.Errorf("HomeTimelineGet: error getting statuses: %w", err) +		return nil, gtserror.NewErrorInternalError(err) +	} + +	count := len(statuses) +	if count == 0 { +		return util.EmptyPageableResponse(), nil +	} + +	var ( +		items          = make([]interface{}, count) +		nextMaxIDValue string +		prevMinIDValue string +	) + +	for i, item := range statuses { +		if i == count-1 { +			nextMaxIDValue = item.GetID() +		} + +		if i == 0 { +			prevMinIDValue = item.GetID() +		} + +		items[i] = item +	} + +	return util.PackagePageableResponse(util.PageableResponseParams{ +		Items:          items, +		Path:           "api/v1/timelines/home", +		NextMaxIDValue: nextMaxIDValue, +		PrevMinIDValue: prevMinIDValue, +		Limit:          limit, +	}) +} diff --git a/internal/processing/timeline/list.go b/internal/processing/timeline/list.go new file mode 100644 index 000000000..adad35197 --- /dev/null +++ b/internal/processing/timeline/list.go @@ -0,0 +1,157 @@ +// GoToSocial +// Copyright (C) GoToSocial Authors admin@gotosocial.org +// SPDX-License-Identifier: AGPL-3.0-or-later +// +// 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/>. + +package timeline + +import ( +	"context" +	"errors" +	"fmt" + +	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" +	"github.com/superseriousbusiness/gotosocial/internal/db" +	"github.com/superseriousbusiness/gotosocial/internal/gtserror" +	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" +	"github.com/superseriousbusiness/gotosocial/internal/oauth" +	"github.com/superseriousbusiness/gotosocial/internal/state" +	"github.com/superseriousbusiness/gotosocial/internal/timeline" +	"github.com/superseriousbusiness/gotosocial/internal/typeutils" +	"github.com/superseriousbusiness/gotosocial/internal/util" +	"github.com/superseriousbusiness/gotosocial/internal/visibility" +) + +// ListTimelineGrab returns a function that satisfies GrabFunction for list timelines. +func ListTimelineGrab(state *state.State) timeline.GrabFunction { +	return func(ctx context.Context, listID string, maxID string, sinceID string, minID string, limit int) ([]timeline.Timelineable, bool, error) { +		statuses, err := state.DB.GetListTimeline(ctx, listID, maxID, sinceID, minID, limit) +		if err != nil { +			if errors.Is(err, db.ErrNoEntries) { +				return nil, true, nil // we just don't have enough statuses left in the db so return stop = true +			} +			return nil, false, fmt.Errorf("ListTimelineGrab: error getting statuses from db: %w", err) +		} + +		items := make([]timeline.Timelineable, len(statuses)) +		for i, s := range statuses { +			items[i] = s +		} + +		return items, false, nil +	} +} + +// HomeTimelineFilter returns a function that satisfies FilterFunction for list timelines. +func ListTimelineFilter(state *state.State, filter *visibility.Filter) timeline.FilterFunction { +	return func(ctx context.Context, listID string, item timeline.Timelineable) (shouldIndex bool, err error) { +		status, ok := item.(*gtsmodel.Status) +		if !ok { +			return false, errors.New("ListTimelineFilter: could not convert item to *gtsmodel.Status") +		} + +		list, err := state.DB.GetListByID(ctx, listID) +		if err != nil { +			return false, fmt.Errorf("ListTimelineFilter: error getting list with id %s: %w", listID, err) +		} + +		requestingAccount, err := state.DB.GetAccountByID(ctx, list.AccountID) +		if err != nil { +			return false, fmt.Errorf("ListTimelineFilter: error getting account with id %s: %w", list.AccountID, err) +		} + +		timelineable, err := filter.StatusHomeTimelineable(ctx, requestingAccount, status) +		if err != nil { +			return false, fmt.Errorf("ListTimelineFilter: error checking hometimelineability of status %s for account %s: %w", status.ID, list.AccountID, err) +		} + +		return timelineable, nil +	} +} + +// ListTimelineStatusPrepare returns a function that satisfies PrepareFunction for list timelines. +func ListTimelineStatusPrepare(state *state.State, tc typeutils.TypeConverter) timeline.PrepareFunction { +	return func(ctx context.Context, listID string, itemID string) (timeline.Preparable, error) { +		status, err := state.DB.GetStatusByID(ctx, itemID) +		if err != nil { +			return nil, fmt.Errorf("ListTimelineStatusPrepare: error getting status with id %s: %w", itemID, err) +		} + +		list, err := state.DB.GetListByID(ctx, listID) +		if err != nil { +			return nil, fmt.Errorf("ListTimelineStatusPrepare: error getting list with id %s: %w", listID, err) +		} + +		requestingAccount, err := state.DB.GetAccountByID(ctx, list.AccountID) +		if err != nil { +			return nil, fmt.Errorf("ListTimelineStatusPrepare: error getting account with id %s: %w", list.AccountID, err) +		} + +		return tc.StatusToAPIStatus(ctx, status, requestingAccount) +	} +} + +func (p *Processor) ListTimelineGet(ctx context.Context, authed *oauth.Auth, listID string, maxID string, sinceID string, minID string, limit int) (*apimodel.PageableResponse, gtserror.WithCode) { +	// Ensure list exists + is owned by this account. +	list, err := p.state.DB.GetListByID(ctx, listID) +	if err != nil { +		if errors.Is(err, db.ErrNoEntries) { +			return nil, gtserror.NewErrorNotFound(err) +		} +		return nil, gtserror.NewErrorInternalError(err) +	} + +	if list.AccountID != authed.Account.ID { +		err = fmt.Errorf("list with id %s does not belong to account %s", list.ID, authed.Account.ID) +		return nil, gtserror.NewErrorNotFound(err) +	} + +	statuses, err := p.state.Timelines.List.GetTimeline(ctx, listID, maxID, sinceID, minID, limit, false) +	if err != nil { +		err = fmt.Errorf("ListTimelineGet: error getting statuses: %w", err) +		return nil, gtserror.NewErrorInternalError(err) +	} + +	count := len(statuses) +	if count == 0 { +		return util.EmptyPageableResponse(), nil +	} + +	var ( +		items          = make([]interface{}, count) +		nextMaxIDValue string +		prevMinIDValue string +	) + +	for i, item := range statuses { +		if i == count-1 { +			nextMaxIDValue = item.GetID() +		} + +		if i == 0 { +			prevMinIDValue = item.GetID() +		} + +		items[i] = item +	} + +	return util.PackagePageableResponse(util.PageableResponseParams{ +		Items:          items, +		Path:           "api/v1/timelines/list/" + listID, +		NextMaxIDValue: nextMaxIDValue, +		PrevMinIDValue: prevMinIDValue, +		Limit:          limit, +	}) +} diff --git a/internal/processing/timeline/notification.go b/internal/processing/timeline/notification.go new file mode 100644 index 000000000..4a79fb82a --- /dev/null +++ b/internal/processing/timeline/notification.go @@ -0,0 +1,144 @@ +// GoToSocial +// Copyright (C) GoToSocial Authors admin@gotosocial.org +// SPDX-License-Identifier: AGPL-3.0-or-later +// +// 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/>. + +package timeline + +import ( +	"context" +	"errors" +	"fmt" + +	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" +	"github.com/superseriousbusiness/gotosocial/internal/db" +	"github.com/superseriousbusiness/gotosocial/internal/gtserror" +	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" +	"github.com/superseriousbusiness/gotosocial/internal/log" +	"github.com/superseriousbusiness/gotosocial/internal/oauth" +	"github.com/superseriousbusiness/gotosocial/internal/util" +) + +func (p *Processor) NotificationsGet(ctx context.Context, authed *oauth.Auth, maxID string, sinceID string, minID string, limit int, excludeTypes []string) (*apimodel.PageableResponse, gtserror.WithCode) { +	notifs, err := p.state.DB.GetAccountNotifications(ctx, authed.Account.ID, maxID, sinceID, minID, limit, excludeTypes) +	if err != nil && !errors.Is(err, db.ErrNoEntries) { +		err = fmt.Errorf("NotificationsGet: db error getting notifications: %w", err) +		return nil, gtserror.NewErrorInternalError(err) +	} + +	count := len(notifs) +	if count == 0 { +		return util.EmptyPageableResponse(), nil +	} + +	var ( +		items          = make([]interface{}, 0, count) +		nextMaxIDValue string +		prevMinIDValue string +	) + +	for i, n := range notifs { +		// Set next + prev values before filtering and API +		// converting, so caller can still page properly. +		if i == count-1 { +			nextMaxIDValue = n.ID +		} + +		if i == 0 { +			prevMinIDValue = n.ID +		} + +		// Ensure this notification should be shown to requester. +		if n.OriginAccount != nil { +			// Account is set, ensure it's visible to notif target. +			visible, err := p.filter.AccountVisible(ctx, authed.Account, n.OriginAccount) +			if err != nil { +				log.Debugf(ctx, "skipping notification %s because of an error checking notification visibility: %s", n.ID, err) +				continue +			} + +			if !visible { +				continue +			} +		} + +		if n.Status != nil { +			// Status is set, ensure it's visible to notif target. +			visible, err := p.filter.StatusVisible(ctx, authed.Account, n.Status) +			if err != nil { +				log.Debugf(ctx, "skipping notification %s because of an error checking notification visibility: %s", n.ID, err) +				continue +			} + +			if !visible { +				continue +			} +		} + +		item, err := p.tc.NotificationToAPINotification(ctx, n) +		if err != nil { +			log.Debugf(ctx, "skipping notification %s because it couldn't be converted to its api representation: %s", n.ID, err) +			continue +		} + +		items = append(items, item) +	} + +	return util.PackagePageableResponse(util.PageableResponseParams{ +		Items:          items, +		Path:           "api/v1/notifications", +		NextMaxIDValue: nextMaxIDValue, +		PrevMinIDValue: prevMinIDValue, +		Limit:          limit, +	}) +} + +func (p *Processor) NotificationGet(ctx context.Context, account *gtsmodel.Account, targetNotifID string) (*apimodel.Notification, gtserror.WithCode) { +	notif, err := p.state.DB.GetNotificationByID(ctx, targetNotifID) +	if err != nil { +		if errors.Is(err, db.ErrNoEntries) { +			return nil, gtserror.NewErrorNotFound(err) +		} + +		// Real error. +		return nil, gtserror.NewErrorInternalError(err) +	} + +	if notifTargetAccountID := notif.TargetAccountID; notifTargetAccountID != account.ID { +		err = fmt.Errorf("account %s does not have permission to view notification belong to account %s", account.ID, notifTargetAccountID) +		return nil, gtserror.NewErrorNotFound(err) +	} + +	apiNotif, err := p.tc.NotificationToAPINotification(ctx, notif) +	if err != nil { +		if errors.Is(err, db.ErrNoEntries) { +			return nil, gtserror.NewErrorNotFound(err) +		} + +		// Real error. +		return nil, gtserror.NewErrorInternalError(err) +	} + +	return apiNotif, nil +} + +func (p *Processor) NotificationsClear(ctx context.Context, authed *oauth.Auth) gtserror.WithCode { +	// Delete all notifications of all types that target the authorized account. +	if err := p.state.DB.DeleteNotifications(ctx, nil, authed.Account.ID, ""); err != nil && !errors.Is(err, db.ErrNoEntries) { +		return gtserror.NewErrorInternalError(err) +	} + +	return nil +} diff --git a/internal/processing/timeline/public.go b/internal/processing/timeline/public.go new file mode 100644 index 000000000..67893ecfa --- /dev/null +++ b/internal/processing/timeline/public.go @@ -0,0 +1,88 @@ +// GoToSocial +// Copyright (C) GoToSocial Authors admin@gotosocial.org +// SPDX-License-Identifier: AGPL-3.0-or-later +// +// 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/>. + +package timeline + +import ( +	"context" +	"errors" +	"fmt" + +	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" +	"github.com/superseriousbusiness/gotosocial/internal/db" +	"github.com/superseriousbusiness/gotosocial/internal/gtserror" +	"github.com/superseriousbusiness/gotosocial/internal/log" +	"github.com/superseriousbusiness/gotosocial/internal/oauth" +	"github.com/superseriousbusiness/gotosocial/internal/util" +) + +func (p *Processor) PublicTimelineGet(ctx context.Context, authed *oauth.Auth, maxID string, sinceID string, minID string, limit int, local bool) (*apimodel.PageableResponse, gtserror.WithCode) { +	statuses, err := p.state.DB.GetPublicTimeline(ctx, maxID, sinceID, minID, limit, local) +	if err != nil && !errors.Is(err, db.ErrNoEntries) { +		err = fmt.Errorf("PublicTimelineGet: db error getting statuses: %w", err) +		return nil, gtserror.NewErrorInternalError(err) +	} + +	count := len(statuses) +	if count == 0 { +		return util.EmptyPageableResponse(), nil +	} + +	var ( +		items          = make([]interface{}, 0, count) +		nextMaxIDValue string +		prevMinIDValue string +	) + +	for i, s := range statuses { +		// Set next + prev values before filtering and API +		// converting, so caller can still page properly. +		if i == count-1 { +			nextMaxIDValue = s.ID +		} + +		if i == 0 { +			prevMinIDValue = s.ID +		} + +		timelineable, err := p.filter.StatusPublicTimelineable(ctx, authed.Account, s) +		if err != nil { +			log.Debugf(ctx, "skipping status %s because of an error checking StatusPublicTimelineable: %s", s.ID, err) +			continue +		} + +		if !timelineable { +			continue +		} + +		apiStatus, err := p.tc.StatusToAPIStatus(ctx, s, authed.Account) +		if err != nil { +			log.Debugf(ctx, "skipping status %s because it couldn't be converted to its api representation: %s", s.ID, err) +			continue +		} + +		items = append(items, apiStatus) +	} + +	return util.PackagePageableResponse(util.PageableResponseParams{ +		Items:          items, +		Path:           "api/v1/timelines/public", +		NextMaxIDValue: nextMaxIDValue, +		PrevMinIDValue: prevMinIDValue, +		Limit:          limit, +	}) +} diff --git a/internal/processing/timeline/timeline.go b/internal/processing/timeline/timeline.go new file mode 100644 index 000000000..7a95f9a11 --- /dev/null +++ b/internal/processing/timeline/timeline.go @@ -0,0 +1,38 @@ +// GoToSocial +// Copyright (C) GoToSocial Authors admin@gotosocial.org +// SPDX-License-Identifier: AGPL-3.0-or-later +// +// 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/>. + +package timeline + +import ( +	"github.com/superseriousbusiness/gotosocial/internal/state" +	"github.com/superseriousbusiness/gotosocial/internal/typeutils" +	"github.com/superseriousbusiness/gotosocial/internal/visibility" +) + +type Processor struct { +	state  *state.State +	tc     typeutils.TypeConverter +	filter *visibility.Filter +} + +func New(state *state.State, tc typeutils.TypeConverter, filter *visibility.Filter) Processor { +	return Processor{ +		state:  state, +		tc:     tc, +		filter: filter, +	} +}  | 
