2023-06-19 17:42:47 -04:00
|
|
|
// SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
2017-11-20 00:10:04 -05:00
|
|
|
|
2023-08-13 22:09:01 -04:00
|
|
|
package urllib // import "miniflux.app/v2/internal/urllib"
|
2017-11-20 00:10:04 -05:00
|
|
|
|
2019-12-26 18:26:23 -05:00
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
)
|
2017-11-20 00:10:04 -05:00
|
|
|
|
2020-01-04 18:18:24 -05:00
|
|
|
// IsAbsoluteURL returns true if the link is absolute.
|
|
|
|
func IsAbsoluteURL(link string) bool {
|
|
|
|
u, err := url.Parse(link)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return u.IsAbs()
|
|
|
|
}
|
|
|
|
|
2017-12-02 01:29:18 -05:00
|
|
|
// AbsoluteURL converts the input URL as absolute URL if necessary.
|
|
|
|
func AbsoluteURL(baseURL, input string) (string, error) {
|
2017-11-20 00:10:04 -05:00
|
|
|
if strings.HasPrefix(input, "//") {
|
|
|
|
input = "https://" + input[2:]
|
|
|
|
}
|
|
|
|
|
|
|
|
u, err := url.Parse(input)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("unable to parse input URL: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if u.IsAbs() {
|
|
|
|
return u.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
base, err := url.Parse(baseURL)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("unable to parse base URL: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return base.ResolveReference(u).String(), nil
|
|
|
|
}
|
|
|
|
|
2017-12-02 01:29:18 -05:00
|
|
|
// RootURL returns absolute URL without the path.
|
|
|
|
func RootURL(websiteURL string) string {
|
2017-11-20 00:10:04 -05:00
|
|
|
if strings.HasPrefix(websiteURL, "//") {
|
|
|
|
websiteURL = "https://" + websiteURL[2:]
|
|
|
|
}
|
|
|
|
|
2017-12-02 01:29:18 -05:00
|
|
|
absoluteURL, err := AbsoluteURL(websiteURL, "")
|
2017-11-20 00:10:04 -05:00
|
|
|
if err != nil {
|
|
|
|
return websiteURL
|
|
|
|
}
|
|
|
|
|
|
|
|
u, err := url.Parse(absoluteURL)
|
|
|
|
if err != nil {
|
|
|
|
return absoluteURL
|
|
|
|
}
|
|
|
|
|
|
|
|
return u.Scheme + "://" + u.Host + "/"
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsHTTPS returns true if the URL is using HTTPS.
|
|
|
|
func IsHTTPS(websiteURL string) bool {
|
|
|
|
parsedURL, err := url.Parse(websiteURL)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.ToLower(parsedURL.Scheme) == "https"
|
|
|
|
}
|
2017-12-02 01:29:18 -05:00
|
|
|
|
|
|
|
// Domain returns only the domain part of the given URL.
|
|
|
|
func Domain(websiteURL string) string {
|
|
|
|
parsedURL, err := url.Parse(websiteURL)
|
|
|
|
if err != nil {
|
|
|
|
return websiteURL
|
|
|
|
}
|
|
|
|
|
|
|
|
return parsedURL.Host
|
|
|
|
}
|
2023-08-12 22:01:22 -04:00
|
|
|
|
|
|
|
// JoinBaseURLAndPath returns a URL string with the provided path elements joined together.
|
|
|
|
func JoinBaseURLAndPath(baseURL, path string) (string, error) {
|
|
|
|
if baseURL == "" {
|
|
|
|
return "", fmt.Errorf("empty base URL")
|
|
|
|
}
|
|
|
|
|
|
|
|
if path == "" {
|
|
|
|
return "", fmt.Errorf("empty path")
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := url.Parse(baseURL)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("invalid base URL: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
finalURL, err := url.JoinPath(baseURL, path)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("unable to join base URL %s and path %s: %w", baseURL, path, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return finalURL, nil
|
|
|
|
}
|