url.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package urllib // import "miniflux.app/v2/internal/urllib"
  4. import (
  5. "errors"
  6. "fmt"
  7. "net/url"
  8. "strings"
  9. )
  10. // IsAbsoluteURL returns true if the link is absolute.
  11. func IsAbsoluteURL(link string) bool {
  12. u, err := url.Parse(link)
  13. if err != nil {
  14. return false
  15. }
  16. return u.IsAbs()
  17. }
  18. // GetAbsoluteURL returns the absolute form of `input` if possible, as well as its parsed form.
  19. func GetAbsoluteURL(input string) (string, *url.URL, error) {
  20. if strings.HasPrefix(input, "//") {
  21. return "https:" + input, nil, nil
  22. }
  23. if strings.HasPrefix(input, "https://") || strings.HasPrefix(input, "http://") {
  24. return input, nil, nil
  25. }
  26. u, err := url.Parse(input)
  27. if err != nil {
  28. return "", nil, fmt.Errorf("unable to parse input URL: %v", err)
  29. }
  30. if u.IsAbs() {
  31. return u.String(), u, nil
  32. }
  33. return "", u, nil
  34. }
  35. // AbsoluteURL converts the input URL as absolute URL if necessary.
  36. func AbsoluteURL(baseURL, input string) (string, error) {
  37. absURL, u, err := GetAbsoluteURL(input)
  38. if err != nil {
  39. return "", err
  40. }
  41. if absURL != "" {
  42. return absURL, nil
  43. }
  44. base, err := url.Parse(baseURL)
  45. if err != nil {
  46. return "", fmt.Errorf("unable to parse base URL: %v", err)
  47. }
  48. return base.ResolveReference(u).String(), nil
  49. }
  50. // RootURL returns absolute URL without the path.
  51. func RootURL(websiteURL string) string {
  52. if strings.HasPrefix(websiteURL, "//") {
  53. websiteURL = "https://" + websiteURL[2:]
  54. }
  55. absoluteURL, err := AbsoluteURL(websiteURL, "")
  56. if err != nil {
  57. return websiteURL
  58. }
  59. u, err := url.Parse(absoluteURL)
  60. if err != nil {
  61. return absoluteURL
  62. }
  63. return u.Scheme + "://" + u.Host + "/"
  64. }
  65. // IsHTTPS returns true if the URL is using HTTPS.
  66. func IsHTTPS(websiteURL string) bool {
  67. parsedURL, err := url.Parse(websiteURL)
  68. if err != nil {
  69. return false
  70. }
  71. return strings.EqualFold(parsedURL.Scheme, "https")
  72. }
  73. // Domain returns only the domain part of the given URL.
  74. func Domain(websiteURL string) string {
  75. parsedURL, err := url.Parse(websiteURL)
  76. if err != nil {
  77. return websiteURL
  78. }
  79. return parsedURL.Host
  80. }
  81. // DomainWithoutWWW returns only the domain part of the given URL, with the "www." prefix removed if present.
  82. func DomainWithoutWWW(websiteURL string) string {
  83. return strings.TrimPrefix(Domain(websiteURL), "www.")
  84. }
  85. // JoinBaseURLAndPath returns a URL string with the provided path elements joined together.
  86. func JoinBaseURLAndPath(baseURL, path string) (string, error) {
  87. if baseURL == "" {
  88. return "", errors.New("empty base URL")
  89. }
  90. if path == "" {
  91. return "", errors.New("empty path")
  92. }
  93. _, err := url.Parse(baseURL)
  94. if err != nil {
  95. return "", fmt.Errorf("invalid base URL: %w", err)
  96. }
  97. finalURL, err := url.JoinPath(baseURL, path)
  98. if err != nil {
  99. return "", fmt.Errorf("unable to join base URL %s and path %s: %w", baseURL, path, err)
  100. }
  101. return finalURL, nil
  102. }