summaryrefslogtreecommitdiff
path: root/internal/federation/dereferencing/attachment.go
blob: 3a4213cdb5d628267bdc7eaa4eb2341c242a63ff (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
/*
   GoToSocial
   Copyright (C) 2021 GoToSocial Authors admin@gotosocial.org

   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"
	"fmt"
	"net/url"

	"github.com/sirupsen/logrus"
	"github.com/superseriousbusiness/gotosocial/internal/db"
	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
)

func (d *deref) GetRemoteAttachment(ctx context.Context, requestingUsername string, minAttachment *gtsmodel.MediaAttachment) (*gtsmodel.MediaAttachment, error) {
	if minAttachment.RemoteURL == "" {
		return nil, fmt.Errorf("GetRemoteAttachment: minAttachment remote URL was empty")
	}
	remoteAttachmentURL := minAttachment.RemoteURL

	l := logrus.WithFields(logrus.Fields{
		"username":            requestingUsername,
		"remoteAttachmentURL": remoteAttachmentURL,
	})

	// return early if we already have the attachment somewhere
	maybeAttachment := &gtsmodel.MediaAttachment{}
	where := []db.Where{
		{
			Key:   "remote_url",
			Value: remoteAttachmentURL,
		},
	}

	if err := d.db.GetWhere(ctx, where, maybeAttachment); err == nil {
		// we already the attachment in the database
		l.Debugf("GetRemoteAttachment: attachment already exists with id %s", maybeAttachment.ID)
		return maybeAttachment, nil
	}

	a, err := d.RefreshAttachment(ctx, requestingUsername, minAttachment)
	if err != nil {
		return nil, fmt.Errorf("GetRemoteAttachment: error refreshing attachment: %s", err)
	}

	if err := d.db.Put(ctx, a); err != nil {
		if err != db.ErrAlreadyExists {
			return nil, fmt.Errorf("GetRemoteAttachment: error inserting attachment: %s", err)
		}
	}

	return a, nil
}

func (d *deref) RefreshAttachment(ctx context.Context, requestingUsername string, minAttachment *gtsmodel.MediaAttachment) (*gtsmodel.MediaAttachment, error) {
	// it just doesn't exist or we have to refresh
	if minAttachment.AccountID == "" {
		return nil, fmt.Errorf("RefreshAttachment: minAttachment account ID was empty")
	}

	if minAttachment.File.ContentType == "" {
		return nil, fmt.Errorf("RefreshAttachment: minAttachment.file.contentType was empty")
	}

	t, err := d.transportController.NewTransportForUsername(ctx, requestingUsername)
	if err != nil {
		return nil, fmt.Errorf("RefreshAttachment: error creating transport: %s", err)
	}

	derefURI, err := url.Parse(minAttachment.RemoteURL)
	if err != nil {
		return nil, err
	}

	attachmentBytes, err := t.DereferenceMedia(ctx, derefURI, minAttachment.File.ContentType)
	if err != nil {
		return nil, fmt.Errorf("RefreshAttachment: error dereferencing media: %s", err)
	}

	a, err := d.mediaHandler.ProcessAttachment(ctx, attachmentBytes, minAttachment)
	if err != nil {
		return nil, fmt.Errorf("RefreshAttachment: error processing attachment: %s", err)
	}

	return a, nil
}