| 
									
										
										
										
											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" | 
					
						
							| 
									
										
										
										
											2021-12-07 13:31:39 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/config" | 
					
						
							| 
									
										
										
										
											2022-06-08 20:38:03 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtserror" | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/oauth" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  | 	targetUsername, errWithCode := apiutil.ParseWebUsername(c.Param(apiutil.WebUsernameKey)) | 
					
						
							|  |  |  | 	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. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Don't require auth for web endpoints, but do take it if it was provided. | 
					
						
							|  |  |  | 	// authed.Account might end up nil here, but that's fine in case of public pages. | 
					
						
							|  |  |  | 	authed, err := oauth.Authed(c, false, false, false, false) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorUnauthorized(err, err.Error()), m.processor.InstanceGetV1) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	// Fetch the target account so we can do some checks on it. | 
					
						
							|  |  |  | 	targetAccount, errWithCode := m.processor.Account().GetLocalByUsername(ctx, authed.Account, 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 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	// Get the status itself from the processor using provided ID and authorization (if any). | 
					
						
							| 
									
										
										
										
											2023-11-17 11:35:28 +01:00
										 |  |  | 	status, errWithCode := m.processor.Status().WebGet(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. | 
					
						
							|  |  |  | 	if status.GetAccountID() != targetAccount.ID { | 
					
						
							|  |  |  | 		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 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 	// Fill in the rest of the thread context. | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	context, errWithCode := m.processor.Status().WebContextGet(ctx, targetStatusID) | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-12 13:14:29 +02:00
										 |  |  | 	stylesheets := []string{ | 
					
						
							| 
									
										
										
										
											2022-11-23 15:32:57 +01:00
										 |  |  | 		assetsPathPrefix + "/Fork-Awesome/css/fork-awesome.min.css", | 
					
						
							|  |  |  | 		distPathPrefix + "/status.css", | 
					
						
							| 
									
										
										
										
											2022-09-12 13:14:29 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if config.GetAccountsAllowCustomCSS() { | 
					
						
							| 
									
										
										
										
											2023-07-07 14:58:53 +02:00
										 |  |  | 		stylesheets = append(stylesheets, "/@"+targetUsername+"/custom.css") | 
					
						
							| 
									
										
										
										
											2022-09-12 13:14:29 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 	c.HTML(http.StatusOK, "thread.tmpl", gin.H{ | 
					
						
							| 
									
										
										
										
											2022-09-12 13:14:29 +02:00
										 |  |  | 		"instance":    instance, | 
					
						
							|  |  |  | 		"status":      status, | 
					
						
							|  |  |  | 		"context":     context, | 
					
						
							|  |  |  | 		"ogMeta":      ogBase(instance).withStatus(status), | 
					
						
							|  |  |  | 		"stylesheets": stylesheets, | 
					
						
							| 
									
										
										
										
											2022-11-23 15:32:57 +01:00
										 |  |  | 		"javascript":  []string{distPathPrefix + "/frontend.js"}, | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | } |