| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | // 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/>. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-06 12:14:37 +01:00
										 |  |  | //go:build !nootel | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-06 12:14:37 +01:00
										 |  |  | package observability | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2024-02-14 12:58:55 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2025-05-05 16:22:45 +00:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-26 15:34:10 +02:00
										 |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/config" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/db" | 
					
						
							| 
									
										
										
										
											2025-02-06 12:14:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/gin-gonic/gin" | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	"github.com/technologize/otel-go-contrib/otelginmetrics" | 
					
						
							| 
									
										
										
										
											2025-05-05 16:22:45 +00:00
										 |  |  | 	"go.opentelemetry.io/contrib/exporters/autoexport" | 
					
						
							| 
									
										
										
										
											2025-05-06 08:18:05 +00:00
										 |  |  | 	"go.opentelemetry.io/contrib/instrumentation/runtime" | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	"go.opentelemetry.io/otel" | 
					
						
							| 
									
										
										
										
											2024-02-14 12:58:55 +02:00
										 |  |  | 	"go.opentelemetry.io/otel/metric" | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	sdk "go.opentelemetry.io/otel/sdk/metric" | 
					
						
							| 
									
										
										
										
											2025-02-06 12:14:37 +01:00
										 |  |  | 	"go.opentelemetry.io/otel/sdk/metric/exemplar" | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-05 16:22:45 +00:00
										 |  |  | func InitializeMetrics(ctx context.Context, db db.DB) error { | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	if !config.GetMetricsEnabled() { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-05 16:22:45 +00:00
										 |  |  | 	r, err := Resource() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		// this can happen if semconv versioning is out-of-sync | 
					
						
							|  |  |  | 		return fmt.Errorf("building tracing resource: %w", err) | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-05 16:22:45 +00:00
										 |  |  | 	mt, err := autoexport.NewMetricReader(ctx) | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	meterProvider := sdk.NewMeterProvider( | 
					
						
							| 
									
										
										
										
											2025-02-06 12:14:37 +01:00
										 |  |  | 		sdk.WithExemplarFilter(exemplar.AlwaysOffFilter), | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 		sdk.WithResource(r), | 
					
						
							| 
									
										
										
										
											2025-05-05 16:22:45 +00:00
										 |  |  | 		sdk.WithReader(mt), | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	) | 
					
						
							| 
									
										
										
										
											2024-02-14 12:58:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	otel.SetMeterProvider(meterProvider) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-06 08:18:05 +00:00
										 |  |  | 	if err := runtime.Start( | 
					
						
							|  |  |  | 		runtime.WithMeterProvider(meterProvider), | 
					
						
							|  |  |  | 	); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-14 12:58:55 +02:00
										 |  |  | 	meter := meterProvider.Meter(serviceName) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	thisInstance := config.GetHost() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_, err = meter.Int64ObservableGauge( | 
					
						
							|  |  |  | 		"gotosocial.instance.total_users", | 
					
						
							|  |  |  | 		metric.WithDescription("Total number of users on this instance"), | 
					
						
							|  |  |  | 		metric.WithInt64Callback(func(c context.Context, o metric.Int64Observer) error { | 
					
						
							|  |  |  | 			userCount, err := db.CountInstanceUsers(c, thisInstance) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			o.Observe(int64(userCount)) | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}), | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_, err = meter.Int64ObservableGauge( | 
					
						
							|  |  |  | 		"gotosocial.instance.total_statuses", | 
					
						
							|  |  |  | 		metric.WithDescription("Total number of statuses on this instance"), | 
					
						
							|  |  |  | 		metric.WithInt64Callback(func(c context.Context, o metric.Int64Observer) error { | 
					
						
							|  |  |  | 			statusCount, err := db.CountInstanceStatuses(c, thisInstance) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			o.Observe(int64(statusCount)) | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}), | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_, err = meter.Int64ObservableGauge( | 
					
						
							|  |  |  | 		"gotosocial.instance.total_federating_instances", | 
					
						
							|  |  |  | 		metric.WithDescription("Total number of other instances this instance is federating with"), | 
					
						
							|  |  |  | 		metric.WithInt64Callback(func(c context.Context, o metric.Int64Observer) error { | 
					
						
							|  |  |  | 			federatingCount, err := db.CountInstanceDomains(c, thisInstance) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			o.Observe(int64(federatingCount)) | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}), | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-06 12:14:37 +01:00
										 |  |  | func MetricsMiddleware() gin.HandlerFunc { | 
					
						
							| 
									
										
										
										
											2023-11-20 17:43:55 +02:00
										 |  |  | 	return otelginmetrics.Middleware(serviceName) | 
					
						
							|  |  |  | } |