mirror of
https://codeberg.org/superseriousbusiness/gotosocial.git
synced 2024-12-29 20:38:17 +03:00
23fc70f4e6
* add support for extracting Updated field from Statusable implementers * add support for status edits in the database, and update status dereferencer to handle them * remove unused AdditionalInfo{}.CreatedAt * remove unused AdditionalEmojiInfo{}.CreatedAt * update new mention creation to use status.UpdatedAt * remove mention.UpdatedAt, fixes related to NewULIDFromTime() change * add migration to remove Mention{}.UpdatedAt field * add migration to add the StatusEdit{} table * start adding tests, add delete function for status edits * add more of status edit migrations, fill in more of the necessary edit delete functionality * remove unused function * allow generating gotosocial compatible ulid via CLI with `go run ./cmd/gen-ulid` * add StatusEdit{} test models * fix new statusedits sql * use model instead of table name * actually remove the Mention.UpdatedAt field... * fix tests now new models are added, add more status edit DB tests * fix panic wording * add test for deleting status edits * don't automatically set `updated_at` field on updated statuses * flesh out more of the dereferencer status edit tests, ensure updated at field set on outgoing AS statuses * remove media_attachments.updated_at column * fix up more tests, further complete the dereferencer status edit tests * update more status serialization tests not expecting 'updated' AS property * gah!! json serialization tests!! * undo some gtscontext wrapping changes * more serialization test fixing 🥲 * more test fixing, ensure the edit.status_id field is actually set 🤦 * fix status edit test * grrr linter * add edited_at field to apimodel status * remove the choice of paging on the timeline public filtered test (otherwise it needs updating every time you add statuses ...) * ensure that status.updated_at always fits chronologically * fix more serialization tests ... * add more code comments * fix envparsing * update swagger file * properly handle media description changes during status edits * slight formatting tweak * code comment
1341 lines
39 KiB
Go
1341 lines
39 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 dereferencing
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"net/http"
|
|
"net/url"
|
|
"slices"
|
|
"time"
|
|
|
|
"github.com/superseriousbusiness/gotosocial/internal/ap"
|
|
"github.com/superseriousbusiness/gotosocial/internal/config"
|
|
"github.com/superseriousbusiness/gotosocial/internal/db"
|
|
"github.com/superseriousbusiness/gotosocial/internal/gtscontext"
|
|
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
|
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
|
"github.com/superseriousbusiness/gotosocial/internal/id"
|
|
"github.com/superseriousbusiness/gotosocial/internal/log"
|
|
"github.com/superseriousbusiness/gotosocial/internal/media"
|
|
"github.com/superseriousbusiness/gotosocial/internal/util"
|
|
)
|
|
|
|
// statusFresh returns true if the given status is still
|
|
// considered "fresh" according to the desired freshness
|
|
// window (falls back to default status freshness if nil).
|
|
//
|
|
// Local statuses will always be considered fresh,
|
|
// because there's no remote state that may have changed.
|
|
//
|
|
// Return value of false indicates that the status
|
|
// is not fresh and should be refreshed from remote.
|
|
func statusFresh(
|
|
status *gtsmodel.Status,
|
|
window *FreshnessWindow,
|
|
) bool {
|
|
// Take default if no
|
|
// freshness window preferred.
|
|
if window == nil {
|
|
window = DefaultStatusFreshness
|
|
}
|
|
|
|
if status.IsLocal() {
|
|
// Can't refresh
|
|
// local statuses.
|
|
return true
|
|
}
|
|
|
|
// Moment when the status is
|
|
// considered stale according to
|
|
// desired freshness window.
|
|
staleAt := status.FetchedAt.Add(
|
|
time.Duration(*window),
|
|
)
|
|
|
|
// It's still fresh if the time now
|
|
// is not past the point of staleness.
|
|
return !time.Now().After(staleAt)
|
|
}
|
|
|
|
// GetStatusByURI will attempt to fetch a status by its URI, first checking the database. In the case of a newly-met remote model, or a remote model whose 'last_fetched' date
|
|
// is beyond a certain interval, the status will be dereferenced. In the case of dereferencing, some low-priority status information may be enqueued for asynchronous fetching,
|
|
// e.g. dereferencing the status thread. An ActivityPub object indicates the status was dereferenced.
|
|
func (d *Dereferencer) GetStatusByURI(ctx context.Context, requestUser string, uri *url.URL) (*gtsmodel.Status, ap.Statusable, error) {
|
|
|
|
// Fetch and dereference / update status if necessary.
|
|
status, statusable, isNew, err := d.getStatusByURI(ctx,
|
|
requestUser,
|
|
uri,
|
|
)
|
|
|
|
if err != nil {
|
|
if status == nil {
|
|
// err with no existing
|
|
// status for fallback.
|
|
return nil, nil, err
|
|
}
|
|
|
|
log.Errorf(ctx, "error updating status %s: %v", uri, err)
|
|
|
|
} else if statusable != nil {
|
|
|
|
// Deref parents + children.
|
|
d.dereferenceThread(ctx,
|
|
requestUser,
|
|
uri,
|
|
status,
|
|
statusable,
|
|
isNew,
|
|
)
|
|
}
|
|
|
|
return status, statusable, nil
|
|
}
|
|
|
|
// getStatusByURI is a package internal form of .GetStatusByURI() that doesn't dereference thread on update, and may return an existing status with error on failed re-fetch.
|
|
func (d *Dereferencer) getStatusByURI(ctx context.Context, requestUser string, uri *url.URL) (*gtsmodel.Status, ap.Statusable, bool, error) {
|
|
var (
|
|
status *gtsmodel.Status
|
|
uriStr = uri.String()
|
|
err error
|
|
)
|
|
|
|
// Search the database for existing by URI.
|
|
status, err = d.state.DB.GetStatusByURI(
|
|
|
|
// request a barebones object, it may be in the
|
|
// db but with related models not yet dereferenced.
|
|
gtscontext.SetBarebones(ctx),
|
|
uriStr,
|
|
)
|
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
|
return nil, nil, false, gtserror.Newf("error checking database for status %s by uri: %w", uriStr, err)
|
|
}
|
|
|
|
if status == nil {
|
|
// Else, search database for existing by URL.
|
|
status, err = d.state.DB.GetStatusByURL(
|
|
gtscontext.SetBarebones(ctx),
|
|
uriStr,
|
|
)
|
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
|
return nil, nil, false, gtserror.Newf("error checking database for status %s by url: %w", uriStr, err)
|
|
}
|
|
}
|
|
|
|
if status == nil {
|
|
// Ensure not a failed search for a local
|
|
// status, if so we know it doesn't exist.
|
|
if uri.Host == config.GetHost() ||
|
|
uri.Host == config.GetAccountDomain() {
|
|
return nil, nil, false, gtserror.SetUnretrievable(err)
|
|
}
|
|
|
|
// Create and pass-through a new bare-bones model for deref.
|
|
return d.enrichStatusSafely(ctx, requestUser, uri, >smodel.Status{
|
|
Local: util.Ptr(false),
|
|
URI: uriStr,
|
|
}, nil)
|
|
}
|
|
|
|
if statusFresh(status, DefaultStatusFreshness) {
|
|
// This is an existing status that is up-to-date,
|
|
// before returning ensure it is fully populated.
|
|
if err := d.state.DB.PopulateStatus(ctx, status); err != nil {
|
|
log.Errorf(ctx, "error populating existing status: %v", err)
|
|
}
|
|
|
|
return status, nil, false, nil
|
|
}
|
|
|
|
// Try to deref and update existing.
|
|
return d.enrichStatusSafely(ctx,
|
|
requestUser,
|
|
uri,
|
|
status,
|
|
nil,
|
|
)
|
|
}
|
|
|
|
// RefreshStatus is functionally equivalent to GetStatusByURI(), except that it requires a pre
|
|
// populated status model (with AT LEAST uri set), and ALL thread dereferencing is asynchronous.
|
|
func (d *Dereferencer) RefreshStatus(
|
|
ctx context.Context,
|
|
requestUser string,
|
|
status *gtsmodel.Status,
|
|
statusable ap.Statusable,
|
|
window *FreshnessWindow,
|
|
) (*gtsmodel.Status, ap.Statusable, error) {
|
|
// If no incoming data is provided,
|
|
// check whether status needs update.
|
|
if statusable == nil &&
|
|
statusFresh(status, window) {
|
|
return status, nil, nil
|
|
}
|
|
|
|
// Parse the URI from status.
|
|
uri, err := url.Parse(status.URI)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("invalid status uri %q: %w", status.URI, err)
|
|
}
|
|
|
|
// Try to update + dereference the passed status model.
|
|
latest, statusable, isNew, err := d.enrichStatusSafely(ctx,
|
|
requestUser,
|
|
uri,
|
|
status,
|
|
statusable,
|
|
)
|
|
|
|
if statusable != nil {
|
|
// Deref parents + children.
|
|
d.dereferenceThread(ctx,
|
|
requestUser,
|
|
uri,
|
|
latest,
|
|
statusable,
|
|
isNew,
|
|
)
|
|
}
|
|
|
|
return latest, statusable, err
|
|
}
|
|
|
|
// RefreshStatusAsync is functionally equivalent to RefreshStatus(), except that ALL
|
|
// dereferencing is queued for asynchronous processing, (both thread AND status).
|
|
func (d *Dereferencer) RefreshStatusAsync(
|
|
ctx context.Context,
|
|
requestUser string,
|
|
status *gtsmodel.Status,
|
|
statusable ap.Statusable,
|
|
window *FreshnessWindow,
|
|
) {
|
|
// If no incoming data is provided,
|
|
// check whether status needs update.
|
|
if statusable == nil &&
|
|
statusFresh(status, window) {
|
|
return
|
|
}
|
|
|
|
// Parse the URI from status.
|
|
uri, err := url.Parse(status.URI)
|
|
if err != nil {
|
|
log.Errorf(ctx, "invalid status uri %q: %v", status.URI, err)
|
|
return
|
|
}
|
|
|
|
// Enqueue a worker function to re-fetch this status entirely async.
|
|
d.state.Workers.Dereference.Queue.Push(func(ctx context.Context) {
|
|
latest, statusable, _, err := d.enrichStatusSafely(ctx,
|
|
requestUser,
|
|
uri,
|
|
status,
|
|
statusable,
|
|
)
|
|
if err != nil {
|
|
log.Errorf(ctx, "error enriching remote status: %v", err)
|
|
return
|
|
}
|
|
if statusable != nil {
|
|
if err := d.DereferenceStatusAncestors(ctx, requestUser, latest); err != nil {
|
|
log.Error(ctx, err)
|
|
}
|
|
if err := d.DereferenceStatusDescendants(ctx, requestUser, uri, statusable); err != nil {
|
|
log.Error(ctx, err)
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
// enrichStatusSafely wraps enrichStatus() to perform it within
|
|
// a State{}.FedLocks mutexmap, which protects it within per-URI
|
|
// mutex locks. This also handles necessary delete of now-deleted
|
|
// statuses, and updating fetched_at on returned HTTP errors.
|
|
func (d *Dereferencer) enrichStatusSafely(
|
|
ctx context.Context,
|
|
requestUser string,
|
|
uri *url.URL,
|
|
status *gtsmodel.Status,
|
|
statusable ap.Statusable,
|
|
) (*gtsmodel.Status, ap.Statusable, bool, error) {
|
|
uriStr := status.URI
|
|
|
|
var isNew bool
|
|
|
|
// Check if this is a new status (to us).
|
|
if isNew = (status.ID == ""); !isNew {
|
|
|
|
// This is an existing status, first try to populate it. This
|
|
// is required by the checks below for existing tags, media etc.
|
|
if err := d.state.DB.PopulateStatus(ctx, status); err != nil {
|
|
log.Errorf(ctx, "error populating existing status %s: %v", uriStr, err)
|
|
}
|
|
}
|
|
|
|
// Acquire per-URI deref lock, wraping unlock
|
|
// to safely defer in case of panic, while still
|
|
// performing more granular unlocks when needed.
|
|
unlock := d.state.FedLocks.Lock(uriStr)
|
|
unlock = util.DoOnce(unlock)
|
|
defer unlock()
|
|
|
|
// Perform status enrichment with passed vars.
|
|
latest, statusable, err := d.enrichStatus(ctx,
|
|
requestUser,
|
|
uri,
|
|
status,
|
|
statusable,
|
|
isNew,
|
|
)
|
|
|
|
// Check for a returned HTTP code via error.
|
|
switch code := gtserror.StatusCode(err); {
|
|
|
|
// Gone (410) definitely indicates deletion.
|
|
// Remove status if it was an existing one.
|
|
case code == http.StatusGone && !isNew:
|
|
if err := d.state.DB.DeleteStatusByID(ctx, status.ID); err != nil {
|
|
log.Error(ctx, "error deleting gone status %s: %v", uriStr, err)
|
|
}
|
|
|
|
// Don't return any status.
|
|
return nil, nil, false, err
|
|
|
|
// Any other HTTP error mesg
|
|
// code, with existing status.
|
|
case code >= 400 && !isNew:
|
|
|
|
// Update fetched_at to slow re-attempts
|
|
// but don't return early. We can still
|
|
// return the model we had stored already.
|
|
status.FetchedAt = time.Now()
|
|
if err := d.state.DB.UpdateStatus(ctx, status, "fetched_at"); err != nil {
|
|
log.Error(ctx, "error updating %s fetched_at: %v", uriStr, err)
|
|
}
|
|
|
|
// See below.
|
|
fallthrough
|
|
|
|
// In case of error with an existing
|
|
// status in the database, return error
|
|
// but still return existing status.
|
|
case err != nil && !isNew:
|
|
latest = status
|
|
statusable = nil
|
|
}
|
|
|
|
// Unlock now
|
|
// we're done.
|
|
unlock()
|
|
|
|
if errors.Is(err, db.ErrAlreadyExists) {
|
|
// We leave 'isNew' set so that caller
|
|
// still dereferences parents, otherwise
|
|
// the version we pass back may not have
|
|
// these attached as inReplyTos yet (since
|
|
// those happen OUTSIDE federator lock).
|
|
//
|
|
// TODO: performance-wise, this won't be
|
|
// great. should improve this if we can!
|
|
|
|
// DATA RACE! We likely lost out to another goroutine
|
|
// in a call to db.Put(Status). Look again in DB by URI.
|
|
latest, err = d.state.DB.GetStatusByURI(ctx, status.URI)
|
|
if err != nil {
|
|
err = gtserror.Newf("error getting status %s from database after race: %w", uriStr, err)
|
|
}
|
|
}
|
|
|
|
return latest, statusable, isNew, err
|
|
}
|
|
|
|
// enrichStatus will enrich the given status, whether a new
|
|
// barebones model, or existing model from the database.
|
|
// It handles necessary dereferencing, database updates, etc.
|
|
func (d *Dereferencer) enrichStatus(
|
|
ctx context.Context,
|
|
requestUser string,
|
|
uri *url.URL,
|
|
status *gtsmodel.Status,
|
|
statusable ap.Statusable,
|
|
isNew bool,
|
|
) (
|
|
*gtsmodel.Status,
|
|
ap.Statusable,
|
|
error,
|
|
) {
|
|
// Pre-fetch a transport for requesting username, used by later dereferencing.
|
|
tsport, err := d.transportController.NewTransportForUsername(ctx, requestUser)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("couldn't create transport: %w", err)
|
|
}
|
|
|
|
// Check whether this account URI is a blocked domain / subdomain.
|
|
if blocked, err := d.state.DB.IsDomainBlocked(ctx, uri.Host); err != nil {
|
|
return nil, nil, gtserror.Newf("error checking blocked domain: %w", err)
|
|
} else if blocked {
|
|
err := gtserror.Newf("%s is blocked", uri.Host)
|
|
return nil, nil, gtserror.SetUnretrievable(err)
|
|
}
|
|
|
|
if statusable == nil {
|
|
// Dereference latest version of the status.
|
|
rsp, err := tsport.Dereference(ctx, uri)
|
|
if err != nil {
|
|
err := gtserror.Newf("error dereferencing %s: %w", uri, err)
|
|
return nil, nil, gtserror.SetUnretrievable(err)
|
|
}
|
|
|
|
// Attempt to resolve ActivityPub status from response.
|
|
statusable, err = ap.ResolveStatusable(ctx, rsp.Body)
|
|
|
|
// Tidy up now done.
|
|
_ = rsp.Body.Close()
|
|
|
|
if err != nil {
|
|
// ResolveStatusable will set gtserror.WrongType
|
|
// on the returned error, so we don't need to do it here.
|
|
err := gtserror.Newf("error resolving statusable %s: %w", uri, err)
|
|
return nil, nil, err
|
|
}
|
|
|
|
// Check whether input URI and final returned URI
|
|
// have changed (i.e. we followed some redirects).
|
|
if finalURIStr := rsp.Request.URL.String(); //
|
|
finalURIStr != uri.String() {
|
|
|
|
// NOTE: this URI check + database call is performed
|
|
// AFTER reading and closing response body, for performance.
|
|
//
|
|
// Check whether we have this status stored under *final* URI.
|
|
alreadyStatus, err := d.state.DB.GetStatusByURI(ctx, finalURIStr)
|
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
|
return nil, nil, gtserror.Newf("db error getting status after redirects: %w", err)
|
|
}
|
|
|
|
if alreadyStatus != nil {
|
|
// We had this status stored
|
|
// under discovered final URI.
|
|
//
|
|
// Proceed with this status.
|
|
status = alreadyStatus
|
|
}
|
|
|
|
// Update the input URI to
|
|
// the final determined URI
|
|
// for later URI checks.
|
|
uri = rsp.Request.URL
|
|
}
|
|
}
|
|
|
|
// Get the attributed-to account in order to fetch profile.
|
|
attributedTo, err := ap.ExtractAttributedToURI(statusable)
|
|
if err != nil {
|
|
return nil, nil, gtserror.New("attributedTo was empty")
|
|
}
|
|
|
|
// Ensure we have the author account of the status dereferenced (+ up-to-date). If this is a new status
|
|
// (i.e. status.AccountID == "") then any error here is irrecoverable. status.AccountID must ALWAYS be set.
|
|
if _, _, err := d.getAccountByURI(ctx, requestUser, attributedTo); err != nil && status.AccountID == "" {
|
|
|
|
// Note that we specifically DO NOT wrap the error, instead collapsing it as string.
|
|
// Errors fetching an account do not necessarily relate to dereferencing the status.
|
|
return nil, nil, gtserror.Newf("failed to dereference status author %s: %v", uri, err)
|
|
}
|
|
|
|
// ActivityPub model was recently dereferenced, so assume passed status
|
|
// may contain out-of-date information. Convert AP model to our GTS model.
|
|
latestStatus, err := d.converter.ASStatusToStatus(ctx, statusable)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error converting statusable to gts model for status %s: %w", uri, err)
|
|
}
|
|
|
|
// Ensure final status isn't attempting
|
|
// to claim being authored by local user.
|
|
if latestStatus.Account.IsLocal() {
|
|
return nil, nil, gtserror.Newf(
|
|
"dereferenced status %s claiming to be local",
|
|
latestStatus.URI,
|
|
)
|
|
}
|
|
|
|
// Ensure the final parsed status URI or URL matches
|
|
// the input URI we fetched (or received) it as.
|
|
matches, err := util.URIMatches(uri,
|
|
append(
|
|
ap.GetURL(statusable), // status URL(s)
|
|
ap.GetJSONLDId(statusable), // status URI
|
|
)...,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf(
|
|
"error checking dereferenced status uri %s: %w",
|
|
latestStatus.URI, err,
|
|
)
|
|
}
|
|
|
|
if !matches {
|
|
return nil, nil, gtserror.Newf(
|
|
"dereferenced status uri %s does not match %s",
|
|
latestStatus.URI, uri.String(),
|
|
)
|
|
}
|
|
|
|
if isNew {
|
|
|
|
// Generate new status ID from the provided creation date.
|
|
latestStatus.ID = id.NewULIDFromTime(latestStatus.CreatedAt)
|
|
} else {
|
|
|
|
// Reuse existing status ID.
|
|
latestStatus.ID = status.ID
|
|
}
|
|
|
|
// Set latest fetch time and carry-
|
|
// over some values from "old" status.
|
|
latestStatus.FetchedAt = time.Now()
|
|
latestStatus.Local = status.Local
|
|
latestStatus.PinnedAt = status.PinnedAt
|
|
|
|
// Carry-over approvals. Remote instances might not yet
|
|
// serve statuses with the `approved_by` field, but we
|
|
// might have marked a status as pre-approved on our side
|
|
// based on the author's inclusion in a followers/following
|
|
// collection, or by providing pre-approval URI on the bare
|
|
// status passed to RefreshStatus. By carrying over previously
|
|
// set values we can avoid marking such statuses as "pending".
|
|
//
|
|
// If a remote has in the meantime retracted its approval,
|
|
// the next call to 'isPermittedStatus' will catch that.
|
|
if latestStatus.ApprovedByURI == "" && status.ApprovedByURI != "" {
|
|
latestStatus.ApprovedByURI = status.ApprovedByURI
|
|
}
|
|
|
|
// Check if this is a permitted status we should accept.
|
|
// Function also sets "PendingApproval" bool as necessary,
|
|
// and handles removal of existing statuses no longer permitted.
|
|
permit, err := d.isPermittedStatus(ctx, requestUser, status, latestStatus, isNew)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error checking permissibility for status %s: %w", uri, err)
|
|
}
|
|
|
|
if !permit {
|
|
// Return a checkable error type that can be ignored.
|
|
err := gtserror.Newf("dropping unpermitted status: %s", uri)
|
|
return nil, nil, gtserror.SetNotPermitted(err)
|
|
}
|
|
|
|
// Insert / update any attached status poll.
|
|
pollChanged, err := d.handleStatusPoll(ctx,
|
|
status,
|
|
latestStatus,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error handling poll for status %s: %w", uri, err)
|
|
}
|
|
|
|
// Populate mentions associated with status, passing
|
|
// in existing status to reuse old where possible.
|
|
// (especially important here to reduce need to dereference).
|
|
mentionsChanged, err := d.fetchStatusMentions(ctx,
|
|
requestUser,
|
|
status,
|
|
latestStatus,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error populating mentions for status %s: %w", uri, err)
|
|
}
|
|
|
|
// Ensure status in a thread is connected.
|
|
threadChanged, err := d.threadStatus(ctx,
|
|
status,
|
|
latestStatus,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error handling threading for status %s: %w", uri, err)
|
|
}
|
|
|
|
// Populate tags associated with status, passing
|
|
// in existing status to reuse old where possible.
|
|
tagsChanged, err := d.fetchStatusTags(ctx,
|
|
status,
|
|
latestStatus,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error populating tags for status %s: %w", uri, err)
|
|
}
|
|
|
|
// Populate media attachments associated with status,
|
|
// passing in existing status to reuse old where possible
|
|
// (especially important here to reduce need to dereference).
|
|
mediaChanged, err := d.fetchStatusAttachments(ctx,
|
|
requestUser,
|
|
status,
|
|
latestStatus,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error populating attachments for status %s: %w", uri, err)
|
|
}
|
|
|
|
// Populate emoji associated with status, passing
|
|
// in existing status to reuse old where possible
|
|
// (especially important here to reduce need to dereference).
|
|
emojiChanged, err := d.fetchStatusEmojis(ctx,
|
|
status,
|
|
latestStatus,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error populating emojis for status %s: %w", uri, err)
|
|
}
|
|
|
|
if isNew {
|
|
// Simplest case, insert this new status into the database.
|
|
if err := d.state.DB.PutStatus(ctx, latestStatus); err != nil {
|
|
return nil, nil, gtserror.Newf("error inserting new status %s: %w", uri, err)
|
|
}
|
|
} else {
|
|
// Check for and handle any edits to status, inserting
|
|
// historical edit if necessary. Also determines status
|
|
// columns that need updating in below query.
|
|
cols, err := d.handleStatusEdit(ctx,
|
|
status,
|
|
latestStatus,
|
|
pollChanged,
|
|
mentionsChanged,
|
|
threadChanged,
|
|
tagsChanged,
|
|
mediaChanged,
|
|
emojiChanged,
|
|
)
|
|
if err != nil {
|
|
return nil, nil, gtserror.Newf("error handling edit for status %s: %w", uri, err)
|
|
}
|
|
|
|
// With returned changed columns, now update the existing status entry.
|
|
if err := d.state.DB.UpdateStatus(ctx, latestStatus, cols...); err != nil {
|
|
return nil, nil, gtserror.Newf("error updating existing status %s: %w", uri, err)
|
|
}
|
|
}
|
|
|
|
return latestStatus, statusable, nil
|
|
}
|
|
|
|
// fetchStatusMentions populates the mentions on 'status', creating
|
|
// new where needed, or using unchanged mentions from 'existing' status.
|
|
func (d *Dereferencer) fetchStatusMentions(
|
|
ctx context.Context,
|
|
requestUser string,
|
|
existing *gtsmodel.Status,
|
|
status *gtsmodel.Status,
|
|
) (
|
|
changed bool,
|
|
err error,
|
|
) {
|
|
|
|
// Allocate new slice to take the yet-to-be created mention IDs.
|
|
status.MentionIDs = make([]string, len(status.Mentions))
|
|
|
|
for i := range status.Mentions {
|
|
var (
|
|
mention = status.Mentions[i]
|
|
alreadyExists bool
|
|
)
|
|
|
|
// Search existing status for a mention already stored,
|
|
// else ensure new mention's target account is populated.
|
|
mention, alreadyExists, err = d.getPopulatedMention(ctx,
|
|
requestUser,
|
|
existing,
|
|
mention,
|
|
)
|
|
if err != nil {
|
|
log.Errorf(ctx, "failed to derive mention: %v", err)
|
|
continue
|
|
}
|
|
|
|
if alreadyExists {
|
|
// This mention was already attached
|
|
// to the status, use it and continue.
|
|
status.Mentions[i] = mention
|
|
status.MentionIDs[i] = mention.ID
|
|
continue
|
|
}
|
|
|
|
// Mark status as
|
|
// having changed.
|
|
changed = true
|
|
|
|
// This mention didn't exist yet.
|
|
// Generate new ID according to latest update.
|
|
mention.ID = id.NewULIDFromTime(status.UpdatedAt)
|
|
|
|
// Set known further mention details.
|
|
mention.CreatedAt = status.UpdatedAt
|
|
mention.OriginAccount = status.Account
|
|
mention.OriginAccountID = status.AccountID
|
|
mention.OriginAccountURI = status.AccountURI
|
|
mention.TargetAccountID = mention.TargetAccount.ID
|
|
mention.TargetAccountURI = mention.TargetAccount.URI
|
|
mention.TargetAccountURL = mention.TargetAccount.URL
|
|
mention.StatusID = status.ID
|
|
mention.Status = status
|
|
|
|
// Place the new mention into the database.
|
|
if err := d.state.DB.PutMention(ctx, mention); err != nil {
|
|
return changed, gtserror.Newf("error putting mention in database: %w", err)
|
|
}
|
|
|
|
// Set the *new* mention and ID.
|
|
status.Mentions[i] = mention
|
|
status.MentionIDs[i] = mention.ID
|
|
}
|
|
|
|
for i := 0; i < len(status.MentionIDs); {
|
|
if status.MentionIDs[i] == "" {
|
|
// This is a failed mention population, likely due
|
|
// to invalid incoming data / now-deleted accounts.
|
|
copy(status.Mentions[i:], status.Mentions[i+1:])
|
|
copy(status.MentionIDs[i:], status.MentionIDs[i+1:])
|
|
status.Mentions = status.Mentions[:len(status.Mentions)-1]
|
|
status.MentionIDs = status.MentionIDs[:len(status.MentionIDs)-1]
|
|
continue
|
|
}
|
|
i++
|
|
}
|
|
|
|
return changed, nil
|
|
}
|
|
|
|
// threadStatus ensures that given status is threaded correctly
|
|
// where necessary. that is it will inherit a thread ID from the
|
|
// existing copy if it is threaded correctly, else it will inherit
|
|
// a thread ID from a parent with existing thread, else it will
|
|
// generate a new thread ID if status mentions a local account.
|
|
func (d *Dereferencer) threadStatus(
|
|
ctx context.Context,
|
|
existing *gtsmodel.Status,
|
|
status *gtsmodel.Status,
|
|
) (
|
|
changed bool,
|
|
err error,
|
|
) {
|
|
|
|
// Check for existing status
|
|
// that is already threaded.
|
|
if existing.ThreadID != "" {
|
|
|
|
// Existing is threaded correctly.
|
|
if existing.InReplyTo == nil ||
|
|
existing.InReplyTo.ThreadID == existing.ThreadID {
|
|
status.ThreadID = existing.ThreadID
|
|
return false, nil
|
|
}
|
|
|
|
// TODO: delete incorrect thread
|
|
}
|
|
|
|
// Check for existing parent to inherit threading from.
|
|
if inReplyTo := status.InReplyTo; inReplyTo != nil &&
|
|
inReplyTo.ThreadID != "" {
|
|
status.ThreadID = inReplyTo.ThreadID
|
|
return true, nil
|
|
}
|
|
|
|
// Parent wasn't threaded. If this
|
|
// status mentions a local account,
|
|
// we should thread it so that local
|
|
// account can mute it if they want.
|
|
mentionsLocal := slices.ContainsFunc(
|
|
status.Mentions,
|
|
func(m *gtsmodel.Mention) bool {
|
|
// If TargetAccount couldn't
|
|
// be deref'd, we know it's not
|
|
// a local account, so only
|
|
// check for non-nil accounts.
|
|
return m.TargetAccount != nil &&
|
|
m.TargetAccount.IsLocal()
|
|
},
|
|
)
|
|
|
|
if !mentionsLocal {
|
|
// Status doesn't mention a
|
|
// local account, so we don't
|
|
// need to thread it.
|
|
return false, nil
|
|
}
|
|
|
|
// Status mentions a local account.
|
|
// Create a new thread and assign
|
|
// it to the status.
|
|
threadID := id.NewULID()
|
|
|
|
// Insert new thread model into db.
|
|
if err := d.state.DB.PutThread(ctx,
|
|
>smodel.Thread{ID: threadID},
|
|
); err != nil {
|
|
return false, gtserror.Newf("error inserting new thread in db: %w", err)
|
|
}
|
|
|
|
// Set thread on latest status.
|
|
status.ThreadID = threadID
|
|
return true, nil
|
|
}
|
|
|
|
// fetchStatusTags populates the tags on 'status', fetching existing
|
|
// from the database and creating new where needed. 'existing' is used
|
|
// to fetch tags that have not changed since previous stored status.
|
|
func (d *Dereferencer) fetchStatusTags(
|
|
ctx context.Context,
|
|
existing *gtsmodel.Status,
|
|
status *gtsmodel.Status,
|
|
) (
|
|
changed bool,
|
|
err error,
|
|
) {
|
|
|
|
// Allocate new slice to take the yet-to-be determined tag IDs.
|
|
status.TagIDs = make([]string, len(status.Tags))
|
|
|
|
for i := range status.Tags {
|
|
tag := status.Tags[i]
|
|
|
|
// Look for tag in existing status with name.
|
|
existing, ok := existing.GetTagByName(tag.Name)
|
|
if ok && existing.ID != "" {
|
|
status.Tags[i] = existing
|
|
status.TagIDs[i] = existing.ID
|
|
continue
|
|
}
|
|
|
|
// Mark status as
|
|
// having changed.
|
|
changed = true
|
|
|
|
// Look for existing tag with name in the database.
|
|
existing, err := d.state.DB.GetTagByName(ctx, tag.Name)
|
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
|
return changed, gtserror.Newf("db error getting tag %s: %w", tag.Name, err)
|
|
} else if existing != nil {
|
|
status.Tags[i] = existing
|
|
status.TagIDs[i] = existing.ID
|
|
continue
|
|
}
|
|
|
|
// Create new ID for tag.
|
|
tag.ID = id.NewULID()
|
|
|
|
// Insert this tag with new name into the database.
|
|
if err := d.state.DB.PutTag(ctx, tag); err != nil {
|
|
log.Errorf(ctx, "db error putting tag %s: %v", tag.Name, err)
|
|
continue
|
|
}
|
|
|
|
// Set new tag ID in slice.
|
|
status.TagIDs[i] = tag.ID
|
|
}
|
|
|
|
// Remove any tag we couldn't get or create.
|
|
for i := 0; i < len(status.TagIDs); {
|
|
if status.TagIDs[i] == "" {
|
|
// This is a failed tag population, likely due
|
|
// to some database peculiarity / race condition.
|
|
copy(status.Tags[i:], status.Tags[i+1:])
|
|
copy(status.TagIDs[i:], status.TagIDs[i+1:])
|
|
status.Tags = status.Tags[:len(status.Tags)-1]
|
|
status.TagIDs = status.TagIDs[:len(status.TagIDs)-1]
|
|
continue
|
|
}
|
|
i++
|
|
}
|
|
|
|
return changed, nil
|
|
}
|
|
|
|
// fetchStatusAttachments populates the attachments on 'status', creating new database
|
|
// entries where needed and dereferencing it, or using unchanged from 'existing' status.
|
|
func (d *Dereferencer) fetchStatusAttachments(
|
|
ctx context.Context,
|
|
requestUser string,
|
|
existing *gtsmodel.Status,
|
|
status *gtsmodel.Status,
|
|
) (
|
|
changed bool,
|
|
err error,
|
|
) {
|
|
|
|
// Allocate new slice to take the yet-to-be fetched attachment IDs.
|
|
status.AttachmentIDs = make([]string, len(status.Attachments))
|
|
|
|
for i := range status.Attachments {
|
|
placeholder := status.Attachments[i]
|
|
|
|
// Look for existing media attachment with remote URL first.
|
|
existing, ok := existing.GetAttachmentByRemoteURL(placeholder.RemoteURL)
|
|
if ok && existing.ID != "" {
|
|
var info media.AdditionalMediaInfo
|
|
|
|
// Look for any difference in stored media description.
|
|
diff := (existing.Description != placeholder.Description)
|
|
if diff {
|
|
info.Description = &placeholder.Description
|
|
}
|
|
|
|
// If description changed,
|
|
// we mark media as changed.
|
|
changed = changed || diff
|
|
|
|
// Store any attachment updates and
|
|
// ensure media is locally cached.
|
|
existing, err := d.RefreshMedia(ctx,
|
|
requestUser,
|
|
existing,
|
|
info,
|
|
diff,
|
|
)
|
|
if err != nil {
|
|
log.Errorf(ctx, "error updating existing attachment: %v", err)
|
|
|
|
// specifically do NOT continue here,
|
|
// we already have a model, we don't
|
|
// want to drop it from the status, just
|
|
// log that an update for it failed.
|
|
}
|
|
|
|
// Set the existing attachment.
|
|
status.Attachments[i] = existing
|
|
status.AttachmentIDs[i] = existing.ID
|
|
continue
|
|
}
|
|
|
|
// Mark status as
|
|
// having changed.
|
|
changed = true
|
|
|
|
// Load this new media attachment.
|
|
attachment, err := d.GetMedia(ctx,
|
|
requestUser,
|
|
status.AccountID,
|
|
placeholder.RemoteURL,
|
|
media.AdditionalMediaInfo{
|
|
StatusID: &status.ID,
|
|
RemoteURL: &placeholder.RemoteURL,
|
|
Description: &placeholder.Description,
|
|
Blurhash: &placeholder.Blurhash,
|
|
},
|
|
)
|
|
if err != nil {
|
|
if attachment == nil {
|
|
log.Errorf(ctx, "error loading attachment %s: %v", placeholder.RemoteURL, err)
|
|
continue
|
|
}
|
|
|
|
// non-fatal error occurred during loading, still use it.
|
|
log.Warnf(ctx, "partially loaded attachment: %v", err)
|
|
}
|
|
|
|
// Set the *new* attachment and ID.
|
|
status.Attachments[i] = attachment
|
|
status.AttachmentIDs[i] = attachment.ID
|
|
}
|
|
|
|
for i := 0; i < len(status.AttachmentIDs); {
|
|
if status.AttachmentIDs[i] == "" {
|
|
// Remove totally failed attachment populations
|
|
copy(status.Attachments[i:], status.Attachments[i+1:])
|
|
copy(status.AttachmentIDs[i:], status.AttachmentIDs[i+1:])
|
|
status.Attachments = status.Attachments[:len(status.Attachments)-1]
|
|
status.AttachmentIDs = status.AttachmentIDs[:len(status.AttachmentIDs)-1]
|
|
continue
|
|
}
|
|
i++
|
|
}
|
|
|
|
return changed, nil
|
|
}
|
|
|
|
// fetchStatusEmojis populates the emojis on 'status', creating new database entries
|
|
// where needed and dereferencing it, or using unchanged from 'existing' status.
|
|
func (d *Dereferencer) fetchStatusEmojis(
|
|
ctx context.Context,
|
|
existing *gtsmodel.Status,
|
|
status *gtsmodel.Status,
|
|
) (
|
|
changed bool,
|
|
err error,
|
|
) {
|
|
|
|
// Fetch the updated emojis for our status.
|
|
emojis, changed, err := d.fetchEmojis(ctx,
|
|
existing.Emojis,
|
|
status.Emojis,
|
|
)
|
|
if err != nil {
|
|
return changed, gtserror.Newf("error fetching emojis: %w", err)
|
|
}
|
|
|
|
if !changed {
|
|
// Use existing status emoji objects.
|
|
status.EmojiIDs = existing.EmojiIDs
|
|
status.Emojis = existing.Emojis
|
|
return false, nil
|
|
}
|
|
|
|
// Set latest emojis.
|
|
status.Emojis = emojis
|
|
|
|
// Iterate over and set changed emoji IDs.
|
|
status.EmojiIDs = make([]string, len(emojis))
|
|
for i, emoji := range emojis {
|
|
status.EmojiIDs[i] = emoji.ID
|
|
}
|
|
|
|
return true, nil
|
|
}
|
|
|
|
// handleStatusPoll handles both inserting of new status poll or the
|
|
// update of an existing poll. this handles the case of simple vote
|
|
// count updates (without being classified as a change of the poll
|
|
// itself), as well as full poll changes that delete existing instance.
|
|
func (d *Dereferencer) handleStatusPoll(
|
|
ctx context.Context,
|
|
existing *gtsmodel.Status,
|
|
status *gtsmodel.Status,
|
|
) (
|
|
changed bool,
|
|
err error,
|
|
) {
|
|
switch {
|
|
case existing.Poll == nil && status.Poll == nil:
|
|
// no poll before or after, nothing to do.
|
|
return false, nil
|
|
|
|
case existing.Poll == nil && status.Poll != nil:
|
|
// no previous poll, insert new status poll!
|
|
return true, d.insertStatusPoll(ctx, status)
|
|
|
|
case status.Poll == nil:
|
|
// existing status poll has been deleted, remove this from the database.
|
|
if err = d.state.DB.DeletePollByID(ctx, existing.Poll.ID); err != nil {
|
|
err = gtserror.Newf("error deleting poll from database: %w", err)
|
|
}
|
|
return true, err
|
|
|
|
case pollChanged(existing.Poll, status.Poll):
|
|
// existing status poll has been changed, remove this from the database.
|
|
if err = d.state.DB.DeletePollByID(ctx, existing.Poll.ID); err != nil {
|
|
return true, gtserror.Newf("error deleting poll from database: %w", err)
|
|
}
|
|
|
|
// insert latest poll version into database.
|
|
return true, d.insertStatusPoll(ctx, status)
|
|
|
|
case pollStateUpdated(existing.Poll, status.Poll):
|
|
// Since we last saw it, the poll has updated!
|
|
// Whether that be stats, or close time.
|
|
poll := existing.Poll
|
|
poll.Closing = pollJustClosed(existing.Poll, status.Poll)
|
|
poll.ClosedAt = status.Poll.ClosedAt
|
|
poll.Voters = status.Poll.Voters
|
|
poll.Votes = status.Poll.Votes
|
|
|
|
// Update poll model in the database (specifically only the possible changed columns).
|
|
if err = d.state.DB.UpdatePoll(ctx, poll, "closed_at", "voters", "votes"); err != nil {
|
|
return false, gtserror.Newf("error updating poll: %w", err)
|
|
}
|
|
|
|
// Update poll on status.
|
|
status.PollID = poll.ID
|
|
status.Poll = poll
|
|
return false, nil
|
|
|
|
default:
|
|
// latest and existing
|
|
// polls are up to date.
|
|
poll := existing.Poll
|
|
status.PollID = poll.ID
|
|
status.Poll = poll
|
|
return false, nil
|
|
}
|
|
}
|
|
|
|
// insertStatusPoll inserts an assumed new poll attached to status into the database, this
|
|
// also handles generating new ID for the poll and setting necessary fields on the status.
|
|
func (d *Dereferencer) insertStatusPoll(ctx context.Context, status *gtsmodel.Status) error {
|
|
var err error
|
|
|
|
// Generate new ID for poll from latest updated time.
|
|
status.Poll.ID = id.NewULIDFromTime(status.UpdatedAt)
|
|
|
|
// Update the status<->poll links.
|
|
status.PollID = status.Poll.ID
|
|
status.Poll.StatusID = status.ID
|
|
status.Poll.Status = status
|
|
|
|
// Insert this latest poll into the database.
|
|
err = d.state.DB.PutPoll(ctx, status.Poll)
|
|
if err != nil {
|
|
return gtserror.Newf("error putting poll in database: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// handleStatusEdit compiles a list of changed status table columns between
|
|
// existing and latest status model, and where necessary inserts a historic
|
|
// edit of the status into the database to store its previous state. the
|
|
// returned slice is a list of columns requiring updating in the database.
|
|
func (d *Dereferencer) handleStatusEdit(
|
|
ctx context.Context,
|
|
existing *gtsmodel.Status,
|
|
status *gtsmodel.Status,
|
|
pollChanged bool,
|
|
mentionsChanged bool,
|
|
threadChanged bool,
|
|
tagsChanged bool,
|
|
mediaChanged bool,
|
|
emojiChanged bool,
|
|
) (
|
|
cols []string,
|
|
err error,
|
|
) {
|
|
var edited bool
|
|
|
|
// Preallocate max slice length.
|
|
cols = make([]string, 0, 13)
|
|
|
|
// Always update `fetched_at`.
|
|
cols = append(cols, "fetched_at")
|
|
|
|
// Check for edited status content.
|
|
if existing.Content != status.Content {
|
|
cols = append(cols, "content")
|
|
edited = true
|
|
}
|
|
|
|
// Check for edited status content warning.
|
|
if existing.ContentWarning != status.ContentWarning {
|
|
cols = append(cols, "content_warning")
|
|
edited = true
|
|
}
|
|
|
|
// Check for edited status sensitive flag.
|
|
if *existing.Sensitive != *status.Sensitive {
|
|
cols = append(cols, "sensitive")
|
|
edited = true
|
|
}
|
|
|
|
// Check for edited status language tag.
|
|
if existing.Language != status.Language {
|
|
cols = append(cols, "language")
|
|
edited = true
|
|
}
|
|
|
|
if pollChanged {
|
|
// Attached poll was changed.
|
|
cols = append(cols, "poll_id")
|
|
edited = true
|
|
}
|
|
|
|
if mentionsChanged {
|
|
cols = append(cols, "mentions") // i.e. MentionIDs
|
|
|
|
// Mentions changed doesn't necessarily
|
|
// indicate an edit, it may just not have
|
|
// been previously populated properly.
|
|
}
|
|
|
|
if threadChanged {
|
|
cols = append(cols, "thread_id")
|
|
|
|
// Thread changed doesn't necessarily
|
|
// indicate an edit, it may just now
|
|
// actually be included in a thread.
|
|
}
|
|
|
|
if tagsChanged {
|
|
cols = append(cols, "tags") // i.e. TagIDs
|
|
|
|
// Tags changed doesn't necessarily
|
|
// indicate an edit, it may just not have
|
|
// been previously populated properly.
|
|
}
|
|
|
|
if mediaChanged {
|
|
// Attached media was changed.
|
|
cols = append(cols, "attachments") // i.e. AttachmentIDs
|
|
edited = true
|
|
}
|
|
|
|
if emojiChanged {
|
|
// Attached emojis changed.
|
|
cols = append(cols, "emojis") // i.e. EmojiIDs
|
|
|
|
// Emojis changed doesn't necessarily
|
|
// indicate an edit, it may just not have
|
|
// been previously populated properly.
|
|
}
|
|
|
|
if edited {
|
|
// We prefer to use provided 'upated_at', but ensure
|
|
// it fits chronologically with creation / last update.
|
|
if !status.UpdatedAt.After(status.CreatedAt) ||
|
|
!status.UpdatedAt.After(existing.UpdatedAt) {
|
|
|
|
// Else fallback to now as update time.
|
|
status.UpdatedAt = status.FetchedAt
|
|
}
|
|
|
|
// Status has been editted since last
|
|
// we saw it, take snapshot of existing.
|
|
var edit gtsmodel.StatusEdit
|
|
edit.ID = id.NewULIDFromTime(status.UpdatedAt)
|
|
edit.Content = existing.Content
|
|
edit.ContentWarning = existing.ContentWarning
|
|
edit.Text = existing.Text
|
|
edit.Language = existing.Language
|
|
edit.Sensitive = existing.Sensitive
|
|
edit.StatusID = status.ID
|
|
|
|
// Copy existing attachments and descriptions.
|
|
edit.AttachmentIDs = existing.AttachmentIDs
|
|
edit.Attachments = existing.Attachments
|
|
if l := len(existing.Attachments); l > 0 {
|
|
edit.AttachmentDescriptions = make([]string, l)
|
|
for i, attach := range existing.Attachments {
|
|
edit.AttachmentDescriptions[i] = attach.Description
|
|
}
|
|
}
|
|
|
|
// Edit creation is last update time.
|
|
edit.CreatedAt = existing.UpdatedAt
|
|
|
|
if existing.Poll != nil {
|
|
// Poll only set if existing contained them.
|
|
edit.PollOptions = existing.Poll.Options
|
|
|
|
if !*existing.Poll.HideCounts || pollChanged {
|
|
// If the counts are allowed to be
|
|
// shown, or poll has changed, then
|
|
// include poll vote counts in edit.
|
|
edit.PollVotes = existing.Poll.Votes
|
|
}
|
|
}
|
|
|
|
// Insert this new edit of existing status into database.
|
|
if err := d.state.DB.PutStatusEdit(ctx, &edit); err != nil {
|
|
return nil, gtserror.Newf("error putting edit in database: %w", err)
|
|
}
|
|
|
|
// Add edit to list of edits on the status.
|
|
status.EditIDs = append(status.EditIDs, edit.ID)
|
|
status.Edits = append(status.Edits, &edit)
|
|
|
|
// Add updated_at and edits to list of cols.
|
|
cols = append(cols, "updated_at", "edits")
|
|
}
|
|
|
|
return cols, nil
|
|
}
|
|
|
|
// getPopulatedMention tries to populate the given
|
|
// mention with the correct TargetAccount and (if not
|
|
// yet set) TargetAccountURI, returning the populated
|
|
// mention.
|
|
//
|
|
// Will check on the existing status if the mention
|
|
// is already there and populated; if so, existing
|
|
// mention will be returned along with `true`.
|
|
//
|
|
// Otherwise, this function will try to parse first
|
|
// the Href of the mention, and then the namestring,
|
|
// to see who it targets, and go fetch that account.
|
|
func (d *Dereferencer) getPopulatedMention(
|
|
ctx context.Context,
|
|
requestUser string,
|
|
existing *gtsmodel.Status,
|
|
mention *gtsmodel.Mention,
|
|
) (
|
|
*gtsmodel.Mention,
|
|
bool, // True if mention already exists in the DB.
|
|
error,
|
|
) {
|
|
// Mentions can be created using Name or Href.
|
|
// Prefer Href (TargetAccountURI), fall back to Name.
|
|
if mention.TargetAccountURI != "" {
|
|
|
|
// Look for existing mention with target account's URI, if so use this.
|
|
existingMention, ok := existing.GetMentionByTargetURI(mention.TargetAccountURI)
|
|
if ok && existingMention.ID != "" {
|
|
return existingMention, true, nil
|
|
}
|
|
|
|
// Ensure that mention account URI is parseable.
|
|
accountURI, err := url.Parse(mention.TargetAccountURI)
|
|
if err != nil {
|
|
err := gtserror.Newf("invalid account uri %q: %w", mention.TargetAccountURI, err)
|
|
return nil, false, err
|
|
}
|
|
|
|
// Ensure we have account of the mention target dereferenced.
|
|
mention.TargetAccount, _, err = d.getAccountByURI(ctx,
|
|
requestUser,
|
|
accountURI,
|
|
)
|
|
if err != nil {
|
|
err := gtserror.Newf("failed to dereference account %s: %w", accountURI, err)
|
|
return nil, false, err
|
|
}
|
|
} else {
|
|
|
|
// Href wasn't set, extract the username and domain parts from namestring.
|
|
username, domain, err := util.ExtractNamestringParts(mention.NameString)
|
|
if err != nil {
|
|
err := gtserror.Newf("failed to parse namestring %s: %w", mention.NameString, err)
|
|
return nil, false, err
|
|
}
|
|
|
|
// Look for existing mention with username domain target, if so use this.
|
|
existingMention, ok := existing.GetMentionByUsernameDomain(username, domain)
|
|
if ok && existingMention.ID != "" {
|
|
return existingMention, true, nil
|
|
}
|
|
|
|
// Ensure we have the account of the mention target dereferenced.
|
|
mention.TargetAccount, _, err = d.getAccountByUsernameDomain(ctx,
|
|
requestUser,
|
|
username,
|
|
domain,
|
|
)
|
|
if err != nil {
|
|
err := gtserror.Newf("failed to dereference account %s: %w", mention.NameString, err)
|
|
return nil, false, err
|
|
}
|
|
|
|
// Look for existing mention with target account's URI, if so use this.
|
|
existingMention, ok = existing.GetMentionByTargetURI(mention.TargetAccountURI)
|
|
if ok && existingMention.ID != "" {
|
|
return existingMention, true, nil
|
|
}
|
|
}
|
|
|
|
// At this point, mention.TargetAccountURI
|
|
// and mention.TargetAccount must be set.
|
|
return mention, false, nil
|
|
}
|