mirror of
https://github.com/superseriousbusiness/gotosocial
synced 2024-12-16 18:50:33 +01:00
1022 lines
34 KiB
Go
1022 lines
34 KiB
Go
// 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 processing
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"net/url"
|
|
|
|
"codeberg.org/gruf/go-kv"
|
|
"codeberg.org/gruf/go-logger/v2/level"
|
|
"github.com/superseriousbusiness/activity/pub"
|
|
"github.com/superseriousbusiness/activity/streams"
|
|
"github.com/superseriousbusiness/gotosocial/internal/ap"
|
|
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
|
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
|
"github.com/superseriousbusiness/gotosocial/internal/log"
|
|
"github.com/superseriousbusiness/gotosocial/internal/messages"
|
|
)
|
|
|
|
func (p *Processor) ProcessFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
// Allocate new log fields slice
|
|
fields := make([]kv.Field, 3, 4)
|
|
fields[0] = kv.Field{"activityType", clientMsg.APActivityType}
|
|
fields[1] = kv.Field{"objectType", clientMsg.APObjectType}
|
|
fields[2] = kv.Field{"fromAccount", clientMsg.OriginAccount.Username}
|
|
|
|
if clientMsg.GTSModel != nil &&
|
|
log.Level() >= level.DEBUG {
|
|
// Append converted model to log
|
|
fields = append(fields, kv.Field{
|
|
"model", clientMsg.GTSModel,
|
|
})
|
|
}
|
|
|
|
// Log this federated message
|
|
l := log.WithContext(ctx).WithFields(fields...)
|
|
l.Info("processing from client")
|
|
|
|
switch clientMsg.APActivityType {
|
|
case ap.ActivityCreate:
|
|
// CREATE
|
|
switch clientMsg.APObjectType {
|
|
case ap.ObjectProfile, ap.ActorPerson:
|
|
// CREATE ACCOUNT/PROFILE
|
|
return p.processCreateAccountFromClientAPI(ctx, clientMsg)
|
|
case ap.ObjectNote:
|
|
// CREATE NOTE
|
|
return p.processCreateStatusFromClientAPI(ctx, clientMsg)
|
|
case ap.ActivityFollow:
|
|
// CREATE FOLLOW REQUEST
|
|
return p.processCreateFollowRequestFromClientAPI(ctx, clientMsg)
|
|
case ap.ActivityLike:
|
|
// CREATE LIKE/FAVE
|
|
return p.processCreateFaveFromClientAPI(ctx, clientMsg)
|
|
case ap.ActivityAnnounce:
|
|
// CREATE BOOST/ANNOUNCE
|
|
return p.processCreateAnnounceFromClientAPI(ctx, clientMsg)
|
|
case ap.ActivityBlock:
|
|
// CREATE BLOCK
|
|
return p.processCreateBlockFromClientAPI(ctx, clientMsg)
|
|
}
|
|
case ap.ActivityUpdate:
|
|
// UPDATE
|
|
switch clientMsg.APObjectType {
|
|
case ap.ObjectProfile, ap.ActorPerson:
|
|
// UPDATE ACCOUNT/PROFILE
|
|
return p.processUpdateAccountFromClientAPI(ctx, clientMsg)
|
|
case ap.ActivityFlag:
|
|
// UPDATE A FLAG/REPORT (mark as resolved/closed)
|
|
return p.processUpdateReportFromClientAPI(ctx, clientMsg)
|
|
}
|
|
case ap.ActivityAccept:
|
|
// ACCEPT
|
|
if clientMsg.APObjectType == ap.ActivityFollow {
|
|
// ACCEPT FOLLOW
|
|
return p.processAcceptFollowFromClientAPI(ctx, clientMsg)
|
|
}
|
|
case ap.ActivityReject:
|
|
// REJECT
|
|
if clientMsg.APObjectType == ap.ActivityFollow {
|
|
// REJECT FOLLOW (request)
|
|
return p.processRejectFollowFromClientAPI(ctx, clientMsg)
|
|
}
|
|
case ap.ActivityUndo:
|
|
// UNDO
|
|
switch clientMsg.APObjectType {
|
|
case ap.ActivityFollow:
|
|
// UNDO FOLLOW
|
|
return p.processUndoFollowFromClientAPI(ctx, clientMsg)
|
|
case ap.ActivityBlock:
|
|
// UNDO BLOCK
|
|
return p.processUndoBlockFromClientAPI(ctx, clientMsg)
|
|
case ap.ActivityLike:
|
|
// UNDO LIKE/FAVE
|
|
return p.processUndoFaveFromClientAPI(ctx, clientMsg)
|
|
case ap.ActivityAnnounce:
|
|
// UNDO ANNOUNCE/BOOST
|
|
return p.processUndoAnnounceFromClientAPI(ctx, clientMsg)
|
|
}
|
|
case ap.ActivityDelete:
|
|
// DELETE
|
|
switch clientMsg.APObjectType {
|
|
case ap.ObjectNote:
|
|
// DELETE STATUS/NOTE
|
|
return p.processDeleteStatusFromClientAPI(ctx, clientMsg)
|
|
case ap.ObjectProfile, ap.ActorPerson:
|
|
// DELETE ACCOUNT/PROFILE
|
|
return p.processDeleteAccountFromClientAPI(ctx, clientMsg)
|
|
}
|
|
case ap.ActivityFlag:
|
|
// FLAG
|
|
if clientMsg.APObjectType == ap.ObjectProfile {
|
|
// FLAG/REPORT A PROFILE
|
|
return p.processReportAccountFromClientAPI(ctx, clientMsg)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (p *Processor) processCreateAccountFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
account, ok := clientMsg.GTSModel.(*gtsmodel.Account)
|
|
if !ok {
|
|
return errors.New("account was not parseable as *gtsmodel.Account")
|
|
}
|
|
|
|
// Do nothing if this isn't our activity.
|
|
if !account.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
// get the user this account belongs to
|
|
user, err := p.state.DB.GetUserByAccountID(ctx, account.ID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// email a confirmation to this user
|
|
return p.User().EmailSendConfirmation(ctx, user, account.Username)
|
|
}
|
|
|
|
func (p *Processor) processCreateStatusFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
status, ok := clientMsg.GTSModel.(*gtsmodel.Status)
|
|
if !ok {
|
|
return gtserror.New("status was not parseable as *gtsmodel.Status")
|
|
}
|
|
|
|
if err := p.timelineAndNotifyStatus(ctx, status); err != nil {
|
|
return gtserror.Newf("error timelining status: %w", err)
|
|
}
|
|
|
|
if status.InReplyToID != "" {
|
|
// Interaction counts changed on the replied status;
|
|
// uncache the prepared version from all timelines.
|
|
p.invalidateStatusFromTimelines(ctx, status.InReplyToID)
|
|
}
|
|
|
|
if err := p.federateStatus(ctx, status); err != nil {
|
|
return gtserror.Newf("error federating status: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *Processor) processCreateFollowRequestFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
followRequest, ok := clientMsg.GTSModel.(*gtsmodel.FollowRequest)
|
|
if !ok {
|
|
return errors.New("followrequest was not parseable as *gtsmodel.FollowRequest")
|
|
}
|
|
|
|
if err := p.notifyFollowRequest(ctx, followRequest); err != nil {
|
|
return err
|
|
}
|
|
|
|
return p.federateFollow(ctx, followRequest, clientMsg.OriginAccount, clientMsg.TargetAccount)
|
|
}
|
|
|
|
func (p *Processor) processCreateFaveFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
statusFave, ok := clientMsg.GTSModel.(*gtsmodel.StatusFave)
|
|
if !ok {
|
|
return gtserror.New("statusFave was not parseable as *gtsmodel.StatusFave")
|
|
}
|
|
|
|
if err := p.notifyFave(ctx, statusFave); err != nil {
|
|
return gtserror.Newf("error notifying status fave: %w", err)
|
|
}
|
|
|
|
// Interaction counts changed on the faved status;
|
|
// uncache the prepared version from all timelines.
|
|
p.invalidateStatusFromTimelines(ctx, statusFave.StatusID)
|
|
|
|
if err := p.federateFave(ctx, statusFave, clientMsg.OriginAccount, clientMsg.TargetAccount); err != nil {
|
|
return gtserror.Newf("error federating status fave: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *Processor) processCreateAnnounceFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
status, ok := clientMsg.GTSModel.(*gtsmodel.Status)
|
|
if !ok {
|
|
return errors.New("boost was not parseable as *gtsmodel.Status")
|
|
}
|
|
|
|
// Timeline and notify.
|
|
if err := p.timelineAndNotifyStatus(ctx, status); err != nil {
|
|
return gtserror.Newf("error timelining boost: %w", err)
|
|
}
|
|
|
|
if err := p.notifyAnnounce(ctx, status); err != nil {
|
|
return gtserror.Newf("error notifying boost: %w", err)
|
|
}
|
|
|
|
// Interaction counts changed on the boosted status;
|
|
// uncache the prepared version from all timelines.
|
|
p.invalidateStatusFromTimelines(ctx, status.BoostOfID)
|
|
|
|
if err := p.federateAnnounce(ctx, status, clientMsg.OriginAccount, clientMsg.TargetAccount); err != nil {
|
|
return gtserror.Newf("error federating boost: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *Processor) processCreateBlockFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
block, ok := clientMsg.GTSModel.(*gtsmodel.Block)
|
|
if !ok {
|
|
return errors.New("block was not parseable as *gtsmodel.Block")
|
|
}
|
|
|
|
// remove any of the blocking account's statuses from the blocked account's timeline, and vice versa
|
|
if err := p.state.Timelines.Home.WipeItemsFromAccountID(ctx, block.AccountID, block.TargetAccountID); err != nil {
|
|
return err
|
|
}
|
|
if err := p.state.Timelines.Home.WipeItemsFromAccountID(ctx, block.TargetAccountID, block.AccountID); err != nil {
|
|
return err
|
|
}
|
|
|
|
// TODO: same with notifications
|
|
// TODO: same with bookmarks
|
|
|
|
return p.federateBlock(ctx, block)
|
|
}
|
|
|
|
func (p *Processor) processUpdateAccountFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
account, ok := clientMsg.GTSModel.(*gtsmodel.Account)
|
|
if !ok {
|
|
return errors.New("account was not parseable as *gtsmodel.Account")
|
|
}
|
|
|
|
return p.federateAccountUpdate(ctx, account, clientMsg.OriginAccount)
|
|
}
|
|
|
|
func (p *Processor) processUpdateReportFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
report, ok := clientMsg.GTSModel.(*gtsmodel.Report)
|
|
if !ok {
|
|
return errors.New("report was not parseable as *gtsmodel.Report")
|
|
}
|
|
|
|
if report.Account.IsRemote() {
|
|
// Report creator is a remote account,
|
|
// we shouldn't email or notify them.
|
|
return nil
|
|
}
|
|
|
|
return p.emailReportClosed(ctx, report)
|
|
}
|
|
|
|
func (p *Processor) processAcceptFollowFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
follow, ok := clientMsg.GTSModel.(*gtsmodel.Follow)
|
|
if !ok {
|
|
return errors.New("accept was not parseable as *gtsmodel.Follow")
|
|
}
|
|
|
|
if err := p.notifyFollow(ctx, follow, clientMsg.TargetAccount); err != nil {
|
|
return err
|
|
}
|
|
|
|
return p.federateAcceptFollowRequest(ctx, follow)
|
|
}
|
|
|
|
func (p *Processor) processRejectFollowFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
followRequest, ok := clientMsg.GTSModel.(*gtsmodel.FollowRequest)
|
|
if !ok {
|
|
return errors.New("reject was not parseable as *gtsmodel.FollowRequest")
|
|
}
|
|
|
|
return p.federateRejectFollowRequest(ctx, followRequest)
|
|
}
|
|
|
|
func (p *Processor) processUndoFollowFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
follow, ok := clientMsg.GTSModel.(*gtsmodel.Follow)
|
|
if !ok {
|
|
return errors.New("undo was not parseable as *gtsmodel.Follow")
|
|
}
|
|
return p.federateUnfollow(ctx, follow, clientMsg.OriginAccount, clientMsg.TargetAccount)
|
|
}
|
|
|
|
func (p *Processor) processUndoBlockFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
block, ok := clientMsg.GTSModel.(*gtsmodel.Block)
|
|
if !ok {
|
|
return errors.New("undo was not parseable as *gtsmodel.Block")
|
|
}
|
|
return p.federateUnblock(ctx, block)
|
|
}
|
|
|
|
func (p *Processor) processUndoFaveFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
statusFave, ok := clientMsg.GTSModel.(*gtsmodel.StatusFave)
|
|
if !ok {
|
|
return gtserror.New("statusFave was not parseable as *gtsmodel.StatusFave")
|
|
}
|
|
|
|
// Interaction counts changed on the faved status;
|
|
// uncache the prepared version from all timelines.
|
|
p.invalidateStatusFromTimelines(ctx, statusFave.StatusID)
|
|
|
|
if err := p.federateUnfave(ctx, statusFave, clientMsg.OriginAccount, clientMsg.TargetAccount); err != nil {
|
|
return gtserror.Newf("error federating status unfave: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *Processor) processUndoAnnounceFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
status, ok := clientMsg.GTSModel.(*gtsmodel.Status)
|
|
if !ok {
|
|
return errors.New("boost was not parseable as *gtsmodel.Status")
|
|
}
|
|
|
|
if err := p.state.DB.DeleteStatusByID(ctx, status.ID); err != nil {
|
|
return gtserror.Newf("db error deleting boost: %w", err)
|
|
}
|
|
|
|
if err := p.deleteStatusFromTimelines(ctx, status.ID); err != nil {
|
|
return gtserror.Newf("error removing boost from timelines: %w", err)
|
|
}
|
|
|
|
// Interaction counts changed on the boosted status;
|
|
// uncache the prepared version from all timelines.
|
|
p.invalidateStatusFromTimelines(ctx, status.BoostOfID)
|
|
|
|
if err := p.federateUnannounce(ctx, status, clientMsg.OriginAccount, clientMsg.TargetAccount); err != nil {
|
|
return gtserror.Newf("error federating status unboost: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *Processor) processDeleteStatusFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
status, ok := clientMsg.GTSModel.(*gtsmodel.Status)
|
|
if !ok {
|
|
return gtserror.New("status was not parseable as *gtsmodel.Status")
|
|
}
|
|
|
|
if err := p.state.DB.PopulateStatus(ctx, status); err != nil {
|
|
return gtserror.Newf("db error populating status: %w", err)
|
|
}
|
|
|
|
// Don't delete attachments, just unattach them: this
|
|
// request comes from the client API and the poster
|
|
// may want to use attachments again in a new post.
|
|
deleteAttachments := false
|
|
if err := p.wipeStatus(ctx, status, deleteAttachments); err != nil {
|
|
return gtserror.Newf("error wiping status: %w", err)
|
|
}
|
|
|
|
if status.InReplyToID != "" {
|
|
// Interaction counts changed on the replied status;
|
|
// uncache the prepared version from all timelines.
|
|
p.invalidateStatusFromTimelines(ctx, status.InReplyToID)
|
|
}
|
|
|
|
if err := p.federateStatusDelete(ctx, status); err != nil {
|
|
return gtserror.Newf("error federating status delete: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *Processor) processDeleteAccountFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
// the origin of the delete could be either a domain block, or an action by another (or this) account
|
|
var origin string
|
|
if domainBlock, ok := clientMsg.GTSModel.(*gtsmodel.DomainBlock); ok {
|
|
// origin is a domain block
|
|
origin = domainBlock.ID
|
|
} else {
|
|
// origin is whichever account caused this message
|
|
origin = clientMsg.OriginAccount.ID
|
|
}
|
|
|
|
if err := p.federateAccountDelete(ctx, clientMsg.TargetAccount); err != nil {
|
|
return err
|
|
}
|
|
|
|
return p.account.Delete(ctx, clientMsg.TargetAccount, origin)
|
|
}
|
|
|
|
func (p *Processor) processReportAccountFromClientAPI(ctx context.Context, clientMsg messages.FromClientAPI) error {
|
|
report, ok := clientMsg.GTSModel.(*gtsmodel.Report)
|
|
if !ok {
|
|
return errors.New("report was not parseable as *gtsmodel.Report")
|
|
}
|
|
|
|
if *report.Forwarded {
|
|
if err := p.federateReport(ctx, report); err != nil {
|
|
return fmt.Errorf("processReportAccountFromClientAPI: error federating report: %w", err)
|
|
}
|
|
}
|
|
|
|
if err := p.emailReport(ctx, report); err != nil {
|
|
return fmt.Errorf("processReportAccountFromClientAPI: error notifying report: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// TODO: move all the below functions into federation.Federator
|
|
|
|
func (p *Processor) federateAccountDelete(ctx context.Context, account *gtsmodel.Account) error {
|
|
// Do nothing if this isn't our activity.
|
|
if !account.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(account.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAccountDelete: error parsing outboxURI %s: %s", account.OutboxURI, err)
|
|
}
|
|
|
|
actorIRI, err := url.Parse(account.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAccountDelete: error parsing actorIRI %s: %s", account.URI, err)
|
|
}
|
|
|
|
followersIRI, err := url.Parse(account.FollowersURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAccountDelete: error parsing followersIRI %s: %s", account.FollowersURI, err)
|
|
}
|
|
|
|
publicIRI, err := url.Parse(pub.PublicActivityPubIRI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAccountDelete: error parsing url %s: %s", pub.PublicActivityPubIRI, err)
|
|
}
|
|
|
|
// create a delete and set the appropriate actor on it
|
|
delete := streams.NewActivityStreamsDelete()
|
|
|
|
// set the actor for the delete; no matter who deleted it we should use the account owner for this
|
|
deleteActor := streams.NewActivityStreamsActorProperty()
|
|
deleteActor.AppendIRI(actorIRI)
|
|
delete.SetActivityStreamsActor(deleteActor)
|
|
|
|
// Set the account IRI as the 'object' property.
|
|
deleteObject := streams.NewActivityStreamsObjectProperty()
|
|
deleteObject.AppendIRI(actorIRI)
|
|
delete.SetActivityStreamsObject(deleteObject)
|
|
|
|
// send to followers...
|
|
deleteTo := streams.NewActivityStreamsToProperty()
|
|
deleteTo.AppendIRI(followersIRI)
|
|
delete.SetActivityStreamsTo(deleteTo)
|
|
|
|
// ... and CC to public
|
|
deleteCC := streams.NewActivityStreamsCcProperty()
|
|
deleteCC.AppendIRI(publicIRI)
|
|
delete.SetActivityStreamsCc(deleteCC)
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, delete)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateStatus(ctx context.Context, status *gtsmodel.Status) error {
|
|
// do nothing if the status shouldn't be federated
|
|
if !*status.Federated {
|
|
return nil
|
|
}
|
|
|
|
if status.Account == nil {
|
|
statusAccount, err := p.state.DB.GetAccountByID(ctx, status.AccountID)
|
|
if err != nil {
|
|
return fmt.Errorf("federateStatus: error fetching status author account: %s", err)
|
|
}
|
|
status.Account = statusAccount
|
|
}
|
|
|
|
// Do nothing if this isn't our activity.
|
|
if !status.Account.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
asStatus, err := p.tc.StatusToAS(ctx, status)
|
|
if err != nil {
|
|
return fmt.Errorf("federateStatus: error converting status to as format: %s", err)
|
|
}
|
|
|
|
create, err := p.tc.WrapNoteInCreate(asStatus, false)
|
|
if err != nil {
|
|
return fmt.Errorf("federateStatus: error wrapping status in create: %s", err)
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(status.Account.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateStatus: error parsing outboxURI %s: %s", status.Account.OutboxURI, err)
|
|
}
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, create)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateStatusDelete(ctx context.Context, status *gtsmodel.Status) error {
|
|
if status.Account == nil {
|
|
statusAccount, err := p.state.DB.GetAccountByID(ctx, status.AccountID)
|
|
if err != nil {
|
|
return fmt.Errorf("federateStatusDelete: error fetching status author account: %s", err)
|
|
}
|
|
status.Account = statusAccount
|
|
}
|
|
|
|
// Do nothing if this isn't our activity.
|
|
if !status.Account.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
delete, err := p.tc.StatusToASDelete(ctx, status)
|
|
if err != nil {
|
|
return fmt.Errorf("federateStatusDelete: error creating Delete: %w", err)
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(status.Account.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateStatusDelete: error parsing outboxURI %s: %w", status.Account.OutboxURI, err)
|
|
}
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, delete)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateFollow(ctx context.Context, followRequest *gtsmodel.FollowRequest, originAccount *gtsmodel.Account, targetAccount *gtsmodel.Account) error {
|
|
// Do nothing if both accounts are local.
|
|
if originAccount.IsLocal() && targetAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
follow := p.tc.FollowRequestToFollow(ctx, followRequest)
|
|
|
|
asFollow, err := p.tc.FollowToAS(ctx, follow, originAccount, targetAccount)
|
|
if err != nil {
|
|
return fmt.Errorf("federateFollow: error converting follow to as format: %s", err)
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(originAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateFollow: error parsing outboxURI %s: %s", originAccount.OutboxURI, err)
|
|
}
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, asFollow)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateUnfollow(ctx context.Context, follow *gtsmodel.Follow, originAccount *gtsmodel.Account, targetAccount *gtsmodel.Account) error {
|
|
// Do nothing if both accounts are local.
|
|
if originAccount.IsLocal() && targetAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
// recreate the follow
|
|
asFollow, err := p.tc.FollowToAS(ctx, follow, originAccount, targetAccount)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnfollow: error converting follow to as format: %s", err)
|
|
}
|
|
|
|
targetAccountURI, err := url.Parse(targetAccount.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("error parsing uri %s: %s", targetAccount.URI, err)
|
|
}
|
|
|
|
// create an Undo and set the appropriate actor on it
|
|
undo := streams.NewActivityStreamsUndo()
|
|
undo.SetActivityStreamsActor(asFollow.GetActivityStreamsActor())
|
|
|
|
// Set the recreated follow as the 'object' property.
|
|
undoObject := streams.NewActivityStreamsObjectProperty()
|
|
undoObject.AppendActivityStreamsFollow(asFollow)
|
|
undo.SetActivityStreamsObject(undoObject)
|
|
|
|
// Set the To of the undo as the target of the recreated follow
|
|
undoTo := streams.NewActivityStreamsToProperty()
|
|
undoTo.AppendIRI(targetAccountURI)
|
|
undo.SetActivityStreamsTo(undoTo)
|
|
|
|
outboxIRI, err := url.Parse(originAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnfollow: error parsing outboxURI %s: %s", originAccount.OutboxURI, err)
|
|
}
|
|
|
|
// send off the Undo
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, undo)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateUnfave(ctx context.Context, fave *gtsmodel.StatusFave, originAccount *gtsmodel.Account, targetAccount *gtsmodel.Account) error {
|
|
// Do nothing if both accounts are local.
|
|
if originAccount.IsLocal() && targetAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
// create the AS fave
|
|
asFave, err := p.tc.FaveToAS(ctx, fave)
|
|
if err != nil {
|
|
return fmt.Errorf("federateFave: error converting fave to as format: %s", err)
|
|
}
|
|
|
|
targetAccountURI, err := url.Parse(targetAccount.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("error parsing uri %s: %s", targetAccount.URI, err)
|
|
}
|
|
|
|
// create an Undo and set the appropriate actor on it
|
|
undo := streams.NewActivityStreamsUndo()
|
|
undo.SetActivityStreamsActor(asFave.GetActivityStreamsActor())
|
|
|
|
// Set the fave as the 'object' property.
|
|
undoObject := streams.NewActivityStreamsObjectProperty()
|
|
undoObject.AppendActivityStreamsLike(asFave)
|
|
undo.SetActivityStreamsObject(undoObject)
|
|
|
|
// Set the To of the undo as the target of the fave
|
|
undoTo := streams.NewActivityStreamsToProperty()
|
|
undoTo.AppendIRI(targetAccountURI)
|
|
undo.SetActivityStreamsTo(undoTo)
|
|
|
|
outboxIRI, err := url.Parse(originAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateFave: error parsing outboxURI %s: %s", originAccount.OutboxURI, err)
|
|
}
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, undo)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateUnannounce(ctx context.Context, boost *gtsmodel.Status, originAccount *gtsmodel.Account, targetAccount *gtsmodel.Account) error {
|
|
// Do nothing if this isn't our activity.
|
|
if !originAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
asAnnounce, err := p.tc.BoostToAS(ctx, boost, originAccount, targetAccount)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnannounce: error converting status to announce: %s", err)
|
|
}
|
|
|
|
// create an Undo and set the appropriate actor on it
|
|
undo := streams.NewActivityStreamsUndo()
|
|
undo.SetActivityStreamsActor(asAnnounce.GetActivityStreamsActor())
|
|
|
|
// Set the boost as the 'object' property.
|
|
undoObject := streams.NewActivityStreamsObjectProperty()
|
|
undoObject.AppendActivityStreamsAnnounce(asAnnounce)
|
|
undo.SetActivityStreamsObject(undoObject)
|
|
|
|
// set the to
|
|
undo.SetActivityStreamsTo(asAnnounce.GetActivityStreamsTo())
|
|
|
|
// set the cc
|
|
undo.SetActivityStreamsCc(asAnnounce.GetActivityStreamsCc())
|
|
|
|
outboxIRI, err := url.Parse(originAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnannounce: error parsing outboxURI %s: %s", originAccount.OutboxURI, err)
|
|
}
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, undo)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateAcceptFollowRequest(ctx context.Context, follow *gtsmodel.Follow) error {
|
|
if follow.Account == nil {
|
|
a, err := p.state.DB.GetAccountByID(ctx, follow.AccountID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
follow.Account = a
|
|
}
|
|
originAccount := follow.Account
|
|
|
|
if follow.TargetAccount == nil {
|
|
a, err := p.state.DB.GetAccountByID(ctx, follow.TargetAccountID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
follow.TargetAccount = a
|
|
}
|
|
targetAccount := follow.TargetAccount
|
|
|
|
// Do nothing if target account *isn't* local,
|
|
// or both origin + target *are* local.
|
|
if targetAccount.IsRemote() || originAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
// recreate the AS follow
|
|
asFollow, err := p.tc.FollowToAS(ctx, follow, originAccount, targetAccount)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnfollow: error converting follow to as format: %s", err)
|
|
}
|
|
|
|
acceptingAccountURI, err := url.Parse(targetAccount.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("error parsing uri %s: %s", targetAccount.URI, err)
|
|
}
|
|
|
|
requestingAccountURI, err := url.Parse(originAccount.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("error parsing uri %s: %s", targetAccount.URI, err)
|
|
}
|
|
|
|
// create an Accept
|
|
accept := streams.NewActivityStreamsAccept()
|
|
|
|
// set the accepting actor on it
|
|
acceptActorProp := streams.NewActivityStreamsActorProperty()
|
|
acceptActorProp.AppendIRI(acceptingAccountURI)
|
|
accept.SetActivityStreamsActor(acceptActorProp)
|
|
|
|
// Set the recreated follow as the 'object' property.
|
|
acceptObject := streams.NewActivityStreamsObjectProperty()
|
|
acceptObject.AppendActivityStreamsFollow(asFollow)
|
|
accept.SetActivityStreamsObject(acceptObject)
|
|
|
|
// Set the To of the accept as the originator of the follow
|
|
acceptTo := streams.NewActivityStreamsToProperty()
|
|
acceptTo.AppendIRI(requestingAccountURI)
|
|
accept.SetActivityStreamsTo(acceptTo)
|
|
|
|
outboxIRI, err := url.Parse(targetAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAcceptFollowRequest: error parsing outboxURI %s: %s", originAccount.OutboxURI, err)
|
|
}
|
|
|
|
// send off the accept using the accepter's outbox
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, accept)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateRejectFollowRequest(ctx context.Context, followRequest *gtsmodel.FollowRequest) error {
|
|
if followRequest.Account == nil {
|
|
a, err := p.state.DB.GetAccountByID(ctx, followRequest.AccountID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
followRequest.Account = a
|
|
}
|
|
originAccount := followRequest.Account
|
|
|
|
if followRequest.TargetAccount == nil {
|
|
a, err := p.state.DB.GetAccountByID(ctx, followRequest.TargetAccountID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
followRequest.TargetAccount = a
|
|
}
|
|
targetAccount := followRequest.TargetAccount
|
|
|
|
// Do nothing if target account *isn't* local,
|
|
// or both origin + target *are* local.
|
|
if targetAccount.IsRemote() || originAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
// recreate the AS follow
|
|
follow := p.tc.FollowRequestToFollow(ctx, followRequest)
|
|
asFollow, err := p.tc.FollowToAS(ctx, follow, originAccount, targetAccount)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnfollow: error converting follow to as format: %s", err)
|
|
}
|
|
|
|
rejectingAccountURI, err := url.Parse(targetAccount.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("error parsing uri %s: %s", targetAccount.URI, err)
|
|
}
|
|
|
|
requestingAccountURI, err := url.Parse(originAccount.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("error parsing uri %s: %s", targetAccount.URI, err)
|
|
}
|
|
|
|
// create a Reject
|
|
reject := streams.NewActivityStreamsReject()
|
|
|
|
// set the rejecting actor on it
|
|
acceptActorProp := streams.NewActivityStreamsActorProperty()
|
|
acceptActorProp.AppendIRI(rejectingAccountURI)
|
|
reject.SetActivityStreamsActor(acceptActorProp)
|
|
|
|
// Set the recreated follow as the 'object' property.
|
|
acceptObject := streams.NewActivityStreamsObjectProperty()
|
|
acceptObject.AppendActivityStreamsFollow(asFollow)
|
|
reject.SetActivityStreamsObject(acceptObject)
|
|
|
|
// Set the To of the reject as the originator of the follow
|
|
acceptTo := streams.NewActivityStreamsToProperty()
|
|
acceptTo.AppendIRI(requestingAccountURI)
|
|
reject.SetActivityStreamsTo(acceptTo)
|
|
|
|
outboxIRI, err := url.Parse(targetAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateRejectFollowRequest: error parsing outboxURI %s: %s", originAccount.OutboxURI, err)
|
|
}
|
|
|
|
// send off the reject using the rejecting account's outbox
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, reject)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateFave(ctx context.Context, fave *gtsmodel.StatusFave, originAccount *gtsmodel.Account, targetAccount *gtsmodel.Account) error {
|
|
// Do nothing if both accounts are local.
|
|
if originAccount.IsLocal() && targetAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
// create the AS fave
|
|
asFave, err := p.tc.FaveToAS(ctx, fave)
|
|
if err != nil {
|
|
return fmt.Errorf("federateFave: error converting fave to as format: %s", err)
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(originAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateFave: error parsing outboxURI %s: %s", originAccount.OutboxURI, err)
|
|
}
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, asFave)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateAnnounce(ctx context.Context, boostWrapperStatus *gtsmodel.Status, boostingAccount *gtsmodel.Account, boostedAccount *gtsmodel.Account) error {
|
|
announce, err := p.tc.BoostToAS(ctx, boostWrapperStatus, boostingAccount, boostedAccount)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAnnounce: error converting status to announce: %s", err)
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(boostingAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAnnounce: error parsing outboxURI %s: %s", boostingAccount.OutboxURI, err)
|
|
}
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, announce)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateAccountUpdate(ctx context.Context, updatedAccount *gtsmodel.Account, originAccount *gtsmodel.Account) error {
|
|
person, err := p.tc.AccountToAS(ctx, updatedAccount)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAccountUpdate: error converting account to person: %s", err)
|
|
}
|
|
|
|
update, err := p.tc.WrapPersonInUpdate(person, originAccount)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAccountUpdate: error wrapping person in update: %s", err)
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(originAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateAnnounce: error parsing outboxURI %s: %s", originAccount.OutboxURI, err)
|
|
}
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, update)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateBlock(ctx context.Context, block *gtsmodel.Block) error {
|
|
if block.Account == nil {
|
|
blockAccount, err := p.state.DB.GetAccountByID(ctx, block.AccountID)
|
|
if err != nil {
|
|
return fmt.Errorf("federateBlock: error getting block account from database: %s", err)
|
|
}
|
|
block.Account = blockAccount
|
|
}
|
|
|
|
if block.TargetAccount == nil {
|
|
blockTargetAccount, err := p.state.DB.GetAccountByID(ctx, block.TargetAccountID)
|
|
if err != nil {
|
|
return fmt.Errorf("federateBlock: error getting block target account from database: %s", err)
|
|
}
|
|
block.TargetAccount = blockTargetAccount
|
|
}
|
|
|
|
// Do nothing if both accounts are local.
|
|
if block.Account.IsLocal() && block.TargetAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
asBlock, err := p.tc.BlockToAS(ctx, block)
|
|
if err != nil {
|
|
return fmt.Errorf("federateBlock: error converting block to AS format: %s", err)
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(block.Account.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateBlock: error parsing outboxURI %s: %s", block.Account.OutboxURI, err)
|
|
}
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, asBlock)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateUnblock(ctx context.Context, block *gtsmodel.Block) error {
|
|
if block.Account == nil {
|
|
blockAccount, err := p.state.DB.GetAccountByID(ctx, block.AccountID)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnblock: error getting block account from database: %s", err)
|
|
}
|
|
block.Account = blockAccount
|
|
}
|
|
|
|
if block.TargetAccount == nil {
|
|
blockTargetAccount, err := p.state.DB.GetAccountByID(ctx, block.TargetAccountID)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnblock: error getting block target account from database: %s", err)
|
|
}
|
|
block.TargetAccount = blockTargetAccount
|
|
}
|
|
|
|
// Do nothing if both accounts are local.
|
|
if block.Account.IsLocal() && block.TargetAccount.IsLocal() {
|
|
return nil
|
|
}
|
|
|
|
asBlock, err := p.tc.BlockToAS(ctx, block)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnblock: error converting block to AS format: %s", err)
|
|
}
|
|
|
|
targetAccountURI, err := url.Parse(block.TargetAccount.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnblock: error parsing uri %s: %s", block.TargetAccount.URI, err)
|
|
}
|
|
|
|
// create an Undo and set the appropriate actor on it
|
|
undo := streams.NewActivityStreamsUndo()
|
|
undo.SetActivityStreamsActor(asBlock.GetActivityStreamsActor())
|
|
|
|
// Set the block as the 'object' property.
|
|
undoObject := streams.NewActivityStreamsObjectProperty()
|
|
undoObject.AppendActivityStreamsBlock(asBlock)
|
|
undo.SetActivityStreamsObject(undoObject)
|
|
|
|
// Set the To of the undo as the target of the block
|
|
undoTo := streams.NewActivityStreamsToProperty()
|
|
undoTo.AppendIRI(targetAccountURI)
|
|
undo.SetActivityStreamsTo(undoTo)
|
|
|
|
outboxIRI, err := url.Parse(block.Account.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateUnblock: error parsing outboxURI %s: %s", block.Account.OutboxURI, err)
|
|
}
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, undo)
|
|
return err
|
|
}
|
|
|
|
func (p *Processor) federateReport(ctx context.Context, report *gtsmodel.Report) error {
|
|
if report.TargetAccount == nil {
|
|
reportTargetAccount, err := p.state.DB.GetAccountByID(ctx, report.TargetAccountID)
|
|
if err != nil {
|
|
return fmt.Errorf("federateReport: error getting report target account from database: %w", err)
|
|
}
|
|
report.TargetAccount = reportTargetAccount
|
|
}
|
|
|
|
if len(report.StatusIDs) > 0 && len(report.Statuses) == 0 {
|
|
statuses, err := p.state.DB.GetStatusesByIDs(ctx, report.StatusIDs)
|
|
if err != nil {
|
|
return fmt.Errorf("federateReport: error getting report statuses from database: %w", err)
|
|
}
|
|
report.Statuses = statuses
|
|
}
|
|
|
|
flag, err := p.tc.ReportToASFlag(ctx, report)
|
|
if err != nil {
|
|
return fmt.Errorf("federateReport: error converting report to AS flag: %w", err)
|
|
}
|
|
|
|
// add bto so that our federating actor knows where to
|
|
// send the Flag; it'll still use a shared inbox if possible
|
|
reportTargetURI, err := url.Parse(report.TargetAccount.URI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateReport: error parsing outboxURI %s: %w", report.TargetAccount.URI, err)
|
|
}
|
|
bTo := streams.NewActivityStreamsBtoProperty()
|
|
bTo.AppendIRI(reportTargetURI)
|
|
flag.SetActivityStreamsBto(bTo)
|
|
|
|
// deliver the flag using the outbox of the
|
|
// instance account to anonymize the report
|
|
instanceAccount, err := p.state.DB.GetInstanceAccount(ctx, "")
|
|
if err != nil {
|
|
return fmt.Errorf("federateReport: error getting instance account: %w", err)
|
|
}
|
|
|
|
outboxIRI, err := url.Parse(instanceAccount.OutboxURI)
|
|
if err != nil {
|
|
return fmt.Errorf("federateReport: error parsing outboxURI %s: %w", instanceAccount.OutboxURI, err)
|
|
}
|
|
|
|
_, err = p.federator.FederatingActor().Send(ctx, outboxIRI, flag)
|
|
return err
|
|
}
|