2025-01-08 10:29:40 +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/>.
|
|
|
|
|
|
|
|
package transport
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
2025-01-20 09:56:00 +00:00
|
|
|
"time"
|
2025-01-08 10:29:40 +00:00
|
|
|
|
|
|
|
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
|
|
|
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
2025-01-20 09:56:00 +00:00
|
|
|
"github.com/superseriousbusiness/gotosocial/internal/log"
|
2025-01-08 10:29:40 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type DereferenceDomainPermissionsResp struct {
|
|
|
|
// Set only if response was 200 OK.
|
|
|
|
// It's up to the caller to close
|
|
|
|
// this when they're done with it.
|
|
|
|
Body io.ReadCloser
|
|
|
|
|
|
|
|
// True if response
|
|
|
|
// was 304 Not Modified.
|
|
|
|
Unmodified bool
|
|
|
|
|
|
|
|
// May be set
|
|
|
|
// if 200 or 304.
|
|
|
|
ETag string
|
2025-01-20 09:56:00 +00:00
|
|
|
|
|
|
|
// May be set
|
|
|
|
// if 200 or 304.
|
|
|
|
LastModified time.Time
|
2025-01-08 10:29:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (t *transport) DereferenceDomainPermissions(
|
|
|
|
ctx context.Context,
|
|
|
|
permSub *gtsmodel.DomainPermissionSubscription,
|
|
|
|
skipCache bool,
|
|
|
|
) (*DereferenceDomainPermissionsResp, error) {
|
|
|
|
// Prepare new HTTP request to endpoint
|
|
|
|
req, err := http.NewRequestWithContext(ctx, "GET", permSub.URI, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set basic auth header if necessary.
|
|
|
|
if permSub.FetchUsername != "" || permSub.FetchPassword != "" {
|
|
|
|
req.SetBasicAuth(permSub.FetchUsername, permSub.FetchPassword)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set relevant Accept headers.
|
|
|
|
// Allow fallback in case target doesn't
|
|
|
|
// negotiate content type correctly.
|
2025-01-20 09:56:00 +00:00
|
|
|
req.Header.Set("Accept-Charset", "utf-8")
|
|
|
|
req.Header.Set("Accept", permSub.ContentType.String()+","+"*/*")
|
2025-01-08 10:29:40 +00:00
|
|
|
|
|
|
|
// If skipCache is true, we want to skip setting Cache
|
|
|
|
// headers so that we definitely don't get a 304 back.
|
|
|
|
if !skipCache {
|
2025-01-20 09:56:00 +00:00
|
|
|
// If we've got a Last-Modified stored for this list,
|
|
|
|
// set If-Modified-Since to make the request conditional.
|
2025-01-08 10:29:40 +00:00
|
|
|
//
|
|
|
|
// See: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/If-Modified-Since
|
2025-01-20 09:56:00 +00:00
|
|
|
if !permSub.LastModified.IsZero() {
|
|
|
|
// http.Time wants UTC.
|
|
|
|
lmUTC := permSub.LastModified.UTC()
|
|
|
|
req.Header.Set("If-Modified-Since", lmUTC.Format(http.TimeFormat))
|
2025-01-08 10:29:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If we've got an ETag stored for this list, set
|
|
|
|
// If-None-Match to make the request conditional.
|
|
|
|
// See https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/ETag#caching_of_unchanged_resources.
|
2025-01-20 09:56:00 +00:00
|
|
|
if permSub.ETag != "" {
|
|
|
|
req.Header.Set("If-None-Match", permSub.ETag)
|
2025-01-08 10:29:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Perform the HTTP request
|
|
|
|
rsp, err := t.GET(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we have an unexpected / error response,
|
|
|
|
// wrap + return as error. This will also drain
|
|
|
|
// and close the response body for us.
|
|
|
|
if rsp.StatusCode != http.StatusOK &&
|
|
|
|
rsp.StatusCode != http.StatusNotModified {
|
|
|
|
err := gtserror.NewFromResponse(rsp)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2025-01-20 09:56:00 +00:00
|
|
|
// Check already if we were given a valid ETag or
|
|
|
|
// Last-Modified we can use, as these cache headers
|
|
|
|
// are often returned even on Not Modified responses.
|
2025-01-08 10:29:40 +00:00
|
|
|
permsResp := &DereferenceDomainPermissionsResp{
|
2025-01-20 09:56:00 +00:00
|
|
|
ETag: rsp.Header.Get("ETag"),
|
|
|
|
LastModified: validateLastModified(ctx, rsp.Header.Get("Last-Modified")),
|
2025-01-08 10:29:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if rsp.StatusCode == http.StatusNotModified {
|
|
|
|
// Nothing has changed on the remote side
|
|
|
|
// since we last fetched, so there's nothing
|
|
|
|
// to do and we don't need to read the body.
|
|
|
|
rsp.Body.Close()
|
|
|
|
permsResp.Unmodified = true
|
|
|
|
} else {
|
|
|
|
// Return the live body to the caller.
|
|
|
|
permsResp.Body = rsp.Body
|
|
|
|
}
|
|
|
|
|
|
|
|
return permsResp, nil
|
|
|
|
}
|
2025-01-20 09:56:00 +00:00
|
|
|
|
|
|
|
// Validate Last-Modified to ensure it's not
|
|
|
|
// garbagio, and not more than a minute in the
|
|
|
|
// future (to allow for clock issues + rounding).
|
|
|
|
func validateLastModified(
|
|
|
|
ctx context.Context,
|
|
|
|
lastModified string,
|
|
|
|
) time.Time {
|
|
|
|
if lastModified == "" {
|
|
|
|
// Not set,
|
|
|
|
// no problem.
|
|
|
|
return time.Time{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to parse and see what we get.
|
|
|
|
switch lm, err := http.ParseTime(lastModified); {
|
|
|
|
case err != nil:
|
|
|
|
// No good,
|
|
|
|
// chuck it.
|
|
|
|
log.Debugf(ctx,
|
|
|
|
"discarding invalid Last-Modified header %s: %+v",
|
|
|
|
lastModified, err,
|
|
|
|
)
|
|
|
|
return time.Time{}
|
|
|
|
|
|
|
|
case lm.Unix() > time.Now().Add(1*time.Minute).Unix():
|
|
|
|
// In the future,
|
|
|
|
// chuck it.
|
|
|
|
log.Debugf(ctx,
|
|
|
|
"discarding in-the-future Last-Modified header %s",
|
|
|
|
lastModified,
|
|
|
|
)
|
|
|
|
return time.Time{}
|
|
|
|
|
|
|
|
default:
|
|
|
|
// It's fine,
|
|
|
|
// keep it.
|
|
|
|
return lm
|
|
|
|
}
|
|
|
|
}
|