mirror of
https://github.com/springzfx/cgproxy.git
synced 2026-04-23 10:11:04 +08:00
add execsnoop in c++
This commit is contained in:
@@ -24,7 +24,6 @@ install(FILES cgnoproxy DESTINATION /usr/bin PERMISSIONS ${basic_permission})
|
|||||||
install(FILES cgproxy.service DESTINATION /usr/lib/systemd/system/)
|
install(FILES cgproxy.service DESTINATION /usr/lib/systemd/system/)
|
||||||
install(FILES config.json DESTINATION /etc/cgproxy/)
|
install(FILES config.json DESTINATION /etc/cgproxy/)
|
||||||
install(FILES cgroup-tproxy.sh DESTINATION /usr/share/cgproxy/scripts/ PERMISSIONS ${basic_permission})
|
install(FILES cgroup-tproxy.sh DESTINATION /usr/share/cgproxy/scripts/ PERMISSIONS ${basic_permission})
|
||||||
install(FILES execsnoop.py DESTINATION /usr/share/cgproxy/scripts/ PERMISSIONS ${basic_permission})
|
|
||||||
install(FILES readme.md DESTINATION /usr/share/doc/cgproxy/)
|
install(FILES readme.md DESTINATION /usr/share/doc/cgproxy/)
|
||||||
|
|
||||||
# man pages
|
# man pages
|
||||||
|
|||||||
130
execsnoop.py
130
execsnoop.py
@@ -1,130 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
# This won't catch all new processes: an application may fork() but not exec().
|
|
||||||
|
|
||||||
from __future__ import print_function
|
|
||||||
import os, sys, signal, shutil
|
|
||||||
def eprint(*args, **kwargs):
|
|
||||||
print(*args, file=sys.stderr, **kwargs)
|
|
||||||
|
|
||||||
try:
|
|
||||||
from bcc import BPF
|
|
||||||
except:
|
|
||||||
eprint("python-bcc not installed")
|
|
||||||
exit(0)
|
|
||||||
|
|
||||||
# define BPF program
|
|
||||||
bpf_text = """
|
|
||||||
#include <uapi/linux/ptrace.h>
|
|
||||||
#include <linux/sched.h>
|
|
||||||
#include <linux/fs.h>
|
|
||||||
|
|
||||||
struct data_t {
|
|
||||||
u32 pid;
|
|
||||||
};
|
|
||||||
|
|
||||||
BPF_PERF_OUTPUT(events);
|
|
||||||
|
|
||||||
int syscall__execve(struct pt_regs *ctx,
|
|
||||||
const char __user *filename,
|
|
||||||
const char __user *const __user *__argv,
|
|
||||||
const char __user *const __user *__envp)
|
|
||||||
{
|
|
||||||
struct data_t data = {};
|
|
||||||
data.pid = bpf_get_current_pid_tgid() >> 32;
|
|
||||||
events.perf_submit(ctx, &data, sizeof(struct data_t));
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
|
|
||||||
def getRealPath(exec_path):
|
|
||||||
path=exec_path.strip()
|
|
||||||
if not path.startswith("/"):
|
|
||||||
path=shutil.which(path)
|
|
||||||
if path:
|
|
||||||
path=os.path.realpath(path)
|
|
||||||
if os.path.isfile(path): return path
|
|
||||||
eprint("'{0}' not exist or broken link".format(exec_path))
|
|
||||||
|
|
||||||
def getParam():
|
|
||||||
global exec_path_proxy, exec_path_noproxy
|
|
||||||
exec_path_str=os.getenv('program_proxy')
|
|
||||||
if exec_path_str:
|
|
||||||
paths=exec_path_str.split(':')
|
|
||||||
exec_path_proxy=[getRealPath(x) for x in paths]
|
|
||||||
print("program with proxy:", end =" ")
|
|
||||||
print(*exec_path_proxy,flush=True)
|
|
||||||
|
|
||||||
exec_path_str=os.getenv('program_noproxy')
|
|
||||||
if exec_path_str:
|
|
||||||
paths=exec_path_str.split(':')
|
|
||||||
exec_path_noproxy=[getRealPath(x) for x in paths]
|
|
||||||
print("program without proxy:", end =" ")
|
|
||||||
print(*exec_path_noproxy, flush=True)
|
|
||||||
|
|
||||||
def exit_gracefully(signum, frame):
|
|
||||||
eprint("execsnoop receive signal: {0}".format(signum),flush=True)
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
def attach(pid, path, proxy=True):
|
|
||||||
if proxy:
|
|
||||||
print("proxy: %-6d %s" % (pid, path),flush=True)
|
|
||||||
os.system("/usr/bin/cgproxy --pid {0}".format(pid))
|
|
||||||
else:
|
|
||||||
print("noproxy: %-6d %s" % (pid, path),flush=True)
|
|
||||||
os.system("/usr/bin/cgproxy --pid {0} --noproxy".format(pid))
|
|
||||||
|
|
||||||
def processAlreadyRunning():
|
|
||||||
from subprocess import check_output
|
|
||||||
def get_pid(name):
|
|
||||||
try:
|
|
||||||
return map(int,check_output(["pidof",name]).split())
|
|
||||||
except:
|
|
||||||
return []
|
|
||||||
global exec_path_proxy, exec_path_noproxy
|
|
||||||
for path in exec_path_proxy:
|
|
||||||
for pid in get_pid(path):
|
|
||||||
attach(pid,path,True)
|
|
||||||
for path in exec_path_noproxy:
|
|
||||||
for pid in get_pid(path):
|
|
||||||
attach(pid,path,False)
|
|
||||||
|
|
||||||
signal.signal(signal.SIGINT, exit_gracefully)
|
|
||||||
signal.signal(signal.SIGHUP, exit_gracefully)
|
|
||||||
signal.signal(signal.SIGTERM, exit_gracefully)
|
|
||||||
|
|
||||||
debug=False
|
|
||||||
if (len(sys.argv)>1 and sys.argv[1]=="--debug"):
|
|
||||||
debug=True
|
|
||||||
|
|
||||||
exec_path_proxy=[]
|
|
||||||
exec_path_noproxy=[]
|
|
||||||
getParam()
|
|
||||||
processAlreadyRunning()
|
|
||||||
|
|
||||||
# initialize BPF
|
|
||||||
b = BPF(text=bpf_text)
|
|
||||||
execve_fnname = b.get_syscall_fnname("execve")
|
|
||||||
b.attach_kprobe(event=execve_fnname, fn_name="syscall__execve")
|
|
||||||
|
|
||||||
# process event
|
|
||||||
def print_event(cpu, data, size):
|
|
||||||
event = b["events"].event(data)
|
|
||||||
pid=event.pid
|
|
||||||
try:
|
|
||||||
exec_path=os.readlink("/proc/{0}/exe".format(pid))
|
|
||||||
except: # in case process exit too early
|
|
||||||
if (debug):
|
|
||||||
print("process exit too early: {0}".format(pid))
|
|
||||||
return
|
|
||||||
if (exec_path in exec_path_noproxy):
|
|
||||||
attach(pid, exec_path, False)
|
|
||||||
elif (exec_path in exec_path_proxy):
|
|
||||||
attach(pid, exec_path, True)
|
|
||||||
elif (debug):
|
|
||||||
print("debug: %d %s" % (pid, exec_path),flush=True)
|
|
||||||
|
|
||||||
|
|
||||||
# loop with callback to print_event
|
|
||||||
b["events"].open_perf_buffer(print_event)
|
|
||||||
while 1:
|
|
||||||
b.perf_buffer_poll()
|
|
||||||
@@ -7,7 +7,7 @@ add_executable(main main.cpp
|
|||||||
common.cpp config.cpp cgroup_attach.cpp
|
common.cpp config.cpp cgroup_attach.cpp
|
||||||
socket_client.cpp socket_server.cpp)
|
socket_client.cpp socket_server.cpp)
|
||||||
|
|
||||||
target_link_libraries(main nlohmann_json::nlohmann_json Threads::Threads)
|
target_link_libraries(main nlohmann_json::nlohmann_json Threads::Threads bcc)
|
||||||
set_target_properties(main PROPERTIES LINKER_LANGUAGE CXX)
|
set_target_properties(main PROPERTIES LINKER_LANGUAGE CXX)
|
||||||
set_target_properties(main PROPERTIES OUTPUT_NAME cgproxy)
|
set_target_properties(main PROPERTIES OUTPUT_NAME cgproxy)
|
||||||
|
|
||||||
|
|||||||
114
src/cgproxyd.hpp
114
src/cgproxyd.hpp
@@ -4,10 +4,14 @@
|
|||||||
#include "cgroup_attach.h"
|
#include "cgroup_attach.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
#include "execsnoop.hpp"
|
||||||
#include "socket_server.h"
|
#include "socket_server.h"
|
||||||
|
#include <algorithm>
|
||||||
#include <csignal>
|
#include <csignal>
|
||||||
|
#include <cstdlib>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <nlohmann/json.hpp>
|
#include <nlohmann/json.hpp>
|
||||||
|
#include <pthread.h>
|
||||||
#include <sched.h>
|
#include <sched.h>
|
||||||
#include <sys/file.h>
|
#include <sys/file.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
@@ -17,17 +21,22 @@ using json = nlohmann::json;
|
|||||||
using namespace ::CGPROXY::SOCKET;
|
using namespace ::CGPROXY::SOCKET;
|
||||||
using namespace ::CGPROXY::CONFIG;
|
using namespace ::CGPROXY::CONFIG;
|
||||||
using namespace ::CGPROXY::CGROUP;
|
using namespace ::CGPROXY::CGROUP;
|
||||||
|
using namespace ::CGPROXY::EXESNOOP;
|
||||||
|
|
||||||
namespace CGPROXY::CGPROXYD {
|
namespace CGPROXY::CGPROXYD {
|
||||||
|
|
||||||
bool print_help = false;
|
bool print_help = false;
|
||||||
|
bool enable_socketserver = true;
|
||||||
bool enable_execsnoop = false;
|
bool enable_execsnoop = false;
|
||||||
|
|
||||||
class cgproxyd {
|
class cgproxyd {
|
||||||
thread_arg arg_t;
|
SOCKET::thread_arg socketserver_thread_arg;
|
||||||
Config config;
|
|
||||||
pthread_t socket_thread_id = -1;
|
pthread_t socket_thread_id = -1;
|
||||||
pid_t exec_snoop_pid = -1;
|
|
||||||
|
EXESNOOP::thread_arg execsnoop_thread_arg;
|
||||||
|
pthread_t execsnoop_thread_id = -1;
|
||||||
|
|
||||||
|
Config config;
|
||||||
|
|
||||||
static cgproxyd *instance;
|
static cgproxyd *instance;
|
||||||
static int handle_msg_static(char *msg) {
|
static int handle_msg_static(char *msg) {
|
||||||
@@ -38,6 +47,40 @@ class cgproxyd {
|
|||||||
return instance->handle_msg(msg);
|
return instance->handle_msg(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int handle_pid_static(int pid) {
|
||||||
|
if (!instance) {
|
||||||
|
error("no cgproxyd instance assigned");
|
||||||
|
return ERROR;
|
||||||
|
}
|
||||||
|
return instance->handle_pid(pid);
|
||||||
|
}
|
||||||
|
|
||||||
|
int handle_pid(int pid) {
|
||||||
|
auto path=realpath(to_str("/proc/",pid,"/exe").c_str(), NULL);
|
||||||
|
if (path==NULL) {
|
||||||
|
debug("pid %d live life too short", pid);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
debug("execsnoop: %d %s", pid, path);
|
||||||
|
|
||||||
|
vector<string> v;
|
||||||
|
|
||||||
|
v = config.program_noproxy;
|
||||||
|
if (find(v.begin(), v.end(), path) != v.end()) {
|
||||||
|
info("exesnoop noproxy: %d %s", pid, path);
|
||||||
|
free(path);
|
||||||
|
return attach(pid, config.cgroup_noproxy_preserved);
|
||||||
|
}
|
||||||
|
v = config.program_proxy;
|
||||||
|
if (find(v.begin(), v.end(), path) != v.end()) {
|
||||||
|
info("exesnoop proxied: %d %s", pid, path);
|
||||||
|
free(path);
|
||||||
|
return attach(pid, config.cgroup_proxy_preserved);
|
||||||
|
}
|
||||||
|
free(path);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static void signalHandler(int signum) {
|
static void signalHandler(int signum) {
|
||||||
debug("Signal %d received.", signum);
|
debug("Signal %d received.", signum);
|
||||||
if (!instance) {
|
if (!instance) {
|
||||||
@@ -85,12 +128,12 @@ class cgproxyd {
|
|||||||
switch (type) {
|
switch (type) {
|
||||||
case MSG_TYPE_CONFIG_JSON:
|
case MSG_TYPE_CONFIG_JSON:
|
||||||
status = config.loadFromJsonStr(j.at("data").dump());
|
status = config.loadFromJsonStr(j.at("data").dump());
|
||||||
if (status == SUCCESS) status = applyConfig(&config);
|
if (status == SUCCESS) status = applyConfig();
|
||||||
return status;
|
return status;
|
||||||
break;
|
break;
|
||||||
case MSG_TYPE_CONFIG_PATH:
|
case MSG_TYPE_CONFIG_PATH:
|
||||||
status = config.loadFromFile(j.at("data").get<string>());
|
status = config.loadFromFile(j.at("data").get<string>());
|
||||||
if (status == SUCCESS) status = applyConfig(&config);
|
if (status == SUCCESS) status = applyConfig();
|
||||||
return status;
|
return status;
|
||||||
break;
|
break;
|
||||||
case MSG_TYPE_PROXY_PID:
|
case MSG_TYPE_PROXY_PID:
|
||||||
@@ -111,27 +154,35 @@ class cgproxyd {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pthread_t startSocketListeningThread() {
|
pthread_t startSocketListeningThread() {
|
||||||
arg_t.handle_msg = &handle_msg_static;
|
socketserver_thread_arg.handle_msg = &handle_msg_static;
|
||||||
pthread_t thread_id;
|
pthread_t thread_id;
|
||||||
int status = pthread_create(&thread_id, NULL, &SocketServer::startThread, &arg_t);
|
int status =
|
||||||
|
pthread_create(&thread_id, NULL, &SOCKET::startThread, &socketserver_thread_arg);
|
||||||
if (status != 0) error("socket thread create failed");
|
if (status != 0) error("socket thread create failed");
|
||||||
return thread_id;
|
return thread_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
void startExecSnoopProc() {
|
pthread_t startExecSnoopThread() {
|
||||||
if (exec_snoop_pid != -1){
|
execsnoop_thread_arg.handle_pid = &handle_pid_static;
|
||||||
kill(exec_snoop_pid, SIGINT);
|
pthread_t thread_id;
|
||||||
exec_snoop_pid=-1;
|
int status =
|
||||||
}
|
pthread_create(&thread_id, NULL, &EXESNOOP::startThread, &execsnoop_thread_arg);
|
||||||
pid_t pid = fork();
|
if (status != 0) error("execsnoop thread create failed");
|
||||||
if (pid == 0) {
|
return thread_id;
|
||||||
execl(BPF_EXEC_SNOOP_START, (char *) NULL);
|
}
|
||||||
exit(0);
|
|
||||||
} else if (pid<0){
|
void processRunningProgram(){
|
||||||
error("fork precess failed");
|
debug("process running program")
|
||||||
}else {
|
for (auto &path:config.program_noproxy)
|
||||||
exec_snoop_pid = pid;
|
for (auto &pid:bash_pidof(path)){
|
||||||
}
|
int status=attach(pid, config.cgroup_noproxy_preserved);
|
||||||
|
if (status==0) info("noproxy running process %d %s",pid, path.c_str());
|
||||||
|
}
|
||||||
|
for (auto &path:config.program_proxy)
|
||||||
|
for (auto &pid:bash_pidof(path)){
|
||||||
|
int status=attach(pid, config.cgroup_proxy_preserved);
|
||||||
|
if (status==0) info("proxied running process %d %s",pid, path.c_str());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void assignStaticInstance() { instance = this; }
|
void assignStaticInstance() { instance = this; }
|
||||||
@@ -143,19 +194,26 @@ public:
|
|||||||
signal(SIGTERM, &signalHandler);
|
signal(SIGTERM, &signalHandler);
|
||||||
signal(SIGHUP, &signalHandler);
|
signal(SIGHUP, &signalHandler);
|
||||||
|
|
||||||
config.loadFromFile(DEFAULT_CONFIG_FILE);
|
|
||||||
applyConfig(&config);
|
|
||||||
|
|
||||||
assignStaticInstance();
|
assignStaticInstance();
|
||||||
socket_thread_id = startSocketListeningThread();
|
|
||||||
|
config.loadFromFile(DEFAULT_CONFIG_FILE);
|
||||||
|
applyConfig();
|
||||||
|
processRunningProgram();
|
||||||
|
|
||||||
|
if (enable_socketserver) { socket_thread_id = startSocketListeningThread(); }
|
||||||
|
if (enable_execsnoop) { execsnoop_thread_id = startExecSnoopThread(); }
|
||||||
|
|
||||||
|
cout<<flush;
|
||||||
|
|
||||||
pthread_join(socket_thread_id, NULL);
|
pthread_join(socket_thread_id, NULL);
|
||||||
|
pthread_join(execsnoop_thread_id, NULL);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
int applyConfig(Config *c) {
|
int applyConfig() {
|
||||||
system(TPROXY_IPTABLS_CLEAN);
|
system(TPROXY_IPTABLS_CLEAN);
|
||||||
c->toEnv();
|
config.print_summary();
|
||||||
|
config.toEnv();
|
||||||
system(TPROXY_IPTABLS_START);
|
system(TPROXY_IPTABLS_START);
|
||||||
if (enable_execsnoop) startExecSnoopProc();
|
|
||||||
// no need to track running status
|
// no need to track running status
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,48 +13,46 @@
|
|||||||
|
|
||||||
namespace CGPROXY::CGROUP {
|
namespace CGPROXY::CGROUP {
|
||||||
|
|
||||||
|
string cgroup2_mount_point = get_cgroup2_mount_point();
|
||||||
|
|
||||||
bool exist(string path) {
|
bool exist(string path) {
|
||||||
struct stat st;
|
struct stat st;
|
||||||
if (stat(path.c_str(), &st) != -1) { return S_ISDIR(st.st_mode); }
|
if (stat(path.c_str(), &st) != -1) { return S_ISDIR(st.st_mode); }
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
string get_cgroup2_mount_point() {
|
||||||
|
stringstream buffer;
|
||||||
|
FILE *fp = popen("findmnt -t cgroup2 -n -o TARGET", "r");
|
||||||
|
if (!fp) return "";
|
||||||
|
char buf[64]; while (fgets(buf,64,fp)!=NULL) { buffer<<buf; }
|
||||||
|
pclose(fp);
|
||||||
|
string s=buffer.str();
|
||||||
|
s.pop_back(); // remove newline character
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
bool validate(string pid, string cgroup) {
|
bool validate(string pid, string cgroup) {
|
||||||
bool pid_v = validPid(pid);
|
bool pid_v = validPid(pid);
|
||||||
bool cg_v = validCgroup(cgroup);
|
bool cg_v = validCgroup(cgroup);
|
||||||
if (pid_v && cg_v) return true;
|
if (pid_v && cg_v) return true;
|
||||||
|
|
||||||
error("attach paramater validate error");
|
error("attach paramater validate error");
|
||||||
return_error
|
return_error;
|
||||||
}
|
|
||||||
|
|
||||||
string get_cgroup2_mount_point(int &status) {
|
|
||||||
char cgroup2_mount_point[100] = "";
|
|
||||||
FILE *fp = popen("findmnt -t cgroup2 -n -o TARGET", "r");
|
|
||||||
int count = fscanf(fp, "%s", cgroup2_mount_point);
|
|
||||||
fclose(fp);
|
|
||||||
if (count == 0) {
|
|
||||||
error("cgroup2 not supported");
|
|
||||||
status = -1;
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
status = 0;
|
|
||||||
return cgroup2_mount_point;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int attach(const string pid, const string cgroup_target) {
|
int attach(const string pid, const string cgroup_target) {
|
||||||
if (getuid() != 0) {
|
if (getuid() != 0) {
|
||||||
error("need root to attach cgroup");
|
error("need root to attach cgroup");
|
||||||
return_error
|
return_error;
|
||||||
}
|
}
|
||||||
|
|
||||||
debug("attaching %s to %s", pid.c_str(), cgroup_target.c_str());
|
debug("attaching %s to %s", pid.c_str(), cgroup_target.c_str());
|
||||||
|
|
||||||
int status;
|
int status;
|
||||||
if (!validate(pid, cgroup_target))
|
if (!validate(pid, cgroup_target)) return_error;
|
||||||
return_error string cgroup_mount_point = get_cgroup2_mount_point(status);
|
if (cgroup2_mount_point.empty()) return_error;
|
||||||
if (status != 0)
|
string cgroup_target_path = cgroup2_mount_point + cgroup_target;
|
||||||
return_error string cgroup_target_path = cgroup_mount_point + cgroup_target;
|
|
||||||
string cgroup_target_procs = cgroup_target_path + "/cgroup.procs";
|
string cgroup_target_procs = cgroup_target_path + "/cgroup.procs";
|
||||||
|
|
||||||
// check if exist, we will create it if not exist
|
// check if exist, we will create it if not exist
|
||||||
@@ -64,7 +62,7 @@ int attach(const string pid, const string cgroup_target) {
|
|||||||
debug("created cgroup %s success", cgroup_target.c_str());
|
debug("created cgroup %s success", cgroup_target.c_str());
|
||||||
} else {
|
} else {
|
||||||
error("created cgroup %s failed, errno %d", cgroup_target.c_str(), errno);
|
error("created cgroup %s failed, errno %d", cgroup_target.c_str(), errno);
|
||||||
return_error
|
return_error;
|
||||||
}
|
}
|
||||||
// error("cgroup %s not exist",cgroup_target.c_str());
|
// error("cgroup %s not exist",cgroup_target.c_str());
|
||||||
// return_error
|
// return_error
|
||||||
@@ -74,7 +72,7 @@ int attach(const string pid, const string cgroup_target) {
|
|||||||
ofstream procs(cgroup_target_procs, ofstream::app);
|
ofstream procs(cgroup_target_procs, ofstream::app);
|
||||||
if (!procs.is_open()) {
|
if (!procs.is_open()) {
|
||||||
error("open file %s failed", cgroup_target_procs.c_str());
|
error("open file %s failed", cgroup_target_procs.c_str());
|
||||||
return_error
|
return_error;
|
||||||
}
|
}
|
||||||
procs << pid.c_str() << endl;
|
procs << pid.c_str() << endl;
|
||||||
procs.close();
|
procs.close();
|
||||||
@@ -83,9 +81,9 @@ int attach(const string pid, const string cgroup_target) {
|
|||||||
if (!procs) {
|
if (!procs) {
|
||||||
error("write %s to %s failed, maybe process %s not exist", pid.c_str(),
|
error("write %s to %s failed, maybe process %s not exist", pid.c_str(),
|
||||||
cgroup_target_procs.c_str(), pid.c_str());
|
cgroup_target_procs.c_str(), pid.c_str());
|
||||||
return_error
|
return_error;
|
||||||
}
|
}
|
||||||
return_success
|
return_success;
|
||||||
}
|
}
|
||||||
|
|
||||||
int attach(const int pid, const string cgroup_target) {
|
int attach(const int pid, const string cgroup_target) {
|
||||||
|
|||||||
@@ -6,10 +6,10 @@
|
|||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
namespace CGPROXY::CGROUP {
|
namespace CGPROXY::CGROUP {
|
||||||
|
extern string cgroup2_mount_point;
|
||||||
bool exist(string path);
|
bool exist(string path);
|
||||||
bool validate(string pid, string cgroup);
|
bool validate(string pid, string cgroup);
|
||||||
string get_cgroup2_mount_point(int &status);
|
string get_cgroup2_mount_point();
|
||||||
int attach(const string pid, const string cgroup_target);
|
int attach(const string pid, const string cgroup_target);
|
||||||
int attach(const int pid, const string cgroup_target);
|
int attach(const int pid, const string cgroup_target);
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,13 @@
|
|||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
#include <fstream>
|
||||||
|
#include <linux/limits.h>
|
||||||
#include <regex>
|
#include <regex>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <limits.h>
|
||||||
|
|
||||||
bool enable_debug = false;
|
bool enable_debug = false;
|
||||||
|
bool enable_info = true;
|
||||||
|
|
||||||
string join2str(const vector<string> t, const char delm) {
|
string join2str(const vector<string> t, const char delm) {
|
||||||
string s;
|
string s;
|
||||||
@@ -32,3 +38,55 @@ bool validCgroup(const vector<string> cgroup) {
|
|||||||
bool validPid(const string pid) { return regex_match(pid, regex("^[0-9]+$")); }
|
bool validPid(const string pid) { return regex_match(pid, regex("^[0-9]+$")); }
|
||||||
|
|
||||||
bool validPort(const int port) { return port > 0; }
|
bool validPort(const int port) { return port > 0; }
|
||||||
|
|
||||||
|
bool fileExist(const string &path) {
|
||||||
|
struct stat st;
|
||||||
|
return (stat(path.c_str(), &st) == 0 && S_ISREG(st.st_mode));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool dirExist(const string &path) {
|
||||||
|
struct stat st;
|
||||||
|
return (stat(path.c_str(), &st) == 0 && S_ISDIR(st.st_mode));
|
||||||
|
}
|
||||||
|
|
||||||
|
vector<int> bash_pidof(const string &path){
|
||||||
|
vector<int> pids;
|
||||||
|
FILE *fp = popen(to_str("pidof ", path).c_str(), "r");
|
||||||
|
if (!fp) return pids;
|
||||||
|
int pid;
|
||||||
|
char buf[64]; while (fscanf(fp,"%d",&pid)!=EOF) { pids.push_back(pid); }
|
||||||
|
pclose(fp);
|
||||||
|
return pids;
|
||||||
|
}
|
||||||
|
|
||||||
|
string bash_which(const string &name) {
|
||||||
|
stringstream buffer;
|
||||||
|
FILE *fp = popen(to_str("which ", name).c_str(), "r");
|
||||||
|
if (!fp) return "";
|
||||||
|
char buf[64]; while (fgets(buf,64,fp)!=NULL) { buffer<<buf; }
|
||||||
|
pclose(fp);
|
||||||
|
string s=buffer.str();
|
||||||
|
s.pop_back(); // remove newline character
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
string bash_readlink(const string &path) {
|
||||||
|
stringstream buffer;
|
||||||
|
FILE *fp = popen(to_str("readlink -e ", path).c_str(), "r");
|
||||||
|
if (!fp) return "";
|
||||||
|
char buf[64]; while (fgets(buf,64,fp)!=NULL) { buffer<<buf; }
|
||||||
|
pclose(fp);
|
||||||
|
string s=buffer.str();
|
||||||
|
s.pop_back(); // remove newline character
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
string getRealExistPath(const string &name) {
|
||||||
|
if (name[0] == '/' && fileExist(name)) return name;
|
||||||
|
string path;
|
||||||
|
path = bash_which(name);
|
||||||
|
if (path.empty()) return "";
|
||||||
|
path = bash_readlink(path);
|
||||||
|
if (!fileExist(path)) return "";
|
||||||
|
return path;
|
||||||
|
}
|
||||||
22
src/common.h
22
src/common.h
@@ -9,7 +9,6 @@ using namespace std;
|
|||||||
|
|
||||||
#define TPROXY_IPTABLS_START "/usr/share/cgproxy/scripts/cgroup-tproxy.sh"
|
#define TPROXY_IPTABLS_START "/usr/share/cgproxy/scripts/cgroup-tproxy.sh"
|
||||||
#define TPROXY_IPTABLS_CLEAN "/usr/share/cgproxy/scripts/cgroup-tproxy.sh stop"
|
#define TPROXY_IPTABLS_CLEAN "/usr/share/cgproxy/scripts/cgroup-tproxy.sh stop"
|
||||||
#define BPF_EXEC_SNOOP_START "/usr/share/cgproxy/scripts/execsnoop.py"
|
|
||||||
|
|
||||||
#define PID_LOCK_FILE "/var/run/cgproxyd.pid"
|
#define PID_LOCK_FILE "/var/run/cgproxyd.pid"
|
||||||
#define SOCKET_PATH "/tmp/cgproxy_unix_socket"
|
#define SOCKET_PATH "/tmp/cgproxy_unix_socket"
|
||||||
@@ -36,6 +35,7 @@ using namespace std;
|
|||||||
#define FILE_ERROR 7
|
#define FILE_ERROR 7
|
||||||
|
|
||||||
extern bool enable_debug;
|
extern bool enable_debug;
|
||||||
|
extern bool enable_info;
|
||||||
|
|
||||||
#define error(...) \
|
#define error(...) \
|
||||||
{ \
|
{ \
|
||||||
@@ -46,13 +46,20 @@ extern bool enable_debug;
|
|||||||
|
|
||||||
#define debug(...) \
|
#define debug(...) \
|
||||||
if (enable_debug) { \
|
if (enable_debug) { \
|
||||||
fprintf(stderr, "debug: "); \
|
fprintf(stdout, "debug: "); \
|
||||||
fprintf(stdout, __VA_ARGS__); \
|
fprintf(stdout, __VA_ARGS__); \
|
||||||
fprintf(stdout, "\n"); \
|
fprintf(stdout, "\n"); \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define return_error return -1;
|
#define info(...) \
|
||||||
#define return_success return 0;
|
if (enable_info) { \
|
||||||
|
fprintf(stdout, "info: "); \
|
||||||
|
fprintf(stdout, __VA_ARGS__); \
|
||||||
|
fprintf(stdout, "\n"); \
|
||||||
|
}
|
||||||
|
|
||||||
|
#define return_error return -1
|
||||||
|
#define return_success return 0
|
||||||
|
|
||||||
template <typename... T> string to_str(T... args) {
|
template <typename... T> string to_str(T... args) {
|
||||||
stringstream ss;
|
stringstream ss;
|
||||||
@@ -70,4 +77,11 @@ bool validCgroup(const vector<string> cgroup);
|
|||||||
bool validPid(const string pid);
|
bool validPid(const string pid);
|
||||||
bool validPort(const int port);
|
bool validPort(const int port);
|
||||||
|
|
||||||
|
bool fileExist(const string &path);
|
||||||
|
bool dirExist(const string &path);
|
||||||
|
vector<int> bash_pidof(const string &path);
|
||||||
|
string bash_which(const string &name);
|
||||||
|
string bash_readlink(const string &path);
|
||||||
|
string getRealExistPath(const string &name);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
@@ -4,6 +4,7 @@
|
|||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
#include <nlohmann/json.hpp>
|
#include <nlohmann/json.hpp>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
#include <vector>
|
||||||
using json = nlohmann::json;
|
using json = nlohmann::json;
|
||||||
|
|
||||||
#define add2json(v) j[#v] = v;
|
#define add2json(v) j[#v] = v;
|
||||||
@@ -89,6 +90,11 @@ int Config::loadFromJsonStr(const string js) {
|
|||||||
tryassign(enable_udp);
|
tryassign(enable_udp);
|
||||||
tryassign(enable_ipv4);
|
tryassign(enable_ipv4);
|
||||||
tryassign(enable_ipv6);
|
tryassign(enable_ipv6);
|
||||||
|
|
||||||
|
// e.g. v2ray -> /usr/bin/v2ray -> /usr/lib/v2ray/v2ray
|
||||||
|
toRealProgramPath(program_noproxy);
|
||||||
|
toRealProgramPath(program_proxy);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -127,4 +133,21 @@ bool Config::validateJsonStr(const string js) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Config::print_summary() {
|
||||||
|
info("noproxy program: %s", join2str(program_noproxy).c_str());
|
||||||
|
info("proxied program: %s", join2str(program_proxy).c_str());
|
||||||
|
info("noproxy cgroup: %s", join2str(cgroup_noproxy).c_str());
|
||||||
|
info("proxied cgroup: %s", join2str(cgroup_proxy).c_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
void Config::toRealProgramPath(vector<string> &v) {
|
||||||
|
vector<string> tmp;
|
||||||
|
for (auto &p : v) {
|
||||||
|
auto rpath = getRealExistPath(p);
|
||||||
|
if (!rpath.empty()) tmp.push_back(rpath);
|
||||||
|
else error("%s not exist or broken link", p.c_str());
|
||||||
|
}
|
||||||
|
v=tmp;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace CGPROXY::CONFIG
|
} // namespace CGPROXY::CONFIG
|
||||||
@@ -30,10 +30,12 @@ public:
|
|||||||
string toJsonStr();
|
string toJsonStr();
|
||||||
int loadFromFile(const string f);
|
int loadFromFile(const string f);
|
||||||
int loadFromJsonStr(const string js);
|
int loadFromJsonStr(const string js);
|
||||||
|
void print_summary();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void mergeReserved();
|
void mergeReserved();
|
||||||
bool validateJsonStr(const string js);
|
bool validateJsonStr(const string js);
|
||||||
|
void toRealProgramPath(vector<string> &v);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace CGPROXY::CONFIG
|
} // namespace CGPROXY::CONFIG
|
||||||
|
|||||||
111
src/execsnoop.hpp
Normal file
111
src/execsnoop.hpp
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
#ifndef EXECSNOOP_HPP
|
||||||
|
#define EXECSNOOP_HPP 1
|
||||||
|
|
||||||
|
#include <exception>
|
||||||
|
#include "bcc/BPF.h"
|
||||||
|
#include "common.h"
|
||||||
|
#include <bcc/libbpf.h>
|
||||||
|
#include <fstream>
|
||||||
|
#include <functional>
|
||||||
|
#include <iostream>
|
||||||
|
#include <string>
|
||||||
|
#include <unistd.h>
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
namespace CGPROXY::EXESNOOP {
|
||||||
|
|
||||||
|
const string BPF_PROGRAM = R"(
|
||||||
|
#include <linux/fs.h>
|
||||||
|
#include <linux/sched.h>
|
||||||
|
#include <uapi/linux/ptrace.h>
|
||||||
|
|
||||||
|
struct data_t {
|
||||||
|
int pid;
|
||||||
|
};
|
||||||
|
|
||||||
|
BPF_PERF_OUTPUT(events);
|
||||||
|
|
||||||
|
int syscall_execve(struct pt_regs *ctx,
|
||||||
|
const char __user *filename,
|
||||||
|
const char __user *const __user *__argv,
|
||||||
|
const char __user *const __user *__envp)
|
||||||
|
{
|
||||||
|
struct data_t data = {};
|
||||||
|
data.pid = bpf_get_current_pid_tgid();
|
||||||
|
events.perf_submit(ctx, &data, sizeof(struct data_t));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ret_syscall_execve(struct pt_regs *ctx){
|
||||||
|
struct data_t data = {};
|
||||||
|
data.pid = bpf_get_current_pid_tgid();
|
||||||
|
int retval = PT_REGS_RC(ctx);
|
||||||
|
if (retval==0)
|
||||||
|
events.perf_submit(ctx, &data, sizeof(struct data_t));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
|
||||||
|
struct data_t {
|
||||||
|
int pid;
|
||||||
|
};
|
||||||
|
|
||||||
|
function<int(int)> callback = NULL;
|
||||||
|
|
||||||
|
void handle_events(void *cb_cookie, void *data, int data_size) {
|
||||||
|
auto event = static_cast<data_t *>(data);
|
||||||
|
int pid = event->pid;
|
||||||
|
|
||||||
|
if (callback) callback(pid);
|
||||||
|
}
|
||||||
|
|
||||||
|
int execsnoop() {
|
||||||
|
debug("starting execsnoop");
|
||||||
|
ebpf::BPF bpf;
|
||||||
|
auto init_res = bpf.init(BPF_PROGRAM);
|
||||||
|
if (init_res.code() != 0) {
|
||||||
|
std::cerr << init_res.msg() << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
string execve_fnname = bpf.get_syscall_fnname("execve");
|
||||||
|
// auto attach_res = bpf.attach_kprobe(execve_fnname, "syscall_execve");
|
||||||
|
auto attach_res =
|
||||||
|
bpf.attach_kprobe(execve_fnname, "ret_syscall_execve", 0, BPF_PROBE_RETURN);
|
||||||
|
if (attach_res.code() != 0) {
|
||||||
|
std::cerr << attach_res.msg() << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto open_res = bpf.open_perf_buffer("events", &handle_events);
|
||||||
|
if (open_res.code() != 0) {
|
||||||
|
std::cerr << open_res.msg() << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bpf.free_bcc_memory()) {
|
||||||
|
std::cerr << "Failed to free llvm/clang memory" << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (true) bpf.poll_perf_buffer("events");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct thread_arg {
|
||||||
|
function<int(int)> handle_pid;
|
||||||
|
};
|
||||||
|
|
||||||
|
void *startThread(void *arg) {
|
||||||
|
thread_arg *p = (thread_arg *)arg;
|
||||||
|
callback = p->handle_pid;
|
||||||
|
try {
|
||||||
|
execsnoop();
|
||||||
|
} catch (exception &e) {
|
||||||
|
error("bcc may not be installed, %s",e.what());
|
||||||
|
}
|
||||||
|
return (void *)0;
|
||||||
|
}
|
||||||
|
} // namespace CGPROXY::EXESNOOP
|
||||||
|
#endif
|
||||||
@@ -49,7 +49,7 @@ void SocketServer::socketListening(function<int(char *)> callback) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void *SocketServer::startThread(void *arg) {
|
void *startThread(void *arg) {
|
||||||
thread_arg *p = (thread_arg *)arg;
|
thread_arg *p = (thread_arg *)arg;
|
||||||
SocketServer server;
|
SocketServer server;
|
||||||
server.socketListening(p->handle_msg);
|
server.socketListening(p->handle_msg);
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ namespace CGPROXY::SOCKET {
|
|||||||
struct thread_arg {
|
struct thread_arg {
|
||||||
function<int(char *)> handle_msg;
|
function<int(char *)> handle_msg;
|
||||||
};
|
};
|
||||||
|
void *startThread(void *arg);
|
||||||
|
|
||||||
class SocketServer {
|
class SocketServer {
|
||||||
public:
|
public:
|
||||||
@@ -25,7 +26,6 @@ public:
|
|||||||
|
|
||||||
void socketListening(function<int(char *)> callback);
|
void socketListening(function<int(char *)> callback);
|
||||||
~SocketServer();
|
~SocketServer();
|
||||||
static void *startThread(void *arg);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace CGPROXY::SOCKET
|
} // namespace CGPROXY::SOCKET
|
||||||
|
|||||||
@@ -1,4 +1,9 @@
|
|||||||
include_directories(${PROJECT_SOURCE_DIR})
|
include_directories(${PROJECT_SOURCE_DIR})
|
||||||
include_directories(${PROJECT_SOURCE_DIR}/src)
|
include_directories(${PROJECT_SOURCE_DIR}/src)
|
||||||
add_executable(cgattach cgattach.cpp ../src/cgroup_attach.cpp ../src/common.cpp)
|
|
||||||
install(TARGETS cgattach DESTINATION /usr/bin PERMISSIONS ${basic_permission})
|
add_executable(cgattach cgattach.cpp ../src/cgroup_attach.cpp ../src/common.cpp)
|
||||||
|
install(TARGETS cgattach DESTINATION /usr/bin PERMISSIONS ${basic_permission})
|
||||||
|
|
||||||
|
add_executable(execsnoop execsnoop.cpp ../src/common.cpp)
|
||||||
|
target_link_libraries(execsnoop bcc)
|
||||||
|
install(TARGETS execsnoop DESTINATION /usr/bin PERMISSIONS ${basic_permission})
|
||||||
23
tools/execsnoop.cpp
Normal file
23
tools/execsnoop.cpp
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
#include "execsnoop.hpp"
|
||||||
|
#include "common.h"
|
||||||
|
using namespace std;
|
||||||
|
using namespace CGPROXY::EXESNOOP;
|
||||||
|
|
||||||
|
#define PATH_MAX_LEN 128
|
||||||
|
|
||||||
|
int handle_pid(int pid) {
|
||||||
|
char path[PATH_MAX_LEN];
|
||||||
|
auto size = readlink(to_str("/proc/", pid, "/exe").c_str(), path, PATH_MAX_LEN);
|
||||||
|
if (size == -1) error("readlink: %s", to_str("/proc/", pid, "/exe").c_str());
|
||||||
|
path[size] = '\0';
|
||||||
|
info("%d %s", pid, path);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
enable_debug = true;
|
||||||
|
enable_info = true;
|
||||||
|
callback = handle_pid;
|
||||||
|
execsnoop();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user