| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091 |
- // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
- // SPDX-License-Identifier: Apache-2.0
- package request // import "miniflux.app/v2/internal/http/request"
- import (
- "net"
- "net/http"
- "strings"
- )
- // IsTrustedIP reports whether the given remote IP address belongs to one of the trusted networks.
- func IsTrustedIP(remoteIP string, trustedNetworks []string) bool {
- if len(trustedNetworks) == 0 {
- return false
- }
- ip := net.ParseIP(remoteIP)
- if ip == nil {
- return false
- }
- for _, cidr := range trustedNetworks {
- _, network, err := net.ParseCIDR(cidr)
- if err != nil {
- continue
- }
- if network.Contains(ip) {
- return true
- }
- }
- return false
- }
- // FindClientIP returns the real client IP address using trusted reverse-proxy headers when allowed.
- func FindClientIP(r *http.Request, isTrustedProxyClient bool) string {
- if isTrustedProxyClient {
- headers := [...]string{"X-Forwarded-For", "X-Real-Ip"}
- for _, header := range headers {
- value := r.Header.Get(header)
- if value != "" {
- addresses := strings.Split(value, ",")
- address := strings.TrimSpace(addresses[0])
- address = dropIPv6zone(address)
- if net.ParseIP(address) != nil {
- return address
- }
- }
- }
- }
- // Fallback to TCP/IP source IP address.
- return FindRemoteIP(r)
- }
- // FindRemoteIP returns the parsed remote IP address from the request,
- // falling back to 127.0.0.1 if the address is empty, a unix socket, or invalid.
- func FindRemoteIP(r *http.Request) string {
- if r.RemoteAddr == "@" || r.RemoteAddr == "" {
- return "127.0.0.1"
- }
- // If it looks like it has a port (IPv4:port or [IPv6]:port), try to split it.
- ip, _, err := net.SplitHostPort(r.RemoteAddr)
- if err != nil {
- // No port — could be a bare IPv4, IPv6, or IPv6 with zone.
- ip = r.RemoteAddr
- }
- // Strip IPv6 zone identifier if present (e.g., %eth0).
- ip = dropIPv6zone(ip)
- // Validate the IP address.
- if net.ParseIP(ip) == nil {
- return "127.0.0.1"
- }
- return ip
- }
- func dropIPv6zone(address string) string {
- idx := strings.IndexByte(address, '%')
- if idx != -1 {
- address = address[:idx]
- }
- return address
- }
|