mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-24 23:39:19 -04:00 
			
		
		
		
	* caddyhttp: Determine real client IP if trusted proxies configured * Support customizing client IP header * Implement client_ip matcher, deprecate remote_ip's forwarded option
		
			
				
	
	
		
			345 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2015 Matthew Holt and The Caddy Authors
 | |
| //
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| //     http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| package caddyhttp
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"net"
 | |
| 	"net/http"
 | |
| 	"net/netip"
 | |
| 	"reflect"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/caddyserver/caddy/v2"
 | |
| 	"github.com/caddyserver/caddy/v2/caddyconfig/caddyfile"
 | |
| 	"github.com/google/cel-go/cel"
 | |
| 	"github.com/google/cel-go/common/types/ref"
 | |
| 	"go.uber.org/zap"
 | |
| )
 | |
| 
 | |
| // MatchRemoteIP matches requests by the remote IP address,
 | |
| // i.e. the IP address of the direct connection to Caddy.
 | |
| type MatchRemoteIP struct {
 | |
| 	// The IPs or CIDR ranges to match.
 | |
| 	Ranges []string `json:"ranges,omitempty"`
 | |
| 
 | |
| 	// If true, prefer the first IP in the request's X-Forwarded-For
 | |
| 	// header, if present, rather than the immediate peer's IP, as
 | |
| 	// the reference IP against which to match. Note that it is easy
 | |
| 	// to spoof request headers. Default: false
 | |
| 	// DEPRECATED: This is insecure, MatchClientIP should be used instead.
 | |
| 	Forwarded bool `json:"forwarded,omitempty"`
 | |
| 
 | |
| 	// cidrs and zones vars should aligned always in the same
 | |
| 	// length and indexes for matching later
 | |
| 	cidrs  []*netip.Prefix
 | |
| 	zones  []string
 | |
| 	logger *zap.Logger
 | |
| }
 | |
| 
 | |
| // MatchClientIP matches requests by the client IP address,
 | |
| // i.e. the resolved address, considering trusted proxies.
 | |
| type MatchClientIP struct {
 | |
| 	// The IPs or CIDR ranges to match.
 | |
| 	Ranges []string `json:"ranges,omitempty"`
 | |
| 
 | |
| 	// cidrs and zones vars should aligned always in the same
 | |
| 	// length and indexes for matching later
 | |
| 	cidrs  []*netip.Prefix
 | |
| 	zones  []string
 | |
| 	logger *zap.Logger
 | |
| }
 | |
| 
 | |
| func init() {
 | |
| 	caddy.RegisterModule(MatchRemoteIP{})
 | |
| 	caddy.RegisterModule(MatchClientIP{})
 | |
| }
 | |
| 
 | |
| // CaddyModule returns the Caddy module information.
 | |
