| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | /* | 
					
						
							|  |  |  |    GoToSocial | 
					
						
							|  |  |  |    Copyright (C) 2021 GoToSocial Authors admin@gotosocial.org | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    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-05-27 16:06:24 +02:00
										 |  |  | package federatingdb | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/go-fed/activity/streams/vocab" | 
					
						
							|  |  |  | 	"github.com/sirupsen/logrus" | 
					
						
							| 
									
										
										
										
											2021-08-31 15:59:12 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/ap" | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/db" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (f *federatingDB) Undo(ctx context.Context, undo vocab.ActivityStreamsUndo) error { | 
					
						
							|  |  |  | 	l := f.log.WithFields( | 
					
						
							|  |  |  | 		logrus.Fields{ | 
					
						
							| 
									
										
										
										
											2021-10-04 15:24:19 +02:00
										 |  |  | 			"func": "Undo", | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2021-10-04 15:24:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-04 16:00:35 +02:00
										 |  |  | 	if f.log.Level >= logrus.DebugLevel { | 
					
						
							| 
									
										
										
										
											2021-10-04 15:24:19 +02:00
										 |  |  | 		i, err := marshalItem(undo) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		l = l.WithField("undo", i) | 
					
						
							|  |  |  | 		l.Debug("entering Undo") | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-10-04 15:24:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	targetAcct, fromFederatorChan, err := extractFromCtx(ctx) | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-10-04 15:24:19 +02:00
										 |  |  | 	if targetAcct == nil || fromFederatorChan == nil { | 
					
						
							|  |  |  | 		// If the target account or federator channel wasn't set on the context, that means this request didn't pass | 
					
						
							|  |  |  | 		// through the API, but came from inside GtS as the result of another activity on this instance. That being so, | 
					
						
							| 
									
										
										
										
											2021-07-27 10:45:22 +02:00
										 |  |  | 		// we can safely just ignore this activity, since we know we've already processed it elsewhere. | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	undoObject := undo.GetActivityStreamsObject() | 
					
						
							|  |  |  | 	if undoObject == nil { | 
					
						
							|  |  |  | 		return errors.New("UNDO: no object set on vocab.ActivityStreamsUndo") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for iter := undoObject.Begin(); iter != undoObject.End(); iter = iter.Next() { | 
					
						
							| 
									
										
										
										
											2021-06-13 18:42:28 +02:00
										 |  |  | 		if iter.GetType() == nil { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 		switch iter.GetType().GetTypeName() { | 
					
						
							| 
									
										
										
										
											2021-09-03 10:30:40 +02:00
										 |  |  | 		case ap.ActivityFollow: | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 			// UNDO FOLLOW | 
					
						
							|  |  |  | 			ASFollow, ok := iter.GetType().(vocab.ActivityStreamsFollow) | 
					
						
							|  |  |  | 			if !ok { | 
					
						
							|  |  |  | 				return errors.New("UNDO: couldn't parse follow into vocab.ActivityStreamsFollow") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// make sure the actor owns the follow | 
					
						
							|  |  |  | 			if !sameActor(undo.GetActivityStreamsActor(), ASFollow.GetActivityStreamsActor()) { | 
					
						
							|  |  |  | 				return errors.New("UNDO: follow actor and activity actor not the same") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// convert the follow to something we can understand | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 			gtsFollow, err := f.typeConverter.ASFollowToFollow(ctx, ASFollow) | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return fmt.Errorf("UNDO: error converting asfollow to gtsfollow: %s", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// make sure the addressee of the original follow is the same as whatever inbox this landed in | 
					
						
							|  |  |  | 			if gtsFollow.TargetAccountID != targetAcct.ID { | 
					
						
							|  |  |  | 				return errors.New("UNDO: follow object account and inbox account were not the same") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// delete any existing FOLLOW | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 			if err := f.db.DeleteWhere(ctx, []db.Where{{Key: "uri", Value: gtsFollow.URI}}, >smodel.Follow{}); err != nil { | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 				return fmt.Errorf("UNDO: db error removing follow: %s", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// delete any existing FOLLOW REQUEST | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 			if err := f.db.DeleteWhere(ctx, []db.Where{{Key: "uri", Value: gtsFollow.URI}}, >smodel.FollowRequest{}); err != nil { | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 				return fmt.Errorf("UNDO: db error removing follow request: %s", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			l.Debug("follow undone") | 
					
						
							|  |  |  | 			return nil | 
					
						
							| 
									
										
										
										
											2021-09-03 10:30:40 +02:00
										 |  |  | 		case ap.ActivityLike: | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 			// UNDO LIKE | 
					
						
							| 
									
										
										
										
											2021-09-03 10:30:40 +02:00
										 |  |  | 		case ap.ActivityAnnounce: | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 			// UNDO BOOST/REBLOG/ANNOUNCE | 
					
						
							| 
									
										
										
										
											2021-09-03 10:30:40 +02:00
										 |  |  | 		case ap.ActivityBlock: | 
					
						
							| 
									
										
										
										
											2021-07-11 16:22:21 +02:00
										 |  |  | 			// UNDO BLOCK | 
					
						
							|  |  |  | 			ASBlock, ok := iter.GetType().(vocab.ActivityStreamsBlock) | 
					
						
							|  |  |  | 			if !ok { | 
					
						
							|  |  |  | 				return errors.New("UNDO: couldn't parse block into vocab.ActivityStreamsBlock") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// make sure the actor owns the follow | 
					
						
							|  |  |  | 			if !sameActor(undo.GetActivityStreamsActor(), ASBlock.GetActivityStreamsActor()) { | 
					
						
							|  |  |  | 				return errors.New("UNDO: block actor and activity actor not the same") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// convert the block to something we can understand | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 			gtsBlock, err := f.typeConverter.ASBlockToBlock(ctx, ASBlock) | 
					
						
							| 
									
										
										
										
											2021-07-11 16:22:21 +02:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return fmt.Errorf("UNDO: error converting asblock to gtsblock: %s", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// make sure the addressee of the original block is the same as whatever inbox this landed in | 
					
						
							|  |  |  | 			if gtsBlock.TargetAccountID != targetAcct.ID { | 
					
						
							|  |  |  | 				return errors.New("UNDO: block object account and inbox account were not the same") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// delete any existing BLOCK | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 			if err := f.db.DeleteWhere(ctx, []db.Where{{Key: "uri", Value: gtsBlock.URI}}, >smodel.Block{}); err != nil { | 
					
						
							| 
									
										
										
										
											2021-07-11 16:22:21 +02:00
										 |  |  | 				return fmt.Errorf("UNDO: db error removing block: %s", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			l.Debug("block undone") | 
					
						
							|  |  |  | 			return nil | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } |