url.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  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. "fmt"
  6. "net/url"
  7. "strings"
  8. )
  9. // IsAbsoluteURL returns true if the link is absolute.
  10. func IsAbsoluteURL(link string) bool {
  11. u, err := url.Parse(link)
  12. if err != nil {
  13. return false
  14. }
  15. return u.IsAbs()
  16. }
  17. // AbsoluteURL converts the input URL as absolute URL if necessary.
  18. func AbsoluteURL(baseURL, input string) (string, error) {
  19. if strings.HasPrefix(input, "//") {
  20. return "https:" + input, nil
  21. }
  22. if strings.HasPrefix(input, "https://") || strings.HasPrefix(input, "http://") {
  23. return input, nil
  24. }
  25. u, err := url.Parse(input)
  26. if err != nil {
  27. return "", fmt.Errorf("unable to parse input URL: %v", err)
  28. }
  29. if u.IsAbs() {
  30. return u.String(), nil
  31. }
  32. base, err := url.Parse(baseURL)
  33. if err != nil {
  34. return "", fmt.Errorf("unable to parse base URL: %v", err)
  35. }
  36. return base.ResolveReference(u).String(), nil
  37. }
  38. // RootURL returns absolute URL without the path.
  39. func RootURL(websiteURL string) string {
  40. if strings.HasPrefix(websiteURL, "//") {
  41. websiteURL = "https://" + websiteURL[2:]
  42. }
  43. absoluteURL, err := AbsoluteURL(websiteURL, "")
  44. if err != nil {
  45. return websiteURL
  46. }
  47. u, err := url.Parse(absoluteURL)
  48. if err != nil {
  49. return absoluteURL
  50. }
  51. return u.Scheme + "://" + u.Host + "/"
  52. }
  53. // IsHTTPS returns true if the URL is using HTTPS.
  54. func IsHTTPS(websiteURL string) bool {
  55. parsedURL, err := url.Parse(websiteURL)
  56. if err != nil {
  57. return false
  58. }
  59. return strings.EqualFold(parsedURL.Scheme, "https")
  60. }
  61. // Domain returns only the domain part of the given URL.
  62. func Domain(websiteURL string) string {
  63. parsedURL, err := url.Parse(websiteURL)
  64. if err != nil {
  65. return websiteURL
  66. }
  67. return parsedURL.Host
  68. }
  69. // JoinBaseURLAndPath returns a URL string with the provided path elements joined together.
  70. func JoinBaseURLAndPath(baseURL, path string) (string, error) {
  71. if baseURL == "" {
  72. return "", fmt.Errorf("empty base URL")
  73. }
  74. if path == "" {
  75. return "", fmt.Errorf("empty path")
  76. }
  77. _, err := url.Parse(baseURL)
  78. if err != nil {
  79. return "", fmt.Errorf("invalid base URL: %w", err)
  80. }
  81. finalURL, err := url.JoinPath(baseURL, path)
  82. if err != nil {
  83. return "", fmt.Errorf("unable to join base URL %s and path %s: %w", baseURL, path, err)
  84. }
  85. return finalURL, nil
  86. }