| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | /* | 
					
						
							|  |  |  |    GoToSocial | 
					
						
							| 
									
										
										
										
											2021-12-20 18:42:19 +01:00
										 |  |  |    Copyright (C) 2021-2022 GoToSocial Authors admin@gotosocial.org | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +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/>. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package federatingdb | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2021-08-27 20:14:20 +01:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-13 17:29:43 +01:00
										 |  |  | 	"github.com/superseriousbusiness/activity/pub" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/activity/streams/vocab" | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/db" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/typeutils" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // DB wraps the pub.Database interface with a couple of custom functions for GoToSocial. | 
					
						
							|  |  |  | type DB interface { | 
					
						
							|  |  |  | 	pub.Database | 
					
						
							|  |  |  | 	Undo(ctx context.Context, undo vocab.ActivityStreamsUndo) error | 
					
						
							|  |  |  | 	Accept(ctx context.Context, accept vocab.ActivityStreamsAccept) error | 
					
						
							| 
									
										
										
										
											2021-10-16 13:27:43 +02:00
										 |  |  | 	Reject(ctx context.Context, reject vocab.ActivityStreamsReject) error | 
					
						
							| 
									
										
										
										
											2021-05-28 19:57:04 +02:00
										 |  |  | 	Announce(ctx context.Context, announce vocab.ActivityStreamsAnnounce) error | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // FederatingDB uses the underlying DB interface to implement the go-fed pub.Database interface. | 
					
						
							|  |  |  | // It doesn't care what the underlying implementation of the DB interface is, as long as it works. | 
					
						
							|  |  |  | type federatingDB struct { | 
					
						
							| 
									
										
										
										
											2021-08-27 20:14:20 +01:00
										 |  |  | 	mutex         sync.Mutex | 
					
						
							|  |  |  | 	locks         map[string]*mutex | 
					
						
							|  |  |  | 	pool          sync.Pool | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 	db            db.DB | 
					
						
							|  |  |  | 	typeConverter typeutils.TypeConverter | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-11 05:37:33 -07:00
										 |  |  | // New returns a DB interface using the given database and config | 
					
						
							| 
									
										
										
										
											2021-12-07 13:31:39 +01:00
										 |  |  | func New(db db.DB) DB { | 
					
						
							| 
									
										
										
										
											2021-08-27 20:14:20 +01:00
										 |  |  | 	fdb := federatingDB{ | 
					
						
							|  |  |  | 		mutex:         sync.Mutex{}, | 
					
						
							|  |  |  | 		locks:         make(map[string]*mutex, 100), | 
					
						
							|  |  |  | 		pool:          sync.Pool{New: func() interface{} { return &mutex{} }}, | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 		db:            db, | 
					
						
							| 
									
										
										
										
											2021-12-07 13:31:39 +01:00
										 |  |  | 		typeConverter: typeutils.NewConverter(db), | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-08-27 20:14:20 +01:00
										 |  |  | 	go fdb.cleanupLocks() | 
					
						
							|  |  |  | 	return &fdb | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (db *federatingDB) cleanupLocks() { | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		// Sleep for a minute... | 
					
						
							|  |  |  | 		time.Sleep(time.Minute) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Delete unused locks from map | 
					
						
							|  |  |  | 		db.mutex.Lock() | 
					
						
							|  |  |  | 		for id, mu := range db.locks { | 
					
						
							|  |  |  | 			if !mu.inUse() { | 
					
						
							|  |  |  | 				delete(db.locks, id) | 
					
						
							|  |  |  | 				db.pool.Put(mu) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		db.mutex.Unlock() | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-05-27 16:06:24 +02:00
										 |  |  | } |