Bumps dependencies (jwt 3.2.0, puma 8.0.2, net-imap 0.6.4.1 and others via bundle update) to resolve bundler-audit advisories, and applies standardrb autofixes so the lint job passes. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
58 lines
1.8 KiB
Ruby
58 lines
1.8 KiB
Ruby
require "ipaddr"
|
|
require "resolv"
|
|
|
|
# SSRF guard for outbound requests to admin-configured URLs (currently the OIDC
|
|
# backchannel logout endpoint). Blocks hosts that are, or resolve to, private,
|
|
# loopback, link-local (incl. the cloud metadata address 169.254.169.254) or
|
|
# otherwise non-public address space.
|
|
module PrivateAddressCheck
|
|
module_function
|
|
|
|
# Hostnames that are internal by definition and must never be dialled.
|
|
BLOCKED_HOSTNAMES = %w[localhost metadata.google.internal].freeze
|
|
|
|
# Fast, DNS-free check: catches IP literals and well-known internal hostnames.
|
|
# Suitable for model validation (deterministic, immediate admin feedback).
|
|
def internal_host?(host)
|
|
host = host.to_s.downcase
|
|
return true if host.blank?
|
|
return true if BLOCKED_HOSTNAMES.include?(host)
|
|
return true if host.end_with?(".localhost")
|
|
|
|
ip = parse_ip(host)
|
|
ip ? internal_ip?(ip) : false
|
|
end
|
|
|
|
# Authoritative check: resolves the hostname and blocks if ANY address is
|
|
# internal. Suitable for request time — also defeats a public hostname that
|
|
# has been pointed at an internal IP (DNS rebinding to internal space).
|
|
def resolves_to_internal?(host)
|
|
addresses(host).any? { |ip| internal_ip?(ip) }
|
|
end
|
|
|
|
def addresses(host)
|
|
ip = parse_ip(host)
|
|
return [ip] if ip
|
|
|
|
Resolv.getaddresses(host.to_s).filter_map { |a| parse_ip(a) }
|
|
rescue
|
|
# Resolution failure: surface no addresses. Callers treat "can't resolve" as
|
|
# not-provably-internal; the dial itself will then fail safely.
|
|
[]
|
|
end
|
|
|
|
def internal_ip?(ip)
|
|
ip.loopback? || ip.private? || ip.link_local? || unspecified?(ip)
|
|
end
|
|
|
|
def parse_ip(str)
|
|
IPAddr.new(str.to_s)
|
|
rescue IPAddr::Error
|
|
nil
|
|
end
|
|
|
|
def unspecified?(ip)
|
|
ip == IPAddr.new("0.0.0.0") || ip == IPAddr.new("::")
|
|
end
|
|
end
|