mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-11-04 09:12:24 -06:00 
			
		
		
		
	* update typeconverter to use state structure * deinterface the typeutils.TypeConverter -> typeutils.Converter * finish copying over old type converter code comments * fix cherry-pick merge issues, fix tests pointing to old typeutils interface type still
		
			
				
	
	
		
			335 lines
		
	
	
	
		
			9.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			335 lines
		
	
	
	
		
			9.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// 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 admin
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"encoding/json"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"mime/multipart"
 | 
						|
	"net/http"
 | 
						|
 | 
						|
	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/db"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/gtserror"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
 | 
						|
)
 | 
						|
 | 
						|
// apiDomainPerm is a cheeky shortcut for returning
 | 
						|
// the API version of the given domain permission
 | 
						|
// (*gtsmodel.DomainBlock or *gtsmodel.DomainAllow),
 | 
						|
// or an appropriate error if something goes wrong.
 | 
						|
func (p *Processor) apiDomainPerm(
 | 
						|
	ctx context.Context,
 | 
						|
	domainPermission gtsmodel.DomainPermission,
 | 
						|
	export bool,
 | 
						|
) (*apimodel.DomainPermission, gtserror.WithCode) {
 | 
						|
	apiDomainPerm, err := p.converter.DomainPermToAPIDomainPerm(ctx, domainPermission, export)
 | 
						|
	if err != nil {
 | 
						|
		err := gtserror.NewfAt(3, "error converting domain permission to api model: %w", err)
 | 
						|
		return nil, gtserror.NewErrorInternalError(err)
 | 
						|
	}
 | 
						|
 | 
						|
	return apiDomainPerm, nil
 | 
						|
}
 | 
						|
 | 
						|
// DomainPermissionCreate creates an instance-level permission
 | 
						|
// targeting the given domain, and then processes any side
 | 
						|
// effects of the permission creation.
 | 
						|
//
 | 
						|
// If the same permission type already exists for the domain,
 | 
						|
// side effects will be retried.
 | 
						|
//
 | 
						|
// Return values for this function are the new or existing
 | 
						|
// domain permission, the ID of the admin action resulting
 | 
						|
// from this call, and/or an error if something goes wrong.
 | 
						|
