| 
									
										
										
										
											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/>. | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | package web | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 	"context" | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2022-04-15 14:33:01 +02:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/gin-gonic/gin" | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" | 
					
						
							| 
									
										
										
										
											2023-01-02 13:10:50 +01:00
										 |  |  | 	apiutil "github.com/superseriousbusiness/gotosocial/internal/api/util" | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtserror" | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | func (m *Module) threadGETHandler(c *gin.Context) { | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 	ctx := c.Request.Context() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	// We'll need the instance later, and we can also use it | 
					
						
							|  |  |  | 	// before then to make it easier to return a web error. | 
					
						
							|  |  |  | 	instance, errWithCode := m.processor.InstanceGetV1(ctx) | 
					
						
							|  |  |  | 	if errWithCode != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, errWithCode, m.processor.InstanceGetV1) | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	// Return instance we already got from the db, | 
					
						
							|  |  |  | 	// don't try to fetch it again when erroring. | 
					
						
							|  |  |  | 	instanceGet := func(ctx context.Context) (*apimodel.InstanceV1, gtserror.WithCode) { | 
					
						
							|  |  |  | 		return instance, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Parse account targetUsername and status ID from the URL. | 
					
						
							| 
									
										
										
										
											2024-04-13 13:25:10 +02:00
										 |  |  | 	targetUsername, errWithCode := apiutil.ParseUsername(c.Param(apiutil.UsernameKey)) | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	if errWithCode != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, errWithCode, instanceGet) | 
					
						
							| 
									
										
										
										
											2022-04-15 14:33:01 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	targetStatusID, errWithCode := apiutil.ParseWebStatusID(c.Param(apiutil.WebStatusIDKey)) | 
					
						
							|  |  |  | 	if errWithCode != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, errWithCode, instanceGet) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	// Normalize requested username + status ID: | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	//   - Usernames on our instance are (currently) always lowercase. | 
					
						
							|  |  |  | 	//   - StatusIDs on our instance are (currently) always ULIDs. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// todo: Update this logic when different username patterns | 
					
						
							|  |  |  | 	// are allowed, and/or when status slugs are introduced. | 
					
						
							|  |  |  | 	targetUsername = strings.ToLower(targetUsername) | 
					
						
							|  |  |  | 	targetStatusID = strings.ToUpper(targetStatusID) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Check what type of content is being requested. If we're getting an AP | 
					
						
							|  |  |  | 	// request on this endpoint we should render the AP representation instead. | 
					
						
							|  |  |  | 	accept, err := apiutil.NegotiateAccept(c, apiutil.HTMLOrActivityPubHeaders...) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorNotAcceptable(err, err.Error()), instanceGet) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	if accept == string(apiutil.AppActivityJSON) || accept == string(apiutil.AppActivityLDJSON) { | 
					
						
							|  |  |  | 		// AP status representation has been requested. | 
					
						
							|  |  |  | 		m.returnAPStatus(c, targetUsername, targetStatusID, accept, instanceGet) | 
					
						
							|  |  |  | 		return | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	// text/html has been requested. Proceed with getting the web view of the status. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Fetch the target account so we can do some checks on it. | 
					
						
							| 
									
										
										
										
											2024-07-08 15:47:03 +02:00
										 |  |  | 	targetAccount, errWithCode := m.processor.Account().GetWeb(ctx, targetUsername) | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 	if errWithCode != nil { | 
					
						
							| 
									
										
										
										
											2023-05-12 08:16:41 +00:00
										 |  |  | 		apiutil.WebErrorHandler(c, errWithCode, instanceGet) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	// If target account is suspended, this page should not be visible. | 
					
						
							|  |  |  | 	if targetAccount.Suspended { | 
					
						
							|  |  |  | 		err := fmt.Errorf("target account %s is suspended", targetUsername) | 
					
						
							| 
									
										
										
										
											2023-05-12 08:16:41 +00:00
										 |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorNotFound(err), instanceGet) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 20:36:03 +02:00
										 |  |  | 	// Get the thread context. This will fetch the target status as well. | 
					
						
							|  |  |  | 	context, errWithCode := m.processor.Status().WebContextGet(ctx, targetStatusID) | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	if errWithCode != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, errWithCode, instanceGet) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Ensure status actually belongs to target account. | 
					
						
							| 
									
										
										
										
											2024-07-21 14:22:08 +02:00
										 |  |  | 	if context.Status.Account.ID != targetAccount.ID { | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 		err := fmt.Errorf("target account %s does not own status %s", targetUsername, targetStatusID) | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorNotFound(err), instanceGet) | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-25 18:32:24 +01:00
										 |  |  | 	// Prepare stylesheets for thread. | 
					
						
							| 
									
										
										
										
											2024-12-02 06:24:48 -05:00
										 |  |  | 	stylesheets := make([]string, 0, 6) | 
					
						
							| 
									
										
										
										
											2024-03-25 18:32:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Basic thread stylesheets. | 
					
						
							|  |  |  | 	stylesheets = append( | 
					
						
							|  |  |  | 		stylesheets, | 
					
						
							|  |  |  | 		[]string{ | 
					
						
							|  |  |  | 			cssFA, | 
					
						
							|  |  |  | 			cssStatus, | 
					
						
							|  |  |  | 			cssThread, | 
					
						
							|  |  |  | 		}..., | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// User-selected theme if set. | 
					
						
							|  |  |  | 	if theme := targetAccount.Theme; theme != "" { | 
					
						
							|  |  |  | 		stylesheets = append( | 
					
						
							|  |  |  | 			stylesheets, | 
					
						
							|  |  |  | 			themesPathPrefix+"/"+theme, | 
					
						
							|  |  |  | 		) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Custom CSS for this user last in cascade. | 
					
						
							|  |  |  | 	stylesheets = append( | 
					
						
							|  |  |  | 		stylesheets, | 
					
						
							|  |  |  | 		"/@"+targetAccount.Username+"/custom.css", | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	page := apiutil.WebPage{ | 
					
						
							| 
									
										
										
										
											2024-03-25 18:32:24 +01:00
										 |  |  | 		Template:    "thread.tmpl", | 
					
						
							|  |  |  | 		Instance:    instance, | 
					
						
							| 
									
										
										
										
											2024-07-12 20:36:03 +02:00
										 |  |  | 		OGMeta:      apiutil.OGBase(instance).WithStatus(context.Status), | 
					
						
							| 
									
										
										
										
											2024-03-25 18:32:24 +01:00
										 |  |  | 		Stylesheets: stylesheets, | 
					
						
							| 
									
										
										
										
											2025-03-31 15:51:17 +02:00
										 |  |  | 		Javascript: []apiutil.JavascriptEntry{ | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Src:   jsFrontend, | 
					
						
							|  |  |  | 				Async: true, | 
					
						
							|  |  |  | 				Defer: true, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Bottom: true, | 
					
						
							| 
									
										
										
										
											2025-04-22 12:20:54 +02:00
										 |  |  | 				Src:    jsFrontendPrerender, | 
					
						
							| 
									
										
										
										
											2025-03-31 15:51:17 +02:00
										 |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 		Extra: map[string]any{ | 
					
						
							|  |  |  | 			"context": context, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2022-09-12 13:14:29 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	apiutil.TemplateWebPage(c, page) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | // returnAPStatus returns an ActivityPub representation of target status, | 
					
						
							|  |  |  | // created by targetUsername. It will do http signature authentication. | 
					
						
							|  |  |  | func (m *Module) returnAPStatus( | 
					
						
							|  |  |  | 	c *gin.Context, | 
					
						
							|  |  |  | 	targetUsername string, | 
					
						
							|  |  |  | 	targetStatusID string, | 
					
						
							|  |  |  | 	accept string, | 
					
						
							|  |  |  | 	instanceGet func(ctx context.Context) (*apimodel.InstanceV1, gtserror.WithCode), | 
					
						
							|  |  |  | ) { | 
					
						
							|  |  |  | 	status, errWithCode := m.processor.Fedi().StatusGet(c.Request.Context(), targetUsername, targetStatusID) | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 	if errWithCode != nil { | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 		apiutil.WebErrorHandler(c, errWithCode, instanceGet) | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	b, err := json.Marshal(status) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		err := gtserror.Newf("could not marshal json: %w", err) | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorInternalError(err), instanceGet) | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c.Data(http.StatusOK, accept, b) | 
					
						
							|  |  |  | } |