Skip to content

Commit d55b588

Browse files
WGH-zimmerle
authored andcommitted
Add proper error handling to @rx operator
1 parent 2d1898a commit d55b588

File tree

8 files changed

+114
-10
lines changed

8 files changed

+114
-10
lines changed

src/operators/rx.cc

+13
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,12 @@ namespace operators {
2929

3030
bool Rx::init(const std::string &arg, std::string *error) {
3131
if (m_string->m_containsMacro == false) {
32+
std::string regex_error;
3233
m_re = new Regex(m_param);
34+
if (!m_re->ok(&regex_error)) {
35+
*error = "Failed to compile regular expression " + m_re->getPattern() + ": " + regex_error;
36+
return false;
37+
}
3338
}
3439

3540
return true;
@@ -47,6 +52,14 @@ bool Rx::evaluate(Transaction *transaction, Rule *rule,
4752
if (m_string->m_containsMacro) {
4853
std::string eparam(m_string->evaluate(transaction));
4954
re = new Regex(eparam);
55+
std::string regex_error;
56+
if (!re->ok(&regex_error)) {
57+
ms_dbg_a(transaction, 2,
58+
"Failed to compile regular expression with macro "
59+
+ re->getPattern() + ": " + regex_error);
60+
delete re;
61+
return false;
62+
}
5063
} else {
5164
re = m_re;
5265
}

src/operators/verify_cc.cc

+3-2
Original file line numberDiff line numberDiff line change
@@ -69,8 +69,9 @@ int VerifyCC::luhnVerify(const char *ccnumber, int len) {
6969
bool VerifyCC::init(const std::string &param2, std::string *error) {
7070
m_re.reset(new modsecurity::regex::Regex(m_param));
7171

72-
if (!m_re->ok()) {
73-
*error = "Failed to compile regular expression " + m_re->getPattern();
72+
std::string regex_error;
73+
if (!m_re->ok(&regex_error)) {
74+
*error = "Failed to compile regular expression " + m_re->getPattern() + ": " + regex_error;
7475
return false;
7576
}
7677
return true;

src/regex/backend/backend.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ class Backend {
2828
public:
2929
virtual ~Backend() {}
3030

31-
virtual bool ok() const = 0;
31+
virtual bool ok(std::string *error = nullptr) const = 0;
3232

3333
virtual std::vector<RegexMatch> searchAll(const std::string& s, bool overlapping = false) const = 0;
3434
virtual bool search(const std::string &s, RegexMatch *m = nullptr, ssize_t max_groups = -1) const = 0;

src/regex/backend/pcre.cc

+19-1
Original file line numberDiff line numberDiff line change
@@ -43,10 +43,28 @@ Pcre::Pcre(const std::string& pattern_)
4343

4444
m_pc = pcre_compile(pattern.c_str(), PCRE_DOTALL|PCRE_MULTILINE,
4545
&errptr, &erroffset, NULL);
46+
if (m_pc == NULL) {
47+
m_error = "pcre_compile error at offset " + std::to_string(erroffset) + ": " + std::string(errptr);
48+
return;
49+
}
4650

4751
m_pce = pcre_study(m_pc, pcre_study_opt, &errptr);
52+
if (m_pce == NULL) {
53+
m_error = "pcre_study error: " + std::string(errptr);
54+
pcre_free(m_pc);
55+
m_pc = nullptr;
56+
return;
57+
}
4858

49-
pcre_fullinfo(m_pc, m_pce, PCRE_INFO_CAPTURECOUNT, &m_capture_count);
59+
int res = pcre_fullinfo(m_pc, m_pce, PCRE_INFO_CAPTURECOUNT, &m_capture_count);
60+
if (res != 0) {
61+
// N.B. There's no need to free m_pce here, because it's freed in
62+
// destructor anyway. However, m_pc must be freed and set to NULL
63+
// here so error condition is properly detected by ok() method.
64+
m_error = "pcre_fullinfo error: code " + std::to_string(res);
65+
pcre_free(m_pc);
66+
m_pc = nullptr;
67+
}
5068
}
5169

5270

src/regex/backend/pcre.h

+11-2
Original file line numberDiff line numberDiff line change
@@ -43,8 +43,15 @@ class Pcre : public Backend {
4343
Pcre(const Pcre&) = delete;
4444
Pcre& operator=(const Pcre&) = delete;
4545

46-
virtual bool ok() const override {
47-
return m_pc != NULL;
46+
virtual bool ok(std::string *error = nullptr) const override {
47+
if (m_pc != NULL) {
48+
return true;
49+
}
50+
if (error != nullptr) {
51+
*error= m_error;
52+
}
53+
54+
return false;
4855
}
4956

5057
std::vector<RegexMatch> searchAll(const std::string& s, bool overlapping = false) const override;
@@ -60,6 +67,8 @@ class Pcre : public Backend {
6067

6168
pcre *m_pc = NULL;
6269
pcre_extra *m_pce = NULL;
70+
71+
std::string m_error;
6372
};
6473

6574
#endif

src/regex/backend/re2.h

+8-2
Original file line numberDiff line numberDiff line change
@@ -40,8 +40,14 @@ class Re2 : public Backend {
4040
Re2(const Re2&) = delete;
4141
Re2& operator=(const Re2&) = delete;
4242

43-
virtual bool ok() const override {
44-
return re.ok();
43+
virtual bool ok(std::string *error = nullptr) const override {
44+
if (re.ok()) {
45+
return true;
46+
}
47+
if (error != nullptr) {
48+
*error = re.error();
49+
}
50+
return false;
4551
}
4652

4753
std::vector<RegexMatch> searchAll(const std::string& s, bool overlapping = false) const override;

src/regex/backend_fallback.h

+2-2
Original file line numberDiff line numberDiff line change
@@ -45,8 +45,8 @@ class BackendFallback : public backend::Backend {
4545
: backend(compile_regex_fallback<Args...>(pattern))
4646
{}
4747

48-
virtual bool ok() const override {
49-
return backend->ok();
48+
virtual bool ok(std::string *error = nullptr) const override {
49+
return backend->ok(error);
5050
}
5151

5252
std::vector<RegexMatch> searchAll(const std::string& s, bool overlapping = false) const override {

test/test-cases/regression/operator-rx.json

+57
Original file line numberDiff line numberDiff line change
@@ -85,5 +85,62 @@
8585
"SecRuleEngine On",
8686
"SecRule REQUEST_HEADERS:Content-Length \"!^0$\" \"id:1,phase:2,pass,t:trim,block\""
8787
]
88+
},
89+
{
90+
"enabled":1,
91+
"version_min":300000,
92+
"version_max":0,
93+
"title":"Testing Operator :: @rx with invalid regular expression",
94+
"expected":{
95+
"parser_error":"Rules error.*Failed to compile regular expression \\(\\(value1\\):"
96+
},
97+
"rules":[
98+
"SecRuleEngine On",
99+
"SecRule ARGS \"@rx ((value1)\" \"id:1,phase:2,pass,t:trim\""
100+
]
101+
},
102+
{
103+
"enabled":1,
104+
"version_min":300000,
105+
"title":"Testing Operator :: @rx with invalid regular expression after macro expansion",
106+
"client":{
107+
"ip":"200.249.12.31",
108+
"port":123
109+
},
110+
"server":{
111+
"ip":"200.249.12.31",
112+
"port":80
113+
},
114+
"request":{
115+
"headers":{
116+
"Host":"localhost",
117+
"User-Agent":"curl/7.38.0",
118+
"Accept":"*/*",
119+
"Content-Length": "27",
120+
"Content-Type": "application/x-www-form-urlencoded"
121+
},
122+
"uri":"/",
123+
"method":"POST",
124+
"body": [
125+
"param1=value1&param2=value2"
126+
]
127+
},
128+
"response":{
129+
"headers":{
130+
"Date":"Mon, 13 Jul 2015 20:02:41 GMT",
131+
"Last-Modified":"Sun, 26 Oct 2014 22:33:37 GMT",
132+
"Content-Type":"text/html"
133+
},
134+
"body":[
135+
"no need."
136+
]
137+
},
138+
"expected":{
139+
"debug_log":"Failed to compile regular expression with macro \\(\\(\\):"
140+
},
141+
"rules":[
142+
"SecRuleEngine On",
143+
"SecRule ARGS \"@rx ((%{TX.DOESNT_NEED_TO_EXIST_IT_WILL_BE_AN_EMPTY_STRING})\" \"id:1,phase:2,pass,t:trim\""
144+
]
88145
}
89146
]

0 commit comments

Comments
 (0)