| 
									
										
										
										
											2024-04-11 11:45:53 +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/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package web | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"net" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/gin-gonic/gin" | 
					
						
							|  |  |  | 	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" | 
					
						
							|  |  |  | 	apiutil "github.com/superseriousbusiness/gotosocial/internal/api/util" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/config" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtserror" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/validate" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (m *Module) signupGETHandler(c *gin.Context) { | 
					
						
							|  |  |  | 	ctx := c.Request.Context() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 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) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 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 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// We only serve text/html at this endpoint. | 
					
						
							|  |  |  | 	if _, err := apiutil.NegotiateAccept(c, apiutil.TextHTML); err != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorNotAcceptable(err, err.Error()), instanceGet) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	page := apiutil.WebPage{ | 
					
						
							|  |  |  | 		Template: "sign-up.tmpl", | 
					
						
							|  |  |  | 		Instance: instance, | 
					
						
							|  |  |  | 		OGMeta:   apiutil.OGBase(instance), | 
					
						
							|  |  |  | 		Extra: map[string]any{ | 
					
						
							| 
									
										
										
										
											2024-06-12 16:00:21 +02:00
										 |  |  | 			"reasonRequired":   config.GetAccountsReasonRequired(), | 
					
						
							|  |  |  | 			"registrationOpen": config.GetAccountsRegistrationOpen(), | 
					
						
							| 
									
										
										
										
											2024-04-11 11:45:53 +02:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	apiutil.TemplateWebPage(c, page) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (m *Module) signupPOSTHandler(c *gin.Context) { | 
					
						
							|  |  |  | 	ctx := c.Request.Context() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 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) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 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 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// We only serve text/html at this endpoint. | 
					
						
							|  |  |  | 	if _, err := apiutil.NegotiateAccept(c, apiutil.TextHTML); err != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorNotAcceptable(err, err.Error()), instanceGet) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	form := &apimodel.AccountCreateRequest{} | 
					
						
							|  |  |  | 	if err := c.ShouldBind(form); err != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorBadRequest(err, err.Error()), instanceGet) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err := validate.CreateAccount(form); err != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorBadRequest(err, err.Error()), instanceGet) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	clientIP := c.ClientIP() | 
					
						
							|  |  |  | 	signUpIP := net.ParseIP(clientIP) | 
					
						
							|  |  |  | 	if signUpIP == nil { | 
					
						
							|  |  |  | 		err := errors.New("ip address could not be parsed from request") | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, gtserror.NewErrorBadRequest(err, err.Error()), instanceGet) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	form.IP = signUpIP | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-06 15:43:25 +02:00
										 |  |  | 	// We have all the info we need, call user+account create | 
					
						
							| 
									
										
										
										
											2024-04-11 11:45:53 +02:00
										 |  |  | 	// (this will also trigger side effects like sending emails etc). | 
					
						
							| 
									
										
										
										
											2024-06-06 15:43:25 +02:00
										 |  |  | 	user, errWithCode := m.processor.User().Create( | 
					
						
							| 
									
										
										
										
											2024-04-11 11:45:53 +02:00
										 |  |  | 		c.Request.Context(), | 
					
						
							|  |  |  | 		// nil to use | 
					
						
							|  |  |  | 		// instance app. | 
					
						
							|  |  |  | 		nil, | 
					
						
							|  |  |  | 		form, | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	if errWithCode != nil { | 
					
						
							|  |  |  | 		apiutil.WebErrorHandler(c, errWithCode, instanceGet) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Serve a page informing the | 
					
						
							|  |  |  | 	// user that they've signed up. | 
					
						
							|  |  |  | 	page := apiutil.WebPage{ | 
					
						
							| 
									
										
										
										
											2024-12-18 17:47:17 +08:00
										 |  |  | 		Template: "signed-up.tmpl", | 
					
						
							|  |  |  | 		Instance: instance, | 
					
						
							|  |  |  | 		OGMeta:   apiutil.OGBase(instance), | 
					
						
							| 
									
										
										
										
											2024-04-11 11:45:53 +02:00
										 |  |  | 		Extra: map[string]any{ | 
					
						
							|  |  |  | 			"email":    user.UnconfirmedEmail, | 
					
						
							|  |  |  | 			"username": user.Account.Username, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	apiutil.TemplateWebPage(c, page) | 
					
						
							|  |  |  | } |