-
-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathplugin.go
105 lines (88 loc) · 3.46 KB
/
plugin.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
package caddydefender
import (
"bufio"
"github.com/caddyserver/caddy/v2"
"github.com/caddyserver/caddy/v2/caddyconfig/caddyfile"
"github.com/caddyserver/caddy/v2/caddyconfig/httpcaddyfile"
"github.com/caddyserver/caddy/v2/modules/caddyhttp"
"go.uber.org/zap"
"net"
"os"
)
func init() {
// Register the module with Caddy
caddy.RegisterModule(Defender{})
httpcaddyfile.RegisterHandlerDirective("defender", parseCaddyfile)
httpcaddyfile.RegisterDirectiveOrder("defender", "before", "basicauth")
}
// Defender implements an HTTP middleware that enforces IP-based rules to protect your site from AIs/Scrapers.
// It allows you to block or manipulate requests based on the client's IP address by specifying IP ranges to block
// or using predefined ranges for popular services like AWS, GCP, OpenAI, and GitHub Copilot.
//
// The middleware supports multiple responder types, including blocking requests, returning garbage data, or
// sending custom messages.
type Defender struct {
// AdditionalRanges specifies additional IP ranges provided by the user to block or manipulate.
// These ranges are in CIDR notation (e.g., "192.168.1.0/24") and are applied alongside predefined ranges.
// This field is optional.
AdditionalRanges []string `json:"additional_ranges,omitempty"`
// RangesFile specifies the path to a file containing IP ranges (one per line) to block or manipulate.
// The file can include CIDR ranges or predefined range keys (e.g., "aws", "gcloud").
// This field is optional.
RangesFile string `json:"ranges_file,omitempty"`
// Message specifies a custom message to return to the client when using the "custom" responder.
// This field is optional and only used when the responder type is set to "custom".
Message string `json:"message,omitempty"`
// RawResponder is an internal field that represents the responder type specified in the configuration.
// Supported values are "block", "garbage", and "custom".
// This field is optional and is used during configuration unmarshaling.
RawResponder string `json:"raw_responder,omitempty"`
// responder is the internal responder interface used to handle requests that match the specified IP ranges.
// It is set based on the value of RawResponder during configuration validation.
responder Responder
// log is the logger used for logging debug and error messages within the middleware.
log *zap.Logger
}
// Provision sets up the middleware and logger.
func (m *Defender) Provision(ctx caddy.Context) error {
m.log = ctx.Logger(m)
// Load ranges from the specified text filez
if m.RangesFile != "" {
file, err := os.Open(m.RangesFile)
if err != nil {
return err
}
defer file.Close()
scanner := bufio.NewScanner(file)
for scanner.Scan() {
line := scanner.Text()
_, _, err := net.ParseCIDR(line)
if err != nil {
return err
}
}
if err := scanner.Err(); err != nil {
return err
}
}
return nil
}
// CaddyModule returns the Caddy module information.
func (Defender) CaddyModule() caddy.ModuleInfo {
return caddy.ModuleInfo{
ID: "http.handlers.defender",
New: func() caddy.Module { return new(Defender) },
}
}
// parseCaddyfile unmarshals tokens from h into a new Defender.
func parseCaddyfile(h httpcaddyfile.Helper) (caddyhttp.MiddlewareHandler, error) {
var m Defender
err := m.UnmarshalCaddyfile(h.Dispenser)
return m, err
}
// Interface guards
var (
_ caddy.Provisioner = (*Defender)(nil)
//_ caddyhttp.MiddlewareHandler = (*Defender)(nil)
_ caddyfile.Unmarshaler = (*Defender)(nil)
)