2023-03-12 15:00:57 +00: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/>.
2023-01-25 10:12:27 +00:00
2021-06-17 16:02:33 +00:00
package typeutils
import (
2021-08-25 13:34:33 +00:00
"context"
2024-09-23 12:42:19 +00:00
"errors"
2021-06-17 16:02:33 +00:00
"fmt"
2024-07-17 15:26:33 +00:00
"math"
2023-01-25 10:12:27 +00:00
"net/url"
2023-11-10 18:29:26 +00:00
"path"
"slices"
"strconv"
"strings"
2021-06-17 16:02:33 +00:00
2024-09-16 12:00:23 +00:00
"github.com/k3a/html2text"
2023-11-10 18:29:26 +00:00
apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
"github.com/superseriousbusiness/gotosocial/internal/config"
2024-09-23 12:42:19 +00:00
"github.com/superseriousbusiness/gotosocial/internal/db"
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
2021-06-17 16:02:33 +00:00
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
2023-11-21 14:13:30 +00:00
"github.com/superseriousbusiness/gotosocial/internal/language"
"github.com/superseriousbusiness/gotosocial/internal/log"
2023-01-25 10:12:27 +00:00
"github.com/superseriousbusiness/gotosocial/internal/regexes"
2023-11-10 18:29:26 +00:00
"github.com/superseriousbusiness/gotosocial/internal/text"
2021-06-17 16:02:33 +00:00
)
2024-07-17 15:26:33 +00:00
// toAPISize converts a set of media dimensions
// to mastodon API compatible size string.
func toAPISize ( width , height int ) string {
return strconv . Itoa ( width ) +
"x" +
strconv . Itoa ( height )
}
// toAPIFrameRate converts a media framerate ptr
// to mastodon API compatible framerate string.
func toAPIFrameRate ( framerate * float32 ) string {
if framerate == nil {
return ""
}
// The masto api expects this as a string in
// the format `integer/1`, so 30fps is `30/1`.
round := math . Round ( float64 ( * framerate ) )
return strconv . Itoa ( int ( round ) ) + "/1"
}
2023-01-25 10:12:27 +00:00
type statusInteractions struct {
2024-04-17 10:41:40 +00:00
Favourited bool
2023-01-25 10:12:27 +00:00
Muted bool
Bookmarked bool
Reblogged bool
2023-02-25 12:16:30 +00:00
Pinned bool
2023-01-25 10:12:27 +00:00
}
2023-09-23 16:44:11 +00:00
func ( c * Converter ) interactionsWithStatusForAccount ( ctx context . Context , s * gtsmodel . Status , requestingAccount * gtsmodel . Account ) ( * statusInteractions , error ) {
2021-06-17 16:02:33 +00:00
si := & statusInteractions { }
if requestingAccount != nil {
2023-09-23 16:44:11 +00:00
faved , err := c . state . DB . IsStatusFavedBy ( ctx , s . ID , requestingAccount . ID )
2021-06-17 16:02:33 +00:00
if err != nil {
return nil , fmt . Errorf ( "error checking if requesting account has faved status: %s" , err )
}
2024-04-17 10:41:40 +00:00
si . Favourited = faved
2021-06-17 16:02:33 +00:00
2023-09-23 16:44:11 +00:00
reblogged , err := c . state . DB . IsStatusBoostedBy ( ctx , s . ID , requestingAccount . ID )
2021-06-17 16:02:33 +00:00
if err != nil {
return nil , fmt . Errorf ( "error checking if requesting account has reblogged status: %s" , err )
}
si . Reblogged = reblogged
2023-10-25 14:04:53 +00:00
muted , err := c . state . DB . IsThreadMutedByAccount ( ctx , s . ThreadID , requestingAccount . ID )
2021-06-17 16:02:33 +00:00
if err != nil {
return nil , fmt . Errorf ( "error checking if requesting account has muted status: %s" , err )
}
si . Muted = muted
2024-06-06 10:44:43 +00:00
bookmarked , err := c . state . DB . IsStatusBookmarkedBy ( ctx , requestingAccount . ID , s . ID )
2021-06-17 16:02:33 +00:00
if err != nil {
return nil , fmt . Errorf ( "error checking if requesting account has bookmarked status: %s" , err )
}
si . Bookmarked = bookmarked
2023-02-25 12:16:30 +00:00
// The only time 'pinned' should be true is if the
// requesting account is looking at its OWN status.
if s . AccountID == requestingAccount . ID {
si . Pinned = ! s . PinnedAt . IsZero ( )
}
2021-06-17 16:02:33 +00:00
}
return si , nil
}
2023-01-25 10:12:27 +00:00
func misskeyReportInlineURLs ( content string ) [ ] * url . URL {
m := regexes . MisskeyReportNotes . FindAllStringSubmatch ( content , - 1 )
urls := make ( [ ] * url . URL , 0 , len ( m ) )
for _ , sm := range m {
url , err := url . Parse ( sm [ 1 ] )
if err == nil && url != nil {
urls = append ( urls , url )
}
}
return urls
2021-06-17 16:02:33 +00:00
}
2023-06-17 15:49:11 +00:00
2024-07-17 15:26:33 +00:00
// placeholderAttachments separates any attachments with missing local URL
2024-01-09 09:41:32 +00:00
// out of the given slice, and returns a piece of text containing links to
2023-11-10 18:29:26 +00:00
// those attachments, as well as the slice of remaining "known" attachments.
// If there are no unknown-type attachments in the provided slice, an empty
// string and the original slice will be returned.
//
2024-01-09 09:41:32 +00:00
// Returned text will be run through the sanitizer before being returned, to
// ensure that malicious links don't cause issues.
2023-11-10 18:29:26 +00:00
//
// Example:
//
2024-01-09 09:41:32 +00:00
// <hr>
2024-07-17 15:26:33 +00:00
// <p><i lang="en">ℹ ️ Note from your.instance.com: 2 attachment(s) in this status were not downloaded. Treat the following external link(s) with care:</i></p>
2024-01-09 09:41:32 +00:00
// <ul>
// <li><a href="http://example.org/fileserver/01HE7Y659ZWZ02JM4AWYJZ176Q/attachment/original/01HE7ZGJYTSYMXF927GF9353KR.svg" rel="nofollow noreferrer noopener" target="_blank">01HE7ZGJYTSYMXF927GF9353KR.svg</a> [SVG line art of a sloth, public domain]</li>
// <li><a href="http://example.org/fileserver/01HE7Y659ZWZ02JM4AWYJZ176Q/attachment/original/01HE892Y8ZS68TQCNPX7J888P3.mp3" rel="nofollow noreferrer noopener" target="_blank">01HE892Y8ZS68TQCNPX7J888P3.mp3</a> [Jolly salsa song, public domain.]</li>
// </ul>
2024-07-17 15:26:33 +00:00
func placeholderAttachments ( arr [ ] * apimodel . Attachment ) ( string , [ ] * apimodel . Attachment ) {
// Extract non-locally stored attachments into a
// separate slice, deleting them from input slice.
var nonLocal [ ] * apimodel . Attachment
2023-12-09 15:54:38 +00:00
arr = slices . DeleteFunc ( arr , func ( elem * apimodel . Attachment ) bool {
2024-07-17 15:26:33 +00:00
if elem . URL == nil {
nonLocal = append ( nonLocal , elem )
return true
2023-11-10 18:29:26 +00:00
}
2024-07-17 15:26:33 +00:00
return false
2023-11-10 18:29:26 +00:00
} )
2024-07-17 15:26:33 +00:00
if len ( nonLocal ) == 0 {
// No non-locally
// stored media.
2023-11-10 18:29:26 +00:00
return "" , arr
}
2024-07-17 15:26:33 +00:00
var note strings . Builder
note . WriteString ( ` <hr> ` )
2024-07-21 09:30:22 +00:00
note . WriteString ( ` <p><i lang="en">ℹ ️ Note from ` )
2024-07-17 15:26:33 +00:00
note . WriteString ( config . GetHost ( ) )
note . WriteString ( ` : ` )
note . WriteString ( strconv . Itoa ( len ( nonLocal ) ) )
if len ( nonLocal ) > 1 {
// Use plural word form.
note . WriteString ( ` attachments in this status were not downloaded. ` +
` Treat the following external links with care: ` )
2023-11-10 18:29:26 +00:00
} else {
2024-07-17 15:26:33 +00:00
// Use singular word form.
note . WriteString ( ` attachment in this status was not downloaded. ` +
` Treat the following external link with care: ` )
2023-11-10 18:29:26 +00:00
}
2024-07-17 15:26:33 +00:00
note . WriteString ( ` </i></p><ul> ` )
for _ , a := range nonLocal {
note . WriteString ( ` <li> ` )
note . WriteString ( ` <a href=" ` )
note . WriteString ( * a . RemoteURL )
note . WriteString ( ` "> ` )
note . WriteString ( path . Base ( * a . RemoteURL ) )
note . WriteString ( ` </a> ` )
2023-11-10 18:29:26 +00:00
if d := a . Description ; d != nil && * d != "" {
2024-07-17 15:26:33 +00:00
note . WriteString ( ` [ ` )
note . WriteString ( * d )
note . WriteString ( ` ] ` )
2023-11-10 18:29:26 +00:00
}
2024-07-17 15:26:33 +00:00
note . WriteString ( ` </li> ` )
2023-11-10 18:29:26 +00:00
}
2024-01-09 09:41:32 +00:00
note . WriteString ( ` </ul> ` )
2023-11-10 18:29:26 +00:00
2024-01-09 09:41:32 +00:00
return text . SanitizeToHTML ( note . String ( ) ) , arr
2023-11-10 18:29:26 +00:00
}
2023-11-21 14:13:30 +00:00
2024-09-23 12:42:19 +00:00
func ( c * Converter ) pendingReplyNote (
ctx context . Context ,
s * gtsmodel . Status ,
) ( string , error ) {
intReq , err := c . state . DB . GetInteractionRequestByInteractionURI ( ctx , s . URI )
if err != nil && ! errors . Is ( err , db . ErrNoEntries ) {
// Something's gone wrong.
err := gtserror . Newf ( "db error getting interaction request for %s: %w" , s . URI , err )
return "" , err
}
// No interaction request present
// for this status. Race condition?
if intReq == nil {
return "" , nil
}
var (
proto = config . GetProtocol ( )
host = config . GetHost ( )
// Build the settings panel URL at which the user
// can view + approve/reject the interaction request.
//
// Eg., https://example.org/settings/user/interaction_requests/01J5QVXCCEATJYSXM9H6MZT4JR
settingsURL = proto + "://" + host + "/settings/user/interaction_requests/" + intReq . ID
)
var note strings . Builder
note . WriteString ( ` <hr> ` )
note . WriteString ( ` <p><i lang="en">ℹ ️ Note from ` + host + ` : ` )
note . WriteString ( ` This reply is pending your approval. You can quickly accept it by liking, boosting or replying to it. You can also accept or reject it at the following link: ` )
note . WriteString ( ` <a href=" ` + settingsURL + ` " ` )
note . WriteString ( ` rel="noreferrer noopener" target="_blank"> ` )
note . WriteString ( settingsURL )
note . WriteString ( ` </a>. ` )
note . WriteString ( ` </i></p> ` )
return text . SanitizeToHTML ( note . String ( ) ) , nil
}
2023-11-21 14:13:30 +00:00
// ContentToContentLanguage tries to
// extract a content string and language
// tag string from the given intermediary
// content.
//
// Either/both of the returned strings may
// be empty, depending on how things go.
func ContentToContentLanguage (
ctx context . Context ,
content gtsmodel . Content ,
) (
string , // content
string , // language
) {
var (
contentStr string
langTagStr string
)
switch contentMap := content . ContentMap ; {
// Simplest case: no `contentMap`.
// Return `content`, even if empty.
case contentMap == nil :
return content . Content , ""
// `content` and `contentMap` set.
// Try to infer "primary" language.
case content . Content != "" :
// Assume `content` is intended
// primary content, and look for
// corresponding language tag.
contentStr = content . Content
for t , c := range contentMap {
if contentStr == c {
langTagStr = t
break
}
}
// `content` not set; `contentMap`
// is set with only one value.
// This must be the "primary" lang.
case len ( contentMap ) == 1 :
// Use an empty loop to
// get the values we want.
// nolint:revive
for langTagStr , contentStr = range contentMap {
}
// Only `contentMap` is set, with more
// than one value. Map order is not
// guaranteed so we can't know the
// "primary" language.
//
// Try to select content using our
// instance's configured languages.
//
// In case of no hits, just take the
// first tag and content in the map.
default :
instanceLangs := config . GetInstanceLanguages ( )
for _ , langTagStr = range instanceLangs . TagStrs ( ) {
if contentStr = contentMap [ langTagStr ] ; contentStr != "" {
// Hit!
break
}
}
// If nothing found, just take
// the first entry we can get by
// breaking after the first iter.
if contentStr == "" {
for langTagStr , contentStr = range contentMap {
break
}
}
}
if langTagStr != "" {
// Found a lang tag for this content,
// make sure it's valid / parseable.
lang , err := language . Parse ( langTagStr )
if err != nil {
log . Warnf (
ctx ,
"could not parse %s as BCP47 language tag in status contentMap: %v" ,
langTagStr , err ,
)
} else {
// Inferred the language!
// Use normalized version.
langTagStr = lang . TagStr
}
}
return contentStr , langTagStr
}
2024-09-16 12:00:23 +00:00
// filterableFields returns text fields from
// a status that we might want to filter on:
//
// - content warning
// - content (converted to plaintext from HTML)
// - media descriptions
// - poll options
//
// Each field should be filtered separately.
// This avoids scenarios where false-positive
// multiple-word matches can be made by matching
// the last word of one field + the first word
// of the next field together.
func filterableFields ( s * gtsmodel . Status ) [ ] string {
// Estimate length of fields.
fieldCount := 2 + len ( s . Attachments )
if s . Poll != nil {
fieldCount += len ( s . Poll . Options )
}
fields := make ( [ ] string , 0 , fieldCount )
// Content warning / title.
if s . ContentWarning != "" {
fields = append ( fields , s . ContentWarning )
}
// Status content. Though we have raw text
// available for statuses created on our
// instance, use the html2text version to
// remove markdown-formatting characters
// and ensure more consistent filtering.
if s . Content != "" {
text := html2text . HTML2TextWithOptions (
s . Content ,
html2text . WithLinksInnerText ( ) ,
html2text . WithUnixLineBreaks ( ) ,
)
if text != "" {
fields = append ( fields , text )
}
}
// Media descriptions.
for _ , attachment := range s . Attachments {
if attachment . Description != "" {
fields = append ( fields , attachment . Description )
}
}
// Poll options.
if s . Poll != nil {
for _ , opt := range s . Poll . Options {
if opt != "" {
fields = append ( fields , opt )
}
}
}
return fields
}