| 
									
										
										
										
											2023-03-12 16:00:57 +01: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/>. | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | package fedi | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-26 15:34:10 +02:00
										 |  |  | 	apimodel "code.superseriousbusiness.org/gotosocial/internal/api/model" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/config" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/gtserror" | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							| 
									
										
										
										
											2023-03-09 18:55:45 +01:00
										 |  |  | 	hostMetaXMLNS                   = "http://docs.oasis-open.org/ns/xri/xrd-1.0" | 
					
						
							|  |  |  | 	hostMetaRel                     = "lrdd" | 
					
						
							|  |  |  | 	hostMetaType                    = "application/xrd+xml" | 
					
						
							|  |  |  | 	hostMetaTemplate                = ".well-known/webfinger?resource={uri}" | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 	nodeInfoSoftwareName            = "gotosocial" | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 	nodeInfo20Rel                   = "http://nodeinfo.diaspora.software/ns/schema/2.0" | 
					
						
							|  |  |  | 	nodeInfo21Rel                   = "http://nodeinfo.diaspora.software/ns/schema/2.1" | 
					
						
							| 
									
										
										
										
											2025-04-26 15:34:10 +02:00
										 |  |  | 	nodeInfoRepo                    = "https://codeberg.org/superseriousbusiness/gotosocial" | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 	nodeInfoHomepage                = "https://docs.gotosocial.org" | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 	webfingerProfilePage            = "http://webfinger.net/rel/profile-page" | 
					
						
							|  |  |  | 	webFingerProfilePageContentType = "text/html" | 
					
						
							|  |  |  | 	webfingerSelf                   = "self" | 
					
						
							|  |  |  | 	webFingerSelfContentType        = "application/activity+json" | 
					
						
							|  |  |  | 	webfingerAccount                = "acct" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	nodeInfoProtocols = []string{"activitypub"} | 
					
						
							|  |  |  | 	nodeInfoInbound   = []string{} | 
					
						
							|  |  |  | 	nodeInfoOutbound  = []string{} | 
					
						
							|  |  |  | 	nodeInfoMetadata  = make(map[string]interface{}) | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NodeInfoRelGet returns a well known response giving the path to node info. | 
					
						
							|  |  |  | func (p *Processor) NodeInfoRelGet(ctx context.Context) (*apimodel.WellKnownResponse, gtserror.WithCode) { | 
					
						
							|  |  |  | 	protocol := config.GetProtocol() | 
					
						
							|  |  |  | 	host := config.GetHost() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return &apimodel.WellKnownResponse{ | 
					
						
							|  |  |  | 		Links: []apimodel.Link{ | 
					
						
							|  |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 				Rel:  nodeInfo20Rel, | 
					
						
							|  |  |  | 				Href: fmt.Sprintf("%s://%s/nodeinfo/2.0", protocol, host), | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Rel:  nodeInfo21Rel, | 
					
						
							|  |  |  | 				Href: fmt.Sprintf("%s://%s/nodeinfo/2.1", protocol, host), | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | // NodeInfoGet returns a node info struct in response to a 2.0 or 2.1 node info request. | 
					
						
							|  |  |  | func (p *Processor) NodeInfoGet(ctx context.Context, schemaVersion string) (*apimodel.Nodeinfo, gtserror.WithCode) { | 
					
						
							|  |  |  | 	const () | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-31 19:27:18 +01:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		userCount int | 
					
						
							|  |  |  | 		postCount int | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 		mau       int | 
					
						
							| 
									
										
										
										
											2025-01-31 19:27:18 +01:00
										 |  |  | 		err       error | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 	switch config.GetInstanceStatsMode() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case config.InstanceStatsModeBaffle: | 
					
						
							| 
									
										
										
										
											2025-01-31 19:27:18 +01:00
										 |  |  | 		// Use randomized stats. | 
					
						
							|  |  |  | 		stats := p.converter.RandomStats() | 
					
						
							|  |  |  | 		userCount = int(stats.TotalUsers) | 
					
						
							|  |  |  | 		postCount = int(stats.Statuses) | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 		mau = int(stats.MonthlyActiveUsers) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case config.InstanceStatsModeZero: | 
					
						
							|  |  |  | 		// Use zeroed stats | 
					
						
							|  |  |  | 		// (don't count anything). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		// Mode is either "serve" or "default". | 
					
						
							| 
									
										
										
										
											2025-01-31 19:27:18 +01:00
										 |  |  | 		// Count actual stats. | 
					
						
							|  |  |  | 		host := config.GetHost() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		userCount, err = p.state.DB.CountInstanceUsers(ctx, host) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, gtserror.NewErrorInternalError(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-31 19:27:18 +01:00
										 |  |  | 		postCount, err = p.state.DB.CountInstanceStatuses(ctx, host) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, gtserror.NewErrorInternalError(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 	nodeInfo := &apimodel.Nodeinfo{ | 
					
						
							|  |  |  | 		Version: schemaVersion, | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 		Software: apimodel.NodeInfoSoftware{ | 
					
						
							|  |  |  | 			Name:    nodeInfoSoftwareName, | 
					
						
							|  |  |  | 			Version: config.GetSoftwareVersion(), | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		Protocols: nodeInfoProtocols, | 
					
						
							|  |  |  | 		Services: apimodel.NodeInfoServices{ | 
					
						
							|  |  |  | 			Inbound:  nodeInfoInbound, | 
					
						
							|  |  |  | 			Outbound: nodeInfoOutbound, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		OpenRegistrations: config.GetAccountsRegistrationOpen(), | 
					
						
							|  |  |  | 		Usage: apimodel.NodeInfoUsage{ | 
					
						
							|  |  |  | 			Users: apimodel.NodeInfoUsers{ | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 				Total:       userCount, | 
					
						
							|  |  |  | 				ActiveMonth: mau, | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 			}, | 
					
						
							|  |  |  | 			LocalPosts: postCount, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		Metadata: nodeInfoMetadata, | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-04 17:52:54 +01:00
										 |  |  | 	if schemaVersion == "2.1" { | 
					
						
							| 
									
										
										
										
											2025-02-04 16:52:42 +01:00
										 |  |  | 		nodeInfo.Software.Repository = nodeInfoRepo | 
					
						
							|  |  |  | 		nodeInfo.Software.Homepage = nodeInfoHomepage | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nodeInfo, nil | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-09 18:55:45 +01:00
										 |  |  | // HostMetaGet returns a host-meta struct in response to a host-meta request. | 
					
						
							|  |  |  | func (p *Processor) HostMetaGet() *apimodel.HostMeta { | 
					
						
							|  |  |  | 	protocol := config.GetProtocol() | 
					
						
							|  |  |  | 	host := config.GetHost() | 
					
						
							|  |  |  | 	return &apimodel.HostMeta{ | 
					
						
							|  |  |  | 		XMLNS: hostMetaXMLNS, | 
					
						
							|  |  |  | 		Link: []apimodel.Link{ | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Rel:      hostMetaRel, | 
					
						
							|  |  |  | 				Type:     hostMetaType, | 
					
						
							|  |  |  | 				Template: fmt.Sprintf("%s://%s/%s", protocol, host, hostMetaTemplate), | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | // WebfingerGet handles the GET for a webfinger resource. Most commonly, it will be used for returning account lookups. | 
					
						
							|  |  |  | func (p *Processor) WebfingerGet(ctx context.Context, requestedUsername string) (*apimodel.WellKnownResponse, gtserror.WithCode) { | 
					
						
							|  |  |  | 	// Get the local account the request is referring to. | 
					
						
							| 
									
										
										
										
											2023-03-01 18:26:53 +00:00
										 |  |  | 	requestedAccount, err := p.state.DB.GetAccountByUsernameDomain(ctx, requestedUsername, "") | 
					
						
							| 
									
										
										
										
											2023-02-22 16:05:26 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, gtserror.NewErrorNotFound(fmt.Errorf("database error getting account with username %s: %s", requestedUsername, err)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return &apimodel.WellKnownResponse{ | 
					
						
							|  |  |  | 		Subject: webfingerAccount + ":" + requestedAccount.Username + "@" + config.GetAccountDomain(), | 
					
						
							|  |  |  | 		Aliases: []string{ | 
					
						
							|  |  |  | 			requestedAccount.URI, | 
					
						
							|  |  |  | 			requestedAccount.URL, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		Links: []apimodel.Link{ | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Rel:  webfingerProfilePage, | 
					
						
							|  |  |  | 				Type: webFingerProfilePageContentType, | 
					
						
							|  |  |  | 				Href: requestedAccount.URL, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Rel:  webfingerSelf, | 
					
						
							|  |  |  | 				Type: webFingerSelfContentType, | 
					
						
							|  |  |  | 				Href: requestedAccount.URI, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, nil | 
					
						
							|  |  |  | } |