forked from corazawaf/coraza
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwaf.go
125 lines (99 loc) · 3 KB
/
waf.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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
// Copyright 2022 Juan Pablo Tosso and the OWASP Coraza contributors
// SPDX-License-Identifier: Apache-2.0
package coraza
import (
"fmt"
"github.com/corazawaf/coraza/v3/internal/corazawaf"
"github.com/corazawaf/coraza/v3/internal/seclang"
"github.com/corazawaf/coraza/v3/types"
)
// WAF instance is used to store configurations and rules
// Every web application should have a different WAF instance,
// but you can share an instance if you are ok with sharing
// configurations, rules and logging.
// Transactions and SecLang parser requires a WAF instance
// You can use as many WAF instances as you want, and they are
// concurrent safe
type WAF interface {
// NewTransaction Creates a new initialized transaction for this WAF instance
NewTransaction() types.Transaction
NewTransactionWithID(id string) types.Transaction
}
// NewWAF creates a new WAF instance with the provided configuration.
func NewWAF(config WAFConfig) (WAF, error) {
c := config.(*wafConfig)
waf := corazawaf.NewWAF()
if c.debugLogger != nil {
waf.Logger = c.debugLogger
}
parser := seclang.NewParser(waf)
if c.fsRoot != nil {
parser.SetRoot(c.fsRoot)
}
for _, r := range c.rules {
switch {
case r.rule != nil:
if err := waf.Rules.Add(r.rule); err != nil {
return nil, fmt.Errorf("invalid WAF config from rule: %w", err)
}
case r.str != "":
if err := parser.FromString(r.str); err != nil {
return nil, fmt.Errorf("invalid WAF config from string: %w", err)
}
case r.file != "":
if err := parser.FromFile(r.file); err != nil {
return nil, fmt.Errorf("invalid WAF config from file: %w", err)
}
}
}
if a := c.auditLog; a != nil {
if a.relevantOnly {
waf.AuditEngine = types.AuditEngineRelevantOnly
} else {
waf.AuditEngine = types.AuditEngineOn
}
waf.AuditLogParts = a.parts
if a.writer != nil {
waf.SetAuditLogWriter(a.writer)
}
}
if err := waf.InitAuditLogWriter(); err != nil {
return nil, fmt.Errorf("invalid WAF config from audit log: %w", err)
}
if c.requestBodyAccess {
waf.RequestBodyAccess = true
}
if c.requestBodyLimit != nil {
waf.RequestBodyLimit = int64(*c.requestBodyLimit)
}
if c.requestBodyInMemoryLimit != nil {
waf.SetRequestBodyInMemoryLimit(int64(*c.requestBodyInMemoryLimit))
}
if c.responseBodyAccess {
waf.ResponseBodyAccess = true
}
if c.responseBodyLimit != nil {
waf.ResponseBodyLimit = int64(*c.responseBodyLimit)
}
if c.responseBodyMimeTypes != nil {
waf.ResponseBodyMimeTypes = c.responseBodyMimeTypes
}
if c.errorCallback != nil {
waf.ErrorLogCb = c.errorCallback
}
if err := waf.Validate(); err != nil {
return nil, err
}
return wafWrapper{waf: waf}, nil
}
type wafWrapper struct {
waf *corazawaf.WAF
}
// NewTransaction implements the same method on WAF.
func (w wafWrapper) NewTransaction() types.Transaction {
return w.waf.NewTransaction()
}
// NewTransactionWithID implements the same method on WAF.
func (w wafWrapper) NewTransactionWithID(id string) types.Transaction {
return w.waf.NewTransactionWithID(id)
}