| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | /* | 
					
						
							|  |  |  |    GoToSocial | 
					
						
							| 
									
										
										
										
											2021-12-20 18:42:19 +01:00
										 |  |  |    Copyright (C) 2021-2022 GoToSocial Authors admin@gotosocial.org | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |    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-08-25 15:34:33 +02:00
										 |  |  | package bundb | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 	"database/sql" | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 	"net/url" | 
					
						
							| 
									
										
										
										
											2022-05-02 12:53:46 +02:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/cache" | 
					
						
							| 
									
										
										
										
											2022-05-23 11:46:50 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/config" | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/db" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | 	"golang.org/x/net/idna" | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type domainDB struct { | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 	conn  *DBConn | 
					
						
							|  |  |  | 	cache *cache.DomainBlockCache | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | // normalizeDomain converts the given domain to lowercase | 
					
						
							|  |  |  | // then to punycode (for international domain names). | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Returns the resulting domain or an error if the | 
					
						
							|  |  |  | // punycode conversion fails. | 
					
						
							|  |  |  | func normalizeDomain(domain string) (out string, err error) { | 
					
						
							|  |  |  | 	out = strings.ToLower(domain) | 
					
						
							|  |  |  | 	out, err = idna.ToASCII(out) | 
					
						
							|  |  |  | 	return out, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | func (d *domainDB) CreateDomainBlock(ctx context.Context, block gtsmodel.DomainBlock) db.Error { | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | 	domain, err := normalizeDomain(block.Domain) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	block.Domain = domain | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Attempt to insert new domain block | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | 	if _, err := d.conn.NewInsert(). | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 		Model(&block). | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | 		Exec(ctx, &block); err != nil { | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 		return d.conn.ProcessError(err) | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 	// Cache this domain block | 
					
						
							|  |  |  | 	d.cache.Put(block.Domain, &block) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *domainDB) GetDomainBlock(ctx context.Context, domain string) (*gtsmodel.DomainBlock, db.Error) { | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | 	var err error | 
					
						
							|  |  |  | 	domain, err = normalizeDomain(domain) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Check for easy case, domain referencing *us* | 
					
						
							|  |  |  | 	if domain == "" || domain == config.GetAccountDomain() { | 
					
						
							|  |  |  | 		return nil, db.ErrNoEntries | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Check for already cached rblock | 
					
						
							|  |  |  | 	if block, ok := d.cache.GetByDomain(domain); ok { | 
					
						
							|  |  |  | 		// A 'nil' return value is a sentinel value for no block | 
					
						
							|  |  |  | 		if block == nil { | 
					
						
							|  |  |  | 			return nil, db.ErrNoEntries | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Else, this block exists | 
					
						
							|  |  |  | 		return block, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	block := >smodel.DomainBlock{} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 	q := d.conn. | 
					
						
							|  |  |  | 		NewSelect(). | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 		Model(block). | 
					
						
							| 
									
										
										
										
											2022-05-02 12:53:46 +02:00
										 |  |  | 		Where("domain = ?", domain). | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | 		Limit(1) | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 	// Query database for domain block | 
					
						
							|  |  |  | 	switch err := q.Scan(ctx); err { | 
					
						
							|  |  |  | 	// No error, block found | 
					
						
							|  |  |  | 	case nil: | 
					
						
							|  |  |  | 		d.cache.Put(domain, block) | 
					
						
							|  |  |  | 		return block, nil | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// No error, simply not found | 
					
						
							|  |  |  | 	case sql.ErrNoRows: | 
					
						
							|  |  |  | 		d.cache.Put(domain, nil) | 
					
						
							|  |  |  | 		return nil, db.ErrNoEntries | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Any other db error | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return nil, d.conn.ProcessError(err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | func (d *domainDB) DeleteDomainBlock(ctx context.Context, domain string) db.Error { | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | 	var err error | 
					
						
							|  |  |  | 	domain, err = normalizeDomain(domain) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 	// Attempt to delete domain block | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | 	if _, err := d.conn.NewDelete(). | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 		Model((*gtsmodel.DomainBlock)(nil)). | 
					
						
							|  |  |  | 		Where("domain = ?", domain). | 
					
						
							| 
									
										
										
										
											2022-10-01 17:48:38 +02:00
										 |  |  | 		Exec(ctx); err != nil { | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 		return d.conn.ProcessError(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Clear domain from cache | 
					
						
							|  |  |  | 	d.cache.InvalidateByDomain(domain) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *domainDB) IsDomainBlocked(ctx context.Context, domain string) (bool, db.Error) { | 
					
						
							|  |  |  | 	block, err := d.GetDomainBlock(ctx, domain) | 
					
						
							|  |  |  | 	if err == nil || err == db.ErrNoEntries { | 
					
						
							|  |  |  | 		return (block != nil), nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *domainDB) AreDomainsBlocked(ctx context.Context, domains []string) (bool, db.Error) { | 
					
						
							|  |  |  | 	for _, domain := range domains { | 
					
						
							|  |  |  | 		if blocked, err := d.IsDomainBlocked(ctx, domain); err != nil { | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 			return false, err | 
					
						
							|  |  |  | 		} else if blocked { | 
					
						
							|  |  |  | 			return blocked, nil | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | func (d *domainDB) IsURIBlocked(ctx context.Context, uri *url.URL) (bool, db.Error) { | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 	return d.IsDomainBlocked(ctx, uri.Hostname()) | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-25 15:34:33 +02:00
										 |  |  | func (d *domainDB) AreURIsBlocked(ctx context.Context, uris []*url.URL) (bool, db.Error) { | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 	for _, uri := range uris { | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 		if blocked, err := d.IsDomainBlocked(ctx, uri.Hostname()); err != nil { | 
					
						
							|  |  |  | 			return false, err | 
					
						
							|  |  |  | 		} else if blocked { | 
					
						
							|  |  |  | 			return blocked, nil | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-09-02 11:17:46 +01:00
										 |  |  | 	return false, nil | 
					
						
							| 
									
										
										
										
											2021-08-20 12:26:56 +02:00
										 |  |  | } |