| func (MatchRemoteIP) CaddyModule() caddy.ModuleInfo {
 | |
| 	return caddy.ModuleInfo{
 | |
| 		ID:  "http.matchers.remote_ip",
 | |
| 		New: func() caddy.Module { return new(MatchRemoteIP) },
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // UnmarshalCaddyfile implements caddyfile.Unmarshaler.
 | |
| func (m *MatchRemoteIP) UnmarshalCaddyfile(d *caddyfile.Dispenser) error {
 | |
| 	for d.Next() {
 | |
| 		for d.NextArg() {
 | |
| 			if d.Val() == "forwarded" {
 | |
| 				if len(m.Ranges) > 0 {
 | |
| 					return d.Err("if used, 'forwarded' must be first argument")
 | |
| 				}
 | |
| 				m.Forwarded = true
 | |
| 				continue
 | |
| 			}
 | |
| 			if d.Val() == "private_ranges" {
 | |
| 				m.Ranges = append(m.Ranges, PrivateRangesCIDR()...)
 | |
| 				continue
 | |
| 			}
 | |
| 			m.Ranges = append(m.Ranges, d.Val())
 | |
| 		}
 | |
| 		if d.NextBlock(0) {
 | |
| 			return d.Err("malformed remote_ip matcher: blocks are not supported")
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // CELLibrary produces options that expose this matcher for use in CEL
 | |
| // expression matchers.
 | |
| //
 | |
| // Example:
 | |
| //
 | |
| //	expression remote_ip('forwarded', '192.168.0.0/16', '172.16.0.0/12', '10.0.0.0/8')
 | |
| func (MatchRemoteIP) CELLibrary(ctx caddy.Context) (cel.Library, error) {
 | |
| 	return CELMatcherImpl(
 | |
| 		// name of the macro, this is the function name that users see when writing expressions.
 | |
| 		"remote_ip",
 | |
| 		// name of the function that the macro will be rewritten to call.
 | |
| 		"remote_ip_match_request_list",
 | |
| 		// internal data type of the MatchPath value.
 | |
| 		[]*cel.Type{cel.ListType(cel.StringType)},
 | |
| 		// function to convert a constant list of strings to a MatchPath instance.
 | |
| 		func(data ref.Val) (RequestMatcher, error) {
 | |
| 			refStringList := reflect.TypeOf([]string{})
 | |
| 			strList, err := data.ConvertToNative(refStringList)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 
 | |
| 			m := MatchRemoteIP{}
 | |
| 
 | |
| 			for _, input := range strList.([]string) {
 | |
| 				if input == "forwarded" {
 | |
| 					if len(m.Ranges) > 0 {
 | |
| 						return nil, errors.New("if used, 'forwarded' must be first argument")
 | |
| 					}
 | |
| 					m.Forwarded = true
 | |
| 					continue
 | |
| 				}
 | |
| 				m.Ranges = append(m.Ranges, input)
 | |
| 			}
 | |
| 
 | |
| 			err = m.Provision(ctx)
 | |
| 			return m, err
 | |
| 		},
 | |
| 	)
 | |
| }
 | |
| 
 | |
| // Provision parses m's IP ranges, either from IP or CIDR expressions.
 | |
| func (m *MatchRemoteIP) Provision(ctx caddy.Context) error {
 | |
| 	m.logger = ctx.Logger()
 | |
| 	cidrs, zones, err := provisionCidrsZonesFromRanges(m.Ranges)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	m.cidrs = cidrs
 | |
| 	m.zones = zones
 | |
| 
 | |
| 	if m.Forwarded {
 | |
| 		m.logger.Warn("remote_ip's forwarded mode is deprecated; use the 'client_ip' matcher instead")
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // Match returns true if r matches m.
 | |
| func (m MatchRemoteIP) Match(r *http.Request) bool {
 | |
| 	address := r.RemoteAddr
 | |
| 	if m.Forwarded {
 | |
| 		if fwdFor := r.Header.Get("X-Forwarded-For"); fwdFor != "" {
 | |
| 			address = strings.TrimSpace(strings.Split(fwdFor, ",")[0])
 | |
| 		}
 | |
| 	}
 | |
| 	clientIP, zoneID, err := parseIPZoneFromString(address)
 | |
| 	if err != nil {
 | |
| 		m.logger.Error("getting remote IP", zap.Error(err))
 | |
| 		return false
 | |
| 	}
 | |
| 	matches, zoneFilter := matchIPByCidrZones(clientIP, zoneID, m.cidrs, m.zones)
 | |
| 	if !matches && !zoneFilter {
 | |
| 		m.logger.Debug("zone ID from remote IP did not match", zap.String("zone", zoneID))
 | |
| 	}
 | |
| 	return matches
 | |
| }
 | |
| 
 | |
| // CaddyModule returns the Caddy module information.
 | |
| func (MatchClientIP) CaddyModule() caddy.ModuleInfo {
 | |
| 	return caddy.ModuleInfo{
 | |
| 		ID:  "http.matchers.client_ip",
 | |
| 		New: func() caddy.Module { return new(MatchClientIP) },
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // UnmarshalCaddyfile implements caddyfile.Unmarshaler.
 | |
| func (m *MatchClientIP) UnmarshalCaddyfile(d *caddyfile.Dispenser) error {
 | |
| 	for d.Next() {
 | |
| 		for d.NextArg() {
 | |
| 			if d.Val() == "private_ranges" {
 | |
| 				m.Ranges = append(m.Ranges, PrivateRangesCIDR()...)
 | |
| 				continue
 | |
| 			}
 | |
| 			m.Ranges = append(m.Ranges, d.Val())
 | |
| 		}
 | |
| 		if d.NextBlock(0) {
 | |
| 			return d.Err("malformed client_ip matcher: blocks are not supported")
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // CELLibrary produces options that expose this matcher for use in CEL
 | |
| // expression matchers.
 | |
| //
 | |
| // Example:
 | |
| //
 | |
| //	expression client_ip('192.168.0.0/16', '172.16.0.0/12', '10.0.0.0/8')
 | |
| func (MatchClientIP) CELLibrary(ctx caddy.Context) (cel.Library, error) {
 | |
| 	return CELMatcherImpl(
 | |
| 		// name of the macro, this is the function name that users see when writing expressions.
 | |
| 		"client_ip",
 | |
| 		// name of the function that the macro will be rewritten to call.
 | |
| 		"client_ip_match_request_list",
 | |
| 		// internal data type of the MatchPath value.
 | |
| 		[]*cel.Type{cel.ListType(cel.StringType)},
 | |
| 		// function to convert a constant list of strings to a MatchPath instance.
 | |
| 		func(data ref.Val) (RequestMatcher, error) {
 | |
| 			refStringList := reflect.TypeOf([]string{})
 | |
| 			strList, err := data.ConvertToNative(refStringList)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 
 | |
| 			m := MatchClientIP{
 | |
| 				Ranges: strList.([]string),
 | |
| 			}
 | |
| 
 | |
| 			err = m.Provision(ctx)
 | |
| 			return m, err
 | |
| 		},
 | |
| 	)
 | |
| }
 | |
| 
 | |
| // Provision parses m's IP ranges, either from IP or CIDR expressions.
 | |
| func (m *MatchClientIP) Provision(ctx caddy.Context) error {
 | |
| 	m.logger = ctx.Logger()
 | |
| 	cidrs, zones, err := provisionCidrsZonesFromRanges(m.Ranges)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	m.cidrs = cidrs
 | |
| 	m.zones = zones
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // Match returns true if r matches m.
 | |
| func (m MatchClientIP) Match(r *http.Request) bool {
 | |
| 	address := GetVar(r.Context(), ClientIPVarKey).(string)
 | |
| 	clientIP, zoneID, err := parseIPZoneFromString(address)
 | |
| 	if err != nil {
 | |
| 		m.logger.Error("getting client IP", zap.Error(err))
 | |
| 		return false
 | |
| 	}
 | |
| 	matches, zoneFilter := matchIPByCidrZones(clientIP, zoneID, m.cidrs, m.zones)
 | |
| 	if !matches && !zoneFilter {
 | |
| 		m.logger.Debug("zone ID from client IP did not match", zap.String("zone", zoneID))
 | |
| 	}
 | |
| 	return matches
 | |
| }
 | |
| 
 | |
| func provisionCidrsZonesFromRanges(ranges []string) ([]*netip.Prefix, []string, error) {
 | |
| 	cidrs := []*netip.Prefix{}
 | |
| 	zones := []string{}
 | |
| 	for _, str := range ranges {
 | |
| 		// Exclude the zone_id from the IP
 | |
| 		if strings.Contains(str, "%") {
 | |
| 			split := strings.Split(str, "%")
 | |
| 			str = split[0]
 | |
| 			// write zone identifiers in m.zones for matching later
 | |
| 			zones = append(zones, split[1])
 | |
| 		} else {
 | |
| 			zones = append(zones, "")
 | |
| 		}
 | |
| 		if strings.Contains(str, "/") {
 | |
| 			ipNet, err := netip.ParsePrefix(str)
 | |
| 			if err != nil {
 | |
| 				return nil, nil, fmt.Errorf("parsing CIDR expression '%s': %v", str, err)
 | |
| 			}
 | |
| 			cidrs = append(cidrs, &ipNet)
 | |
| 		} else {
 | |
| 			ipAddr, err := netip.ParseAddr(str)
 | |
| 			if err != nil {
 | |
| 				return nil, nil, fmt.Errorf("invalid IP address: '%s': %v", str, err)
 | |
| 			}
 | |
| 			ipNew := netip.PrefixFrom(ipAddr, ipAddr.BitLen())
 | |
| 			cidrs = append(cidrs, &ipNew)
 | |
| 		}
 | |
| 	}
 | |
| 	return cidrs, zones, nil
 | |
| }
 | |
| 
 | |
| func parseIPZoneFromString(address string) (netip.Addr, string, error) {
 | |
| 	ipStr, _, err := net.SplitHostPort(address)
 | |
| 	if err != nil {
 | |
| 		ipStr = address // OK; probably didn't have a port
 | |
| 	}
 | |
| 
 | |
| 	// Some IPv6-Adresses can contain zone identifiers at the end,
 | |
| 	// which are separated with "%"
 | |
| 	zoneID := ""
 | |
| 	if strings.Contains(ipStr, "%") {
 | |
| 		split := strings.Split(ipStr, "%")
 | |
| 		ipStr = split[0]
 | |
| 		zoneID = split[1]
 | |
| 	}
 | |
| 
 | |
| 	ipAddr, err := netip.ParseAddr(ipStr)
 | |
| 	if err != nil {
 | |
| 		return netip.IPv4Unspecified(), "", err
 | |
| 	}
 | |
| 
 | |
| 	return ipAddr, zoneID, nil
 | |
| }
 | |
| 
 | |
| func matchIPByCidrZones(clientIP netip.Addr, zoneID string, cidrs []*netip.Prefix, zones []string) (bool, bool) {
 | |
| 	zoneFilter := true
 | |
| 	for i, ipRange := range cidrs {
 | |
| 		if ipRange.Contains(clientIP) {
 | |
| 			// Check if there are zone filters assigned and if they match.
 | |
| 			if zones[i] == "" || zoneID == zones[i] {
 | |
| 				return true, false
 | |
| 			}
 | |
| 			zoneFilter = false
 | |
| 		}
 | |
| 	}
 | |
| 	return false, zoneFilter
 | |
| }
 | |
| 
 | |
| // Interface guards
 | |
| var (
 | |
| 	_ RequestMatcher        = (*MatchRemoteIP)(nil)
 | |
| 	_ caddy.Provisioner     = (*MatchRemoteIP)(nil)
 | |
| 	_ caddyfile.Unmarshaler = (*MatchRemoteIP)(nil)
 | |
| 	_ CELLibraryProducer    = (*MatchRemoteIP)(nil)
 | |
| 
 | |
| 	_ RequestMatcher        = (*MatchClientIP)(nil)
 | |
| 	_ caddy.Provisioner     = (*MatchClientIP)(nil)
 | |
| 	_ caddyfile.Unmarshaler = (*MatchClientIP)(nil)
 | |
| 	_ CELLibraryProducer    = (*MatchClientIP)(nil)
 | |
| )
 |