-
Notifications
You must be signed in to change notification settings - Fork 13
Expand file tree
/
Copy pathGoWrappers.cpp
More file actions
178 lines (169 loc) · 6.4 KB
/
Copy pathGoWrappers.cpp
File metadata and controls
178 lines (169 loc) · 6.4 KB
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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
#include "Includes.h"
GoString GoCreateString(const std::string& s) {
return GoString{ s.c_str(), static_cast<ptrdiff_t>(s.size()) };
}
GoSlice GoCreateSlice(const std::vector<int64_t>& v) {
return GoSlice{ static_cast<void*>(const_cast<int64_t*>(v.data())),
static_cast<GoInt>(v.size()),
static_cast<GoInt>(v.capacity()) };
}
/*
Helper function to safely get a string field from EventCache.
Returns empty string if stack is empty, otherwise returns the field value.
*/
static inline std::string GetEventCacheField(std::string EventCache::*field) {
auto& eventCacheStack = AIKIDO_GLOBAL(eventCacheStack);
return eventCacheStack.Empty() ? "" : eventCacheStack.Top().*field;
}
/*
Callback wrapper called by the RequestProcessor (GO) whenever it needs data from PHP (C++ extension).
*/
CallbackResult GoContextCallback(int callbackId) {
std::string ctx;
std::string ret;
auto& server = AIKIDO_GLOBAL(server);
const auto& requestCache = AIKIDO_GLOBAL(requestCache);
try {
switch (callbackId) {
case CONTEXT_REMOTE_ADDRESS:
ctx = "REMOTE_ADDRESS";
ret = server.GetVar("REMOTE_ADDR");
break;
case CONTEXT_METHOD:
ctx = "METHOD";
ret = server.GetMethod();
break;
case CONTEXT_ROUTE:
ctx = "ROUTE";
ret = server.GetRoute();
break;
case CONTEXT_STATUS_CODE:
ctx = "STATUS_CODE";
ret = server.GetStatusCode();
break;
case CONTEXT_BODY:
ctx = "BODY";
ret = server.GetBody();
break;
case CONTEXT_HEADER_X_FORWARDED_FOR:
ctx = "HEADER_X_FORWARDED_FOR";
ret = server.GetVar("HTTP_X_FORWARDED_FOR");
break;
case CONTEXT_COOKIES:
ctx = "COOKIES";
ret = server.GetVar("HTTP_COOKIE");
break;
case CONTEXT_QUERY:
ctx = "QUERY";
ret = server.GetQuery();
break;
case CONTEXT_HTTPS:
ctx = "HTTPS";
ret = server.GetVar("HTTPS");
break;
case CONTEXT_URL:
ctx = "URL";
ret = server.GetUrl();
break;
case CONTEXT_HEADERS:
ctx = "HEADERS";
ret = server.GetHeaders();
break;
case CONTEXT_HEADER_USER_AGENT:
ctx = "USER_AGENT";
ret = server.GetVar("HTTP_USER_AGENT");
break;
case CONTEXT_USER_ID:
ctx = "USER_ID";
ret = requestCache.userId;
break;
case CONTEXT_USER_NAME:
ctx = "USER_NAME";
ret = requestCache.userName;
break;
case CONTEXT_RATE_LIMIT_GROUP:
ctx = "RATE_LIMIT_GROUP";
ret = requestCache.rateLimitGroup;
break;
case FUNCTION_NAME:
ctx = "FUNCTION_NAME";
ret = GetEventCacheField(&EventCache::functionName);
break;
case OUTGOING_REQUEST_URL:
ctx = "OUTGOING_REQUEST_URL";
ret = GetEventCacheField(&EventCache::outgoingRequestUrl);
break;
case OUTGOING_REQUEST_EFFECTIVE_URL:
ctx = "OUTGOING_REQUEST_EFFECTIVE_URL";
ret = GetEventCacheField(&EventCache::outgoingRequestEffectiveUrl);
break;
case OUTGOING_REQUEST_PORT:
ctx = "OUTGOING_REQUEST_PORT";
ret = GetEventCacheField(&EventCache::outgoingRequestPort);
break;
case OUTGOING_REQUEST_EFFECTIVE_URL_PORT:
ctx = "OUTGOING_REQUEST_EFFECTIVE_URL_PORT";
ret = GetEventCacheField(&EventCache::outgoingRequestEffectiveUrlPort);
break;
case OUTGOING_REQUEST_RESOLVED_IP:
ctx = "OUTGOING_REQUEST_RESOLVED_IP";
ret = GetEventCacheField(&EventCache::outgoingRequestResolvedIp);
break;
case CMD:
ctx = "CMD";
ret = GetEventCacheField(&EventCache::cmd);
break;
case FILENAME:
ctx = "FILENAME";
ret = GetEventCacheField(&EventCache::filename);
break;
case FILENAME2:
ctx = "FILENAME2";
ret = GetEventCacheField(&EventCache::filename2);
break;
case SQL_QUERY:
ctx = "SQL_QUERY";
ret = GetEventCacheField(&EventCache::sqlQuery);
break;
case SQL_DIALECT:
ctx = "SQL_DIALECT";
ret = GetEventCacheField(&EventCache::sqlDialect);
break;
case MODULE:
ctx = "MODULE";
ret = GetEventCacheField(&EventCache::moduleName);
break;
case STACK_TRACE:
ctx = "STACK_TRACE";
ret = GetStackTrace();
break;
case PARAM_MATCHER_PARAM:
ctx = "PARAM_MATCHER_PARAM";
ret = GetEventCacheField(&EventCache::paramMatcherParam);
break;
case PARAM_MATCHER_REGEX:
ctx = "PARAM_MATCHER_REGEX";
ret = GetEventCacheField(&EventCache::paramMatcherRegex);
break;
}
} catch (std::exception& e) {
AIKIDO_LOG_DEBUG("Exception in GoContextCallback: %s\n", e.what());
}
if (!ret.length()) {
AIKIDO_LOG_DEBUG("Callback %s -> NULL\n", ctx.c_str());
return CallbackResult{nullptr, 0};
}
if (ret.length() > 10000) {
AIKIDO_LOG_DEBUG("Callback %s -> (Result too large to print)\n", ctx.c_str());
} else {
AIKIDO_LOG_DEBUG("Callback %s -> %s\n", ctx.c_str(), ret.c_str());
}
int len = static_cast<int>(ret.length());
char *buf = static_cast<char *>(malloc(len));
if (!buf) {
AIKIDO_LOG_WARN("Failed to allocate %d bytes in GoContextCallback\n", len);
return CallbackResult{nullptr, 0};
}
memcpy(buf, ret.data(), len);
return CallbackResult{buf, len};
}