mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-10-31 01:12:24 -05:00 
			
		
		
		
	[chore] improved startup / shutdown (#2925)
* improved server shutdown with more precise shutdown of modules + deferring of ALL of it * make the same changes to the testrig server * use testrig specific func * update variable name to fix nilptr * fix removal of setting db on state
This commit is contained in:
		
					parent
					
						
							
								975e92b7f1
							
						
					
				
			
			
				commit
				
					
						32e570abfd
					
				
			
		
					 4 changed files with 194 additions and 159 deletions
				
			
		|  | @ -48,7 +48,6 @@ import ( | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/email" | 	"github.com/superseriousbusiness/gotosocial/internal/email" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/federation" | 	"github.com/superseriousbusiness/gotosocial/internal/federation" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/federation/federatingdb" | 	"github.com/superseriousbusiness/gotosocial/internal/federation/federatingdb" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/gotosocial" |  | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/httpclient" | 	"github.com/superseriousbusiness/gotosocial/internal/httpclient" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/log" | 	"github.com/superseriousbusiness/gotosocial/internal/log" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/media" | 	"github.com/superseriousbusiness/gotosocial/internal/media" | ||||||
|  | @ -69,59 +68,107 @@ import ( | ||||||
| // Start creates and starts a gotosocial server | // Start creates and starts a gotosocial server | ||||||
| var Start action.GTSAction = func(ctx context.Context) error { | var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	if _, err := maxprocs.Set(maxprocs.Logger(nil)); err != nil { | 	if _, err := maxprocs.Set(maxprocs.Logger(nil)); err != nil { | ||||||
| 		log.Infof(ctx, "could not set CPU limits from cgroup: %s", err) | 		log.Warnf(ctx, "could not set CPU limits from cgroup: %s", err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	var state state.State | 	var ( | ||||||
|  | 		// Define necessary core variables | ||||||
|  | 		// before anything so we can prepare | ||||||
|  | 		// defer function for safe shutdown | ||||||
|  | 		// depending on what services were | ||||||
|  | 		// managed to be started. | ||||||
| 
 | 
 | ||||||
| 	// Initialize caches | 		state state.State | ||||||
| 	state.Caches.Init() | 		route *router.Router | ||||||
| 	state.Caches.Start() | 	) | ||||||
| 	defer state.Caches.Stop() |  | ||||||
| 
 | 
 | ||||||
| 	// Initialize Tracing | 	defer func() { | ||||||
|  | 		// Stop caches with | ||||||
|  | 		// background tasks. | ||||||
|  | 		state.Caches.Stop() | ||||||
|  | 
 | ||||||
|  | 		if route != nil { | ||||||
|  | 			// We reached a point where the API router | ||||||
|  | 			// was created + setup. Ensure it gets stopped | ||||||
|  | 			// first to stop processing new information. | ||||||
|  | 			if err := route.Stop(); err != nil { | ||||||
|  | 				log.Errorf(ctx, "error stopping router: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// Stop any currently running | ||||||
|  | 		// worker processes / scheduled | ||||||
|  | 		// tasks from being executed. | ||||||
|  | 		state.Workers.Stop() | ||||||
|  | 
 | ||||||
|  | 		if state.Timelines.Home != nil { | ||||||
|  | 			// Home timeline mgr was setup, ensure it gets stopped. | ||||||
|  | 			if err := state.Timelines.Home.Stop(); err != nil { | ||||||
|  | 				log.Errorf(ctx, "error stopping home timeline: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if state.Timelines.List != nil { | ||||||
|  | 			// List timeline mgr was setup, ensure it gets stopped. | ||||||
|  | 			if err := state.Timelines.List.Stop(); err != nil { | ||||||
|  | 				log.Errorf(ctx, "error stopping list timeline: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if state.DB != nil { | ||||||
|  | 			// Lastly, if database service was started, | ||||||
|  | 			// ensure it gets closed now all else stopped. | ||||||
|  | 			if err := state.DB.Close(); err != nil { | ||||||
|  | 				log.Errorf(ctx, "error stopping database: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// Finally reached end of shutdown. | ||||||
|  | 		log.Info(ctx, "done! exiting...") | ||||||
|  | 	}() | ||||||
|  | 
 | ||||||
|  | 	// Initialize tracing (noop if not enabled). | ||||||
| 	if err := tracing.Initialize(); err != nil { | 	if err := tracing.Initialize(); err != nil { | ||||||
| 		return fmt.Errorf("error initializing tracing: %w", err) | 		return fmt.Errorf("error initializing tracing: %w", err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Open connection to the database | 	// Initialize caches | ||||||
|  | 	state.Caches.Init() | ||||||
|  | 	state.Caches.Start() | ||||||
|  | 
 | ||||||
|  | 	// Open connection to the database now caches started. | ||||||
| 	dbService, err := bundb.NewBunDBService(ctx, &state) | 	dbService, err := bundb.NewBunDBService(ctx, &state) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("error creating dbservice: %s", err) | 		return fmt.Errorf("error creating dbservice: %s", err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Set the state DB connection | 	// Set DB on state. | ||||||
| 	state.DB = dbService | 	state.DB = dbService | ||||||
| 
 | 
 | ||||||
|  | 	// Ensure necessary database instance prerequisites exist. | ||||||
| 	if err := dbService.CreateInstanceAccount(ctx); err != nil { | 	if err := dbService.CreateInstanceAccount(ctx); err != nil { | ||||||
| 		return fmt.Errorf("error creating instance account: %s", err) | 		return fmt.Errorf("error creating instance account: %s", err) | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	if err := dbService.CreateInstanceInstance(ctx); err != nil { | 	if err := dbService.CreateInstanceInstance(ctx); err != nil { | ||||||
| 		return fmt.Errorf("error creating instance instance: %s", err) | 		return fmt.Errorf("error creating instance instance: %s", err) | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	if err := dbService.CreateInstanceApplication(ctx); err != nil { | 	if err := dbService.CreateInstanceApplication(ctx); err != nil { | ||||||
| 		return fmt.Errorf("error creating instance application: %s", err) | 		return fmt.Errorf("error creating instance application: %s", err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Get the instance account | 	// Get the instance account (we'll need this later). | ||||||
| 	// (we'll need this later). |  | ||||||
| 	instanceAccount, err := dbService.GetInstanceAccount(ctx, "") | 	instanceAccount, err := dbService.GetInstanceAccount(ctx, "") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("error retrieving instance account: %w", err) | 		return fmt.Errorf("error retrieving instance account: %w", err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Open the storage backend | 	// Open the storage backend according to config. | ||||||
| 	storage, err := gtsstorage.AutoConfig() | 	state.Storage, err = gtsstorage.AutoConfig() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("error creating storage backend: %w", err) | 		return fmt.Errorf("error opening storage backend: %w", err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Set the state storage driver | 	// Prepare wrapped httpclient with config. | ||||||
| 	state.Storage = storage |  | ||||||
| 
 |  | ||||||
| 	// Build HTTP client |  | ||||||
| 	client := httpclient.New(httpclient.Config{ | 	client := httpclient.New(httpclient.Config{ | ||||||
| 		AllowRanges:           config.MustParseIPPrefixes(config.GetHTTPClientAllowIPs()), | 		AllowRanges:           config.MustParseIPPrefixes(config.GetHTTPClientAllowIPs()), | ||||||
| 		BlockRanges:           config.MustParseIPPrefixes(config.GetHTTPClientBlockIPs()), | 		BlockRanges:           config.MustParseIPPrefixes(config.GetHTTPClientBlockIPs()), | ||||||
|  | @ -156,7 +203,7 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Initialize timelines. | 	// Initialize both home / list timelines. | ||||||
| 	state.Timelines.Home = timeline.NewManager( | 	state.Timelines.Home = timeline.NewManager( | ||||||
| 		tlprocessor.HomeTimelineGrab(&state), | 		tlprocessor.HomeTimelineGrab(&state), | ||||||
| 		tlprocessor.HomeTimelineFilter(&state, visFilter), | 		tlprocessor.HomeTimelineFilter(&state, visFilter), | ||||||
|  | @ -166,7 +213,6 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	if err := state.Timelines.Home.Start(); err != nil { | 	if err := state.Timelines.Home.Start(); err != nil { | ||||||
| 		return fmt.Errorf("error starting home timeline: %s", err) | 		return fmt.Errorf("error starting home timeline: %s", err) | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	state.Timelines.List = timeline.NewManager( | 	state.Timelines.List = timeline.NewManager( | ||||||
| 		tlprocessor.ListTimelineGrab(&state), | 		tlprocessor.ListTimelineGrab(&state), | ||||||
| 		tlprocessor.ListTimelineFilter(&state, visFilter), | 		tlprocessor.ListTimelineFilter(&state, visFilter), | ||||||
|  | @ -196,6 +242,11 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	// Create background cleaner. | 	// Create background cleaner. | ||||||
| 	cleaner := cleaner.New(&state) | 	cleaner := cleaner.New(&state) | ||||||
| 
 | 
 | ||||||
|  | 	// Now schedule background cleaning tasks. | ||||||
|  | 	if err := cleaner.ScheduleJobs(); err != nil { | ||||||
|  | 		return fmt.Errorf("error scheduling cleaner jobs: %w", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	// Create the processor using all the | 	// Create the processor using all the | ||||||
| 	// other services we've created so far. | 	// other services we've created so far. | ||||||
| 	processor := processing.NewProcessor( | 	processor := processing.NewProcessor( | ||||||
|  | @ -208,18 +259,17 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 		emailSender, | 		emailSender, | ||||||
| 	) | 	) | ||||||
| 
 | 
 | ||||||
| 	// Initialize the specialized workers. | 	// Initialize the specialized workers pools. | ||||||
| 	state.Workers.Client.Init(messages.ClientMsgIndices()) | 	state.Workers.Client.Init(messages.ClientMsgIndices()) | ||||||
| 	state.Workers.Federator.Init(messages.FederatorMsgIndices()) | 	state.Workers.Federator.Init(messages.FederatorMsgIndices()) | ||||||
| 	state.Workers.Delivery.Init(client) | 	state.Workers.Delivery.Init(client) | ||||||
| 	state.Workers.Client.Process = processor.Workers().ProcessFromClientAPI | 	state.Workers.Client.Process = processor.Workers().ProcessFromClientAPI | ||||||
| 	state.Workers.Federator.Process = processor.Workers().ProcessFromFediAPI | 	state.Workers.Federator.Process = processor.Workers().ProcessFromFediAPI | ||||||
| 
 | 
 | ||||||
| 	// Initialize workers. | 	// Now start workers! | ||||||
| 	state.Workers.Start() | 	state.Workers.Start() | ||||||
| 	defer state.Workers.Stop() |  | ||||||
| 
 | 
 | ||||||
| 	// Schedule tasks for all existing poll expiries. | 	// Schedule notif tasks for all existing poll expiries. | ||||||
| 	if err := processor.Polls().ScheduleAll(ctx); err != nil { | 	if err := processor.Polls().ScheduleAll(ctx); err != nil { | ||||||
| 		return fmt.Errorf("error scheduling poll expiries: %w", err) | 		return fmt.Errorf("error scheduling poll expiries: %w", err) | ||||||
| 	} | 	} | ||||||
|  | @ -233,7 +283,7 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 		HTTP router initialization | 		HTTP router initialization | ||||||
| 	*/ | 	*/ | ||||||
| 
 | 
 | ||||||
| 	router, err := router.New(ctx) | 	route, err = router.New(ctx) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("error creating router: %s", err) | 		return fmt.Errorf("error creating router: %s", err) | ||||||
| 	} | 	} | ||||||
|  | @ -288,10 +338,10 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	middlewares = append(middlewares, middleware.ContentSecurityPolicy(cspExtraURIs...)) | 	middlewares = append(middlewares, middleware.ContentSecurityPolicy(cspExtraURIs...)) | ||||||
| 
 | 
 | ||||||
| 	// attach global middlewares which are used for every request | 	// attach global middlewares which are used for every request | ||||||
| 	router.AttachGlobalMiddleware(middlewares...) | 	route.AttachGlobalMiddleware(middlewares...) | ||||||
| 
 | 
 | ||||||
| 	// attach global no route / 404 handler to the router | 	// attach global no route / 404 handler to the router | ||||||
| 	router.AttachNoRouteHandler(func(c *gin.Context) { | 	route.AttachNoRouteHandler(func(c *gin.Context) { | ||||||
| 		apiutil.ErrorHandler(c, gtserror.NewErrorNotFound(errors.New(http.StatusText(http.StatusNotFound))), processor.InstanceGetV1) | 		apiutil.ErrorHandler(c, gtserror.NewErrorNotFound(errors.New(http.StatusText(http.StatusNotFound))), processor.InstanceGetV1) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
|  | @ -347,22 +397,21 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 
 | 
 | ||||||
| 	// these should be routed in order; | 	// these should be routed in order; | ||||||
| 	// apply throttling *after* rate limiting | 	// apply throttling *after* rate limiting | ||||||
| 	authModule.Route(router, clLimit, clThrottle, gzip) | 	authModule.Route(route, clLimit, clThrottle, gzip) | ||||||
| 	clientModule.Route(router, clLimit, clThrottle, gzip) | 	clientModule.Route(route, clLimit, clThrottle, gzip) | ||||||
| 	metricsModule.Route(router, clLimit, clThrottle, gzip) | 	metricsModule.Route(route, clLimit, clThrottle, gzip) | ||||||
| 	healthModule.Route(router, clLimit, clThrottle) | 	healthModule.Route(route, clLimit, clThrottle) | ||||||
| 	fileserverModule.Route(router, fsMainLimit, fsThrottle) | 	fileserverModule.Route(route, fsMainLimit, fsThrottle) | ||||||
| 	fileserverModule.RouteEmojis(router, instanceAccount.ID, fsEmojiLimit, fsThrottle) | 	fileserverModule.RouteEmojis(route, instanceAccount.ID, fsEmojiLimit, fsThrottle) | ||||||
| 	wellKnownModule.Route(router, gzip, s2sLimit, s2sThrottle) | 	wellKnownModule.Route(route, gzip, s2sLimit, s2sThrottle) | ||||||
| 	nodeInfoModule.Route(router, s2sLimit, s2sThrottle, gzip) | 	nodeInfoModule.Route(route, s2sLimit, s2sThrottle, gzip) | ||||||
| 	activityPubModule.Route(router, s2sLimit, s2sThrottle, gzip) | 	activityPubModule.Route(route, s2sLimit, s2sThrottle, gzip) | ||||||
| 	activityPubModule.RoutePublicKey(router, s2sLimit, pkThrottle, gzip) | 	activityPubModule.RoutePublicKey(route, s2sLimit, pkThrottle, gzip) | ||||||
| 	webModule.Route(router, fsMainLimit, fsThrottle, gzip) | 	webModule.Route(route, fsMainLimit, fsThrottle, gzip) | ||||||
| 
 | 
 | ||||||
| 	// Start the GoToSocial server. | 	// Finally start the main http server! | ||||||
| 	server := gotosocial.NewServer(dbService, router, cleaner) | 	if err := route.Start(); err != nil { | ||||||
| 	if err := server.Start(ctx); err != nil { | 		return fmt.Errorf("error starting router: %w", err) | ||||||
| 		return fmt.Errorf("error starting gotosocial service: %s", err) |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// catch shutdown signals from the operating system | 	// catch shutdown signals from the operating system | ||||||
|  | @ -371,11 +420,5 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	sig := <-sigs // block until signal received | 	sig := <-sigs // block until signal received | ||||||
| 	log.Infof(ctx, "received signal %s, shutting down", sig) | 	log.Infof(ctx, "received signal %s, shutting down", sig) | ||||||
| 
 | 
 | ||||||
| 	// close down all running services in order |  | ||||||
| 	if err := server.Stop(ctx); err != nil { |  | ||||||
| 		return fmt.Errorf("error closing gotosocial service: %s", err) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	log.Info(ctx, "done! exiting...") |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -35,7 +35,6 @@ import ( | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/cleaner" | 	"github.com/superseriousbusiness/gotosocial/internal/cleaner" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/config" | 	"github.com/superseriousbusiness/gotosocial/internal/config" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/filter/visibility" | 	"github.com/superseriousbusiness/gotosocial/internal/filter/visibility" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/gotosocial" |  | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/gtserror" | 	"github.com/superseriousbusiness/gotosocial/internal/gtserror" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/language" | 	"github.com/superseriousbusiness/gotosocial/internal/language" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/log" | 	"github.com/superseriousbusiness/gotosocial/internal/log" | ||||||
|  | @ -43,6 +42,7 @@ import ( | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/middleware" | 	"github.com/superseriousbusiness/gotosocial/internal/middleware" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/oidc" | 	"github.com/superseriousbusiness/gotosocial/internal/oidc" | ||||||
| 	tlprocessor "github.com/superseriousbusiness/gotosocial/internal/processing/timeline" | 	tlprocessor "github.com/superseriousbusiness/gotosocial/internal/processing/timeline" | ||||||
|  | 	"github.com/superseriousbusiness/gotosocial/internal/router" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/state" | 	"github.com/superseriousbusiness/gotosocial/internal/state" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/storage" | 	"github.com/superseriousbusiness/gotosocial/internal/storage" | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/timeline" | 	"github.com/superseriousbusiness/gotosocial/internal/timeline" | ||||||
|  | @ -54,11 +54,71 @@ import ( | ||||||
| 
 | 
 | ||||||
| // Start creates and starts a gotosocial testrig server | // Start creates and starts a gotosocial testrig server | ||||||
| var Start action.GTSAction = func(ctx context.Context) error { | var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	var state state.State |  | ||||||
| 
 |  | ||||||
| 	testrig.InitTestConfig() | 	testrig.InitTestConfig() | ||||||
| 	testrig.InitTestLog() | 	testrig.InitTestLog() | ||||||
| 
 | 
 | ||||||
|  | 	var ( | ||||||
|  | 		// Define necessary core variables | ||||||
|  | 		// before anything so we can prepare | ||||||
|  | 		// defer function for safe shutdown | ||||||
|  | 		// depending on what services were | ||||||
|  | 		// managed to be started. | ||||||
|  | 
 | ||||||
|  | 		state state.State | ||||||
|  | 		route *router.Router | ||||||
|  | 	) | ||||||
|  | 
 | ||||||
|  | 	defer func() { | ||||||
|  | 		// Stop caches with | ||||||
|  | 		// background tasks. | ||||||
|  | 		state.Caches.Stop() | ||||||
|  | 
 | ||||||
|  | 		if route != nil { | ||||||
|  | 			// We reached a point where the API router | ||||||
|  | 			// was created + setup. Ensure it gets stopped | ||||||
|  | 			// first to stop processing new information. | ||||||
|  | 			if err := route.Stop(); err != nil { | ||||||
|  | 				log.Errorf(ctx, "error stopping router: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// Stop any currently running | ||||||
|  | 		// worker processes / scheduled | ||||||
|  | 		// tasks from being executed. | ||||||
|  | 		testrig.StopWorkers(&state) | ||||||
|  | 
 | ||||||
|  | 		if state.Timelines.Home != nil { | ||||||
|  | 			// Home timeline mgr was setup, ensure it gets stopped. | ||||||
|  | 			if err := state.Timelines.Home.Stop(); err != nil { | ||||||
|  | 				log.Errorf(ctx, "error stopping home timeline: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if state.Timelines.List != nil { | ||||||
|  | 			// List timeline mgr was setup, ensure it gets stopped. | ||||||
|  | 			if err := state.Timelines.List.Stop(); err != nil { | ||||||
|  | 				log.Errorf(ctx, "error stopping list timeline: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if state.Storage != nil { | ||||||
|  | 			// If storage was created, ensure torn down. | ||||||
|  | 			testrig.StandardStorageTeardown(state.Storage) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if state.DB != nil { | ||||||
|  | 			// Lastly, if database service was started, | ||||||
|  | 			// ensure it gets closed now all else stopped. | ||||||
|  | 			testrig.StandardDBTeardown(state.DB) | ||||||
|  | 			if err := state.DB.Close(); err != nil { | ||||||
|  | 				log.Errorf(ctx, "error stopping database: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// Finally reached end of shutdown. | ||||||
|  | 		log.Info(ctx, "done! exiting...") | ||||||
|  | 	}() | ||||||
|  | 
 | ||||||
| 	parsedLangs, err := language.InitLangs(config.GetInstanceLanguages().TagStrs()) | 	parsedLangs, err := language.InitLangs(config.GetInstanceLanguages().TagStrs()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("error initializing languages: %w", err) | 		return fmt.Errorf("error initializing languages: %w", err) | ||||||
|  | @ -75,12 +135,10 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	// New test db inits caches so we don't need to do | 	// New test db inits caches so we don't need to do | ||||||
| 	// that twice, we can just start the initialized caches. | 	// that twice, we can just start the initialized caches. | ||||||
| 	state.Caches.Start() | 	state.Caches.Start() | ||||||
| 	defer state.Caches.Stop() |  | ||||||
| 
 | 
 | ||||||
| 	testrig.StandardDBSetup(state.DB, nil) | 	testrig.StandardDBSetup(state.DB, nil) | ||||||
| 
 | 
 | ||||||
| 	// Get the instance account | 	// Get the instance account (we'll need this later). | ||||||
| 	// (we'll need this later). |  | ||||||
| 	instanceAccount, err := state.DB.GetInstanceAccount(ctx, "") | 	instanceAccount, err := state.DB.GetInstanceAccount(ctx, "") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("error retrieving instance account: %w", err) | 		return fmt.Errorf("error retrieving instance account: %w", err) | ||||||
|  | @ -119,7 +177,7 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	typeConverter := typeutils.NewConverter(&state) | 	typeConverter := typeutils.NewConverter(&state) | ||||||
| 	filter := visibility.NewFilter(&state) | 	filter := visibility.NewFilter(&state) | ||||||
| 
 | 
 | ||||||
| 	// Initialize timelines. | 	// Initialize both home / list timelines. | ||||||
| 	state.Timelines.Home = timeline.NewManager( | 	state.Timelines.Home = timeline.NewManager( | ||||||
| 		tlprocessor.HomeTimelineGrab(&state), | 		tlprocessor.HomeTimelineGrab(&state), | ||||||
| 		tlprocessor.HomeTimelineFilter(&state, filter), | 		tlprocessor.HomeTimelineFilter(&state, filter), | ||||||
|  | @ -129,7 +187,6 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	if err := state.Timelines.Home.Start(); err != nil { | 	if err := state.Timelines.Home.Start(); err != nil { | ||||||
| 		return fmt.Errorf("error starting home timeline: %s", err) | 		return fmt.Errorf("error starting home timeline: %s", err) | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	state.Timelines.List = timeline.NewManager( | 	state.Timelines.List = timeline.NewManager( | ||||||
| 		tlprocessor.ListTimelineGrab(&state), | 		tlprocessor.ListTimelineGrab(&state), | ||||||
| 		tlprocessor.ListTimelineFilter(&state, filter), | 		tlprocessor.ListTimelineFilter(&state, filter), | ||||||
|  | @ -151,7 +208,7 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 		HTTP router initialization | 		HTTP router initialization | ||||||
| 	*/ | 	*/ | ||||||
| 
 | 
 | ||||||
| 	router := testrig.NewTestRouter(state.DB) | 	route = testrig.NewTestRouter(state.DB) | ||||||
| 	middlewares := []gin.HandlerFunc{ | 	middlewares := []gin.HandlerFunc{ | ||||||
| 		middleware.AddRequestID(config.GetRequestIDHeader()), // requestID middleware must run before tracing | 		middleware.AddRequestID(config.GetRequestIDHeader()), // requestID middleware must run before tracing | ||||||
| 	} | 	} | ||||||
|  | @ -195,10 +252,10 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	middlewares = append(middlewares, middleware.ContentSecurityPolicy(cspExtraURIs...)) | 	middlewares = append(middlewares, middleware.ContentSecurityPolicy(cspExtraURIs...)) | ||||||
| 
 | 
 | ||||||
| 	// attach global middlewares which are used for every request | 	// attach global middlewares which are used for every request | ||||||
| 	router.AttachGlobalMiddleware(middlewares...) | 	route.AttachGlobalMiddleware(middlewares...) | ||||||
| 
 | 
 | ||||||
| 	// attach global no route / 404 handler to the router | 	// attach global no route / 404 handler to the router | ||||||
| 	router.AttachNoRouteHandler(func(c *gin.Context) { | 	route.AttachNoRouteHandler(func(c *gin.Context) { | ||||||
| 		apiutil.ErrorHandler(c, gtserror.NewErrorNotFound(errors.New(http.StatusText(http.StatusNotFound))), processor.InstanceGetV1) | 		apiutil.ErrorHandler(c, gtserror.NewErrorNotFound(errors.New(http.StatusText(http.StatusNotFound))), processor.InstanceGetV1) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
|  | @ -234,23 +291,29 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	) | 	) | ||||||
| 
 | 
 | ||||||
| 	// these should be routed in order | 	// these should be routed in order | ||||||
| 	authModule.Route(router) | 	authModule.Route(route) | ||||||
| 	clientModule.Route(router) | 	clientModule.Route(route) | ||||||
| 	metricsModule.Route(router) | 	metricsModule.Route(route) | ||||||
| 	healthModule.Route(router) | 	healthModule.Route(route) | ||||||
| 	fileserverModule.Route(router) | 	fileserverModule.Route(route) | ||||||
| 	fileserverModule.RouteEmojis(router, instanceAccount.ID) | 	fileserverModule.RouteEmojis(route, instanceAccount.ID) | ||||||
| 	wellKnownModule.Route(router) | 	wellKnownModule.Route(route) | ||||||
| 	nodeInfoModule.Route(router) | 	nodeInfoModule.Route(route) | ||||||
| 	activityPubModule.Route(router) | 	activityPubModule.Route(route) | ||||||
| 	activityPubModule.RoutePublicKey(router) | 	activityPubModule.RoutePublicKey(route) | ||||||
| 	webModule.Route(router) | 	webModule.Route(route) | ||||||
| 
 | 
 | ||||||
|  | 	// Create background cleaner. | ||||||
| 	cleaner := cleaner.New(&state) | 	cleaner := cleaner.New(&state) | ||||||
| 
 | 
 | ||||||
| 	gts := gotosocial.NewServer(state.DB, router, cleaner) | 	// Now schedule background cleaning tasks. | ||||||
| 	if err := gts.Start(ctx); err != nil { | 	if err := cleaner.ScheduleJobs(); err != nil { | ||||||
| 		return fmt.Errorf("error starting gotosocial service: %s", err) | 		return fmt.Errorf("error scheduling cleaner jobs: %w", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Finally start the main http server! | ||||||
|  | 	if err := route.Start(); err != nil { | ||||||
|  | 		return fmt.Errorf("error starting router: %w", err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// catch shutdown signals from the operating system | 	// catch shutdown signals from the operating system | ||||||
|  | @ -259,14 +322,5 @@ var Start action.GTSAction = func(ctx context.Context) error { | ||||||
| 	sig := <-sigs | 	sig := <-sigs | ||||||
| 	log.Infof(ctx, "received signal %s, shutting down", sig) | 	log.Infof(ctx, "received signal %s, shutting down", sig) | ||||||
| 
 | 
 | ||||||
| 	testrig.StandardDBTeardown(state.DB) |  | ||||||
| 	testrig.StandardStorageTeardown(state.Storage) |  | ||||||
| 
 |  | ||||||
| 	// close down all running services in order |  | ||||||
| 	if err := gts.Stop(ctx); err != nil { |  | ||||||
| 		return fmt.Errorf("error closing gotosocial service: %s", err) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	log.Info(ctx, "done! exiting...") |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,66 +0,0 @@ | ||||||
| // 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 gotosocial |  | ||||||
| 
 |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 
 |  | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/cleaner" |  | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/db" |  | ||||||
| 	"github.com/superseriousbusiness/gotosocial/internal/router" |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| // Server represents a long-running |  | ||||||
| // GoToSocial server instance. |  | ||||||
| type Server struct { |  | ||||||
| 	db        db.DB |  | ||||||
| 	apiRouter *router.Router |  | ||||||
| 	cleaner   *cleaner.Cleaner |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // NewServer returns a new |  | ||||||
| // GoToSocial server instance. |  | ||||||
| func NewServer( |  | ||||||
| 	db db.DB, |  | ||||||
| 	apiRouter *router.Router, |  | ||||||
| 	cleaner *cleaner.Cleaner, |  | ||||||
| ) *Server { |  | ||||||
| 	return &Server{ |  | ||||||
| 		db:        db, |  | ||||||
| 		apiRouter: apiRouter, |  | ||||||
| 		cleaner:   cleaner, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // Start starts up the GoToSocial server by starting the router, |  | ||||||
| // then the cleaner. If something goes wrong while starting the |  | ||||||
| // server, then an error will be returned. |  | ||||||
| func (s *Server) Start(ctx context.Context) error { |  | ||||||
| 	s.apiRouter.Start() |  | ||||||
| 	return s.cleaner.ScheduleJobs() |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // Stop closes down the GoToSocial server, first closing the cleaner, |  | ||||||
| // then the router, then the database. If something goes wrong while |  | ||||||
| // stopping, an error will be returned. |  | ||||||
| func (s *Server) Stop(ctx context.Context) error { |  | ||||||
| 	if err := s.apiRouter.Stop(ctx); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return s.db.Close() |  | ||||||
| } |  | ||||||
|  | @ -122,7 +122,7 @@ func New(ctx context.Context) (*Router, error) { | ||||||
| // | // | ||||||
| // It will serve two handlers if letsencrypt is enabled, | // It will serve two handlers if letsencrypt is enabled, | ||||||
| // and only the web/API handler if letsencrypt is not enabled. | // and only the web/API handler if letsencrypt is not enabled. | ||||||
| func (r *Router) Start() { | func (r *Router) Start() error { | ||||||
| 	var ( | 	var ( | ||||||
| 		// listen is the server start function. | 		// listen is the server start function. | ||||||
| 		// By default this points to a regular | 		// By default this points to a regular | ||||||
|  | @ -143,10 +143,16 @@ func (r *Router) Start() { | ||||||
| 		// that either both or neither of Chain and Key | 		// that either both or neither of Chain and Key | ||||||
| 		// are set, so we can forego checking again here. | 		// are set, so we can forego checking again here. | ||||||
| 		listen, err = r.customTLS(certFile, keyFile) | 		listen, err = r.customTLS(certFile, keyFile) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
| 
 | 
 | ||||||
| 	// TLS with letsencrypt. | 	// TLS with letsencrypt. | ||||||
| 	case leEnabled: | 	case leEnabled: | ||||||
| 		listen, err = r.letsEncryptTLS() | 		listen, err = r.letsEncryptTLS() | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
| 
 | 
 | ||||||
| 	// Default listen. TLS must | 	// Default listen. TLS must | ||||||
| 	// be handled by reverse proxy. | 	// be handled by reverse proxy. | ||||||
|  | @ -154,10 +160,6 @@ func (r *Router) Start() { | ||||||
| 		listen = r.srv.ListenAndServe | 		listen = r.srv.ListenAndServe | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if err != nil { |  | ||||||
| 		log.Fatal(nil, err) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	// Pass the server handler through a debug pprof middleware handler. | 	// Pass the server handler through a debug pprof middleware handler. | ||||||
| 	// For standard production builds this will be a no-op, but when the | 	// For standard production builds this will be a no-op, but when the | ||||||
| 	// "debug" or "debugenv" build-tag is set pprof stats will be served | 	// "debug" or "debugenv" build-tag is set pprof stats will be served | ||||||
|  | @ -177,12 +179,14 @@ func (r *Router) Start() { | ||||||
| 			log.Fatalf(nil, "listen: %s", err) | 			log.Fatalf(nil, "listen: %s", err) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|  | 
 | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Stop shuts down the router nicely | // Stop shuts down the router nicely. | ||||||
| func (r *Router) Stop(ctx context.Context) error { | func (r *Router) Stop() error { | ||||||
| 	log.Infof(nil, "shutting down http router with %s grace period", shutdownTimeout) | 	log.Infof(nil, "shutting down http router with %s grace period", shutdownTimeout) | ||||||
| 	timeout, cancel := context.WithTimeout(ctx, shutdownTimeout) | 	timeout, cancel := context.WithTimeout(context.Background(), shutdownTimeout) | ||||||
| 	defer cancel() | 	defer cancel() | ||||||
| 
 | 
 | ||||||
| 	if err := r.srv.Shutdown(timeout); err != nil { | 	if err := r.srv.Shutdown(timeout); err != nil { | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue