/*
   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 := d.log.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
}