func (p *Processor) DomainPermissionCreate(
 | 
						|
	ctx context.Context,
 | 
						|
	permissionType gtsmodel.DomainPermissionType,
 | 
						|
	adminAcct *gtsmodel.Account,
 | 
						|
	domain string,
 | 
						|
	obfuscate bool,
 | 
						|
	publicComment string,
 | 
						|
	privateComment string,
 | 
						|
	subscriptionID string,
 | 
						|
) (*apimodel.DomainPermission, string, gtserror.WithCode) {
 | 
						|
	switch permissionType {
 | 
						|
 | 
						|
	// Explicitly block a domain.
 | 
						|
	case gtsmodel.DomainPermissionBlock:
 | 
						|
		return p.createDomainBlock(
 | 
						|
			ctx,
 | 
						|
			adminAcct,
 | 
						|
			domain,
 | 
						|
			obfuscate,
 | 
						|
			publicComment,
 | 
						|
			privateComment,
 | 
						|
			subscriptionID,
 | 
						|
		)
 | 
						|
 | 
						|
	// Explicitly allow a domain.
 | 
						|
	case gtsmodel.DomainPermissionAllow:
 | 
						|
		return p.createDomainAllow(
 | 
						|
			ctx,
 | 
						|
			adminAcct,
 | 
						|
			domain,
 | 
						|
			obfuscate,
 | 
						|
			publicComment,
 | 
						|
			privateComment,
 | 
						|
			subscriptionID,
 | 
						|
		)
 | 
						|
 | 
						|
	// Weeping, roaring, red-faced.
 | 
						|
	default:
 | 
						|
		err := gtserror.Newf("unrecognized permission type %d", permissionType)
 | 
						|
		return nil, "", gtserror.NewErrorInternalError(err)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// DomainPermissionDelete removes one domain block with the given ID,
 | 
						|
// and processes side effects of removing the block asynchronously.
 | 
						|
//
 | 
						|
// Return values for this function are the deleted domain block, the ID of the admin
 | 
						|
// action resulting from this call, and/or an error if something goes wrong.
 | 
						|
func (p *Processor) DomainPermissionDelete(
 | 
						|
	ctx context.Context,
 | 
						|
	permissionType gtsmodel.DomainPermissionType,
 | 
						|
	adminAcct *gtsmodel.Account,
 | 
						|
	domainBlockID string,
 | 
						|
) (*apimodel.DomainPermission, string, gtserror.WithCode) {
 | 
						|
	switch permissionType {
 | 
						|
 | 
						|
	// Delete explicit domain block.
 | 
						|
	case gtsmodel.DomainPermissionBlock:
 | 
						|
		return p.deleteDomainBlock(
 | 
						|
			ctx,
 | 
						|
			adminAcct,
 | 
						|
			domainBlockID,
 | 
						|
		)
 | 
						|
 | 
						|
	// Delete explicit domain allow.
 | 
						|
	case gtsmodel.DomainPermissionAllow:
 | 
						|
		return p.deleteDomainAllow(
 | 
						|
			ctx,
 | 
						|
			adminAcct,
 | 
						|
			domainBlockID,
 | 
						|
		)
 | 
						|
 | 
						|
	// You do the hokey-cokey and you turn
 | 
						|
	// around, that's what it's all about.
 | 
						|
	default:
 | 
						|
		err := gtserror.Newf("unrecognized permission type %d", permissionType)
 | 
						|
		return nil, "", gtserror.NewErrorInternalError(err)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// DomainPermissionsImport handles the import of multiple
 | 
						|
// domain permissions, by calling the DomainPermissionCreate
 | 
						|
// function for each domain in the provided file. Will return
 | 
						|
// a slice of processed domain permissions.
 | 
						|
//
 | 
						|
// In the case of total failure, a gtserror.WithCode will be
 | 
						|
// returned so that the caller can respond appropriately. In
 | 
						|
// the case of partial or total success, a MultiStatus model
 | 
						|
// will be returned, which contains information about success
 | 
						|
// + failure count, so that the caller can retry any failures
 | 
						|
// as they wish.
 | 
						|
func (p *Processor) DomainPermissionsImport(
 | 
						|
	ctx context.Context,
 | 
						|
	permissionType gtsmodel.DomainPermissionType,
 | 
						|
	account *gtsmodel.Account,
 | 
						|
	domainsF *multipart.FileHeader,
 | 
						|
) (*apimodel.MultiStatus, gtserror.WithCode) {
 | 
						|
	// Ensure known permission type.
 | 
						|
	if permissionType != gtsmodel.DomainPermissionBlock &&
 | 
						|
		permissionType != gtsmodel.DomainPermissionAllow {
 | 
						|
		err := gtserror.Newf("unrecognized permission type %d", permissionType)
 | 
						|
		return nil, gtserror.NewErrorInternalError(err)
 | 
						|
	}
 | 
						|
 | 
						|
	// Open the provided file.
 | 
						|
	file, err := domainsF.Open()
 | 
						|
	if err != nil {
 | 
						|
		err = gtserror.Newf("error opening attachment: %w", err)
 | 
						|
		return nil, gtserror.NewErrorBadRequest(err, err.Error())
 | 
						|
	}
 | 
						|
	defer file.Close()
 | 
						|
 | 
						|
	// Parse file as slice of domain blocks.
 | 
						|
	domainPerms := make([]*apimodel.DomainPermission, 0)
 | 
						|
	if err := json.NewDecoder(file).Decode(&domainPerms); err != nil {
 | 
						|
		err = gtserror.Newf("error parsing attachment as domain permissions: %w", err)
 | 
						|
		return nil, gtserror.NewErrorBadRequest(err, err.Error())
 | 
						|
	}
 | 
						|
 | 
						|
	count := len(domainPerms)
 | 
						|
	if count == 0 {
 | 
						|
		err = gtserror.New("error importing domain permissions: 0 entries provided")
 | 
						|
		return nil, gtserror.NewErrorBadRequest(err, err.Error())
 | 
						|
	}
 | 
						|
 | 
						|
	// Try to process each domain permission, differentiating
 | 
						|
	// between successes and errors so that the caller can
 | 
						|
	// try failed imports again if desired.
 | 
						|
	multiStatusEntries := make([]apimodel.MultiStatusEntry, 0, count)
 | 
						|
 | 
						|
	for _, domainPerm := range domainPerms {
 | 
						|
		var (
 | 
						|
			domain         = domainPerm.Domain.Domain
 | 
						|
			obfuscate      = domainPerm.Obfuscate
 | 
						|
			publicComment  = domainPerm.PublicComment
 | 
						|
			privateComment = domainPerm.PrivateComment
 | 
						|
			subscriptionID = "" // No sub ID for imports.
 | 
						|
			errWithCode    gtserror.WithCode
 | 
						|
		)
 | 
						|
 | 
						|
		domainPerm, _, errWithCode = p.DomainPermissionCreate(
 | 
						|
			ctx,
 | 
						|
			permissionType,
 | 
						|
			account,
 | 
						|
			domain,
 | 
						|
			obfuscate,
 | 
						|
			publicComment,
 | 
						|
			privateComment,
 | 
						|
			subscriptionID,
 | 
						|
		)
 | 
						|
 | 
						|
		var entry *apimodel.MultiStatusEntry
 | 
						|
 | 
						|
		if errWithCode != nil {
 | 
						|
			entry = &apimodel.MultiStatusEntry{
 | 
						|
				// Use the failed domain entry as the resource value.
 | 
						|
				Resource: domain,
 | 
						|
				Message:  errWithCode.Safe(),
 | 
						|
				Status:   errWithCode.Code(),
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			entry = &apimodel.MultiStatusEntry{
 | 
						|
				// Use successfully created API model domain block as the resource value.
 | 
						|
				Resource: domainPerm,
 | 
						|
				Message:  http.StatusText(http.StatusOK),
 | 
						|
				Status:   http.StatusOK,
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		multiStatusEntries = append(multiStatusEntries, *entry)
 | 
						|
	}
 | 
						|
 | 
						|
	return apimodel.NewMultiStatus(multiStatusEntries), nil
 | 
						|
}
 | 
						|
 | 
						|
// DomainPermissionsGet returns all existing domain
 | 
						|
// permissions of the requested type. If export is
 | 
						|
// true, the format will be suitable for writing out
 | 
						|
// to an export.
 | 
						|
func (p *Processor) DomainPermissionsGet(
 | 
						|
	ctx context.Context,
 | 
						|
	permissionType gtsmodel.DomainPermissionType,
 | 
						|
	account *gtsmodel.Account,
 | 
						|
	export bool,
 | 
						|
) ([]*apimodel.DomainPermission, gtserror.WithCode) {
 | 
						|
	var (
 | 
						|
		domainPerms []gtsmodel.DomainPermission
 | 
						|
		err         error
 | 
						|
	)
 | 
						|
 | 
						|
	switch permissionType {
 | 
						|
	case gtsmodel.DomainPermissionBlock:
 | 
						|
		var blocks []*gtsmodel.DomainBlock
 | 
						|
 | 
						|
		blocks, err = p.state.DB.GetDomainBlocks(ctx)
 | 
						|
		if err != nil {
 | 
						|
			break
 | 
						|
		}
 | 
						|
 | 
						|
		for _, block := range blocks {
 | 
						|
			domainPerms = append(domainPerms, block)
 | 
						|
		}
 | 
						|
 | 
						|
	case gtsmodel.DomainPermissionAllow:
 | 
						|
		var allows []*gtsmodel.DomainAllow
 | 
						|
 | 
						|
		allows, err = p.state.DB.GetDomainAllows(ctx)
 | 
						|
		if err != nil {
 | 
						|
			break
 | 
						|
		}
 | 
						|
 | 
						|
		for _, allow := range allows {
 | 
						|
			domainPerms = append(domainPerms, allow)
 | 
						|
		}
 | 
						|
 | 
						|
	default:
 | 
						|
		err = errors.New("unrecognized permission type")
 | 
						|
	}
 | 
						|
 | 
						|
	if err != nil {
 | 
						|
		err := gtserror.Newf("error getting %ss: %w", permissionType.String(), err)
 | 
						|
		return nil, gtserror.NewErrorInternalError(err)
 | 
						|
	}
 | 
						|
 | 
						|
	apiDomainPerms := make([]*apimodel.DomainPermission, len(domainPerms))
 | 
						|
	for i, domainPerm := range domainPerms {
 | 
						|
		apiDomainBlock, errWithCode := p.apiDomainPerm(ctx, domainPerm, export)
 | 
						|
		if errWithCode != nil {
 | 
						|
			return nil, errWithCode
 | 
						|
		}
 | 
						|
 | 
						|
		apiDomainPerms[i] = apiDomainBlock
 | 
						|
	}
 | 
						|
 | 
						|
	return apiDomainPerms, nil
 | 
						|
}
 | 
						|
 | 
						|
// DomainPermissionGet returns one domain
 | 
						|
// permission with the given id and type.
 | 
						|
//
 | 
						|
// If export is true, the format will be
 | 
						|
// suitable for writing out to an export.
 | 
						|
func (p *Processor) DomainPermissionGet(
 | 
						|
	ctx context.Context,
 | 
						|
	permissionType gtsmodel.DomainPermissionType,
 | 
						|
	id string,
 | 
						|
	export bool,
 | 
						|
) (*apimodel.DomainPermission, gtserror.WithCode) {
 | 
						|
	var (
 | 
						|
		domainPerm gtsmodel.DomainPermission
 | 
						|
		err        error
 | 
						|
	)
 | 
						|
 | 
						|
	switch permissionType {
 | 
						|
	case gtsmodel.DomainPermissionBlock:
 | 
						|
		domainPerm, err = p.state.DB.GetDomainBlockByID(ctx, id)
 | 
						|
	case gtsmodel.DomainPermissionAllow:
 | 
						|
		domainPerm, err = p.state.DB.GetDomainAllowByID(ctx, id)
 | 
						|
	default:
 | 
						|
		err = gtserror.New("unrecognized permission type")
 | 
						|
	}
 | 
						|
 | 
						|
	if err != nil {
 | 
						|
		if errors.Is(err, db.ErrNoEntries) {
 | 
						|
			err = fmt.Errorf("no domain %s exists with id %s", permissionType.String(), id)
 | 
						|
			return nil, gtserror.NewErrorNotFound(err, err.Error())
 | 
						|
		}
 | 
						|
 | 
						|
		err = gtserror.Newf("error getting domain %s with id %s: %w", permissionType.String(), id, err)
 | 
						|
		return nil, gtserror.NewErrorInternalError(err)
 | 
						|
	}
 | 
						|
 | 
						|
	return p.apiDomainPerm(ctx, domainPerm, export)
 | 
						|
}
 |