Compare commits
5 commits
a2fc2bf2bc
...
1b77c0daa6
| Author | SHA1 | Date | |
|---|---|---|---|
|
1b77c0daa6 |
|||
|
a4eedcbc26 |
|||
|
d1af86078e |
|||
|
3651624861 |
|||
|
54f858aee9 |
6 changed files with 1408 additions and 297 deletions
24
Cargo.lock
generated
24
Cargo.lock
generated
|
|
@ -636,6 +636,7 @@ dependencies = [
|
||||||
"rlua",
|
"rlua",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
|
"tempfile",
|
||||||
"tokio",
|
"tokio",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
@ -646,9 +647,15 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "976dd42dc7e85965fe702eb8164f21f450704bdde31faefd6471dba214cb594e"
|
checksum = "976dd42dc7e85965fe702eb8164f21f450704bdde31faefd6471dba214cb594e"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"libc",
|
"libc",
|
||||||
"windows-sys 0.52.0",
|
"windows-sys 0.59.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "fastrand"
|
||||||
|
version = "2.3.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "37909eebbb50d72f9059c3b6d82c0463f2ff062c9e95845c43a6c9c0355411be"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "flate2"
|
name = "flate2"
|
||||||
version = "1.1.1"
|
version = "1.1.1"
|
||||||
|
|
@ -1580,7 +1587,7 @@ dependencies = [
|
||||||
"errno",
|
"errno",
|
||||||
"libc",
|
"libc",
|
||||||
"linux-raw-sys",
|
"linux-raw-sys",
|
||||||
"windows-sys 0.52.0",
|
"windows-sys 0.59.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|
@ -1740,6 +1747,19 @@ dependencies = [
|
||||||
"syn 2.0.101",
|
"syn 2.0.101",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "tempfile"
|
||||||
|
version = "3.19.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7437ac7763b9b123ccf33c338a5cc1bac6f69b45a136c19bdd8a65e3916435bf"
|
||||||
|
dependencies = [
|
||||||
|
"fastrand",
|
||||||
|
"getrandom",
|
||||||
|
"once_cell",
|
||||||
|
"rustix",
|
||||||
|
"windows-sys 0.59.0",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "thiserror"
|
name = "thiserror"
|
||||||
version = "1.0.69"
|
version = "1.0.69"
|
||||||
|
|
|
||||||
|
|
@ -19,3 +19,4 @@ serde_json = "1.0.96"
|
||||||
tokio = { version = "1.28.0", features = ["full"] }
|
tokio = { version = "1.28.0", features = ["full"] }
|
||||||
log = "0.4.27"
|
log = "0.4.27"
|
||||||
env_logger = "0.11.8"
|
env_logger = "0.11.8"
|
||||||
|
tempfile = "3.19.1"
|
||||||
|
|
|
||||||
210
resources/default_script.lua
Normal file
210
resources/default_script.lua
Normal file
|
|
@ -0,0 +1,210 @@
|
||||||
|
--[[
|
||||||
|
Eris Default Script
|
||||||
|
|
||||||
|
This script demonstrates how to use the Eris Lua API to customize
|
||||||
|
the tarpit's behavior, and will be loaded by default if no other
|
||||||
|
scripts are loaded.
|
||||||
|
|
||||||
|
Available events:
|
||||||
|
- connection: When a new connection is established
|
||||||
|
- request: When a request is received
|
||||||
|
- response_gen: When generating a response
|
||||||
|
- response_chunk: Before sending each response chunk
|
||||||
|
- disconnection: When a connection is closed
|
||||||
|
- block_ip: When an IP is being considered for blocking
|
||||||
|
- startup: When the application starts
|
||||||
|
- shutdown: When the application is shutting down
|
||||||
|
- periodic: Called periodically
|
||||||
|
|
||||||
|
API Functions:
|
||||||
|
- eris.debug(message): Log a debug message
|
||||||
|
- eris.info(message): Log an info message
|
||||||
|
- eris.warn(message): Log a warning message
|
||||||
|
- eris.error(message): Log an error message
|
||||||
|
- eris.set_state(key, value): Store persistent state
|
||||||
|
- eris.get_state(key): Retrieve persistent state
|
||||||
|
- eris.inc_counter(key, [amount]): Increment a counter
|
||||||
|
- eris.get_counter(key): Get a counter value
|
||||||
|
- eris.gen_token([prefix]): Generate a unique token
|
||||||
|
- eris.timestamp(): Get current Unix timestamp
|
||||||
|
--]]
|
||||||
|
|
||||||
|
-- Called when the application starts
|
||||||
|
eris.on("startup", function(ctx)
|
||||||
|
eris.info("Initializing default script")
|
||||||
|
|
||||||
|
-- Initialize counters
|
||||||
|
eris.inc_counter("total_connections", 0)
|
||||||
|
eris.inc_counter("total_responses", 0)
|
||||||
|
eris.inc_counter("blocked_ips", 0)
|
||||||
|
|
||||||
|
-- Initialize banned keywords
|
||||||
|
eris.set_state("banned_keywords", "eval,exec,system,shell,<?php,/bin/bash")
|
||||||
|
end)
|
||||||
|
|
||||||
|
-- Called for each new connection
|
||||||
|
eris.on("connection", function(ctx)
|
||||||
|
eris.inc_counter("total_connections")
|
||||||
|
eris.debug("New connection from " .. ctx.ip)
|
||||||
|
|
||||||
|
-- You can reject connections by returning false
|
||||||
|
-- This example checks a blocklist
|
||||||
|
local blocklist = eris.get_state("manual_blocklist") or ""
|
||||||
|
if blocklist:find(ctx.ip) then
|
||||||
|
eris.info("Rejecting connection from manually blocked IP: " .. ctx.ip)
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
return true -- accept the connection
|
||||||
|
end)
|
||||||
|
|
||||||
|
-- Called when generating a response
|
||||||
|
eris.on("response_gen", function(ctx)
|
||||||
|
eris.inc_counter("total_responses")
|
||||||
|
|
||||||
|
-- Generate a unique traceable token for this request
|
||||||
|
local token = eris.gen_token("ERIS-")
|
||||||
|
|
||||||
|
-- Add some believable but fake honeytokens based on the request path
|
||||||
|
local enhanced_content = ctx.content
|
||||||
|
|
||||||
|
if ctx.path:find("wp%-") then
|
||||||
|
-- For WordPress paths
|
||||||
|
enhanced_content = enhanced_content
|
||||||
|
.. "\n<!-- WordPress Debug: "
|
||||||
|
.. token
|
||||||
|
.. " -->"
|
||||||
|
.. "\n<!-- WP_HOME: http://stop.crawlingmysite.com/wordpress -->"
|
||||||
|
.. "\n<!-- DB_USER: wp_user_"
|
||||||
|
.. math.random(1000, 9999)
|
||||||
|
.. " -->"
|
||||||
|
elseif ctx.path:find("phpunit") or ctx.path:find("eval") then
|
||||||
|
-- For PHP exploit attempts
|
||||||
|
-- Turns out you can just google "PHP error log" and search random online forums where people
|
||||||
|
-- dump their service logs in full.
|
||||||
|
enhanced_content = enhanced_content
|
||||||
|
.. "\nPHP Notice: Undefined variable: _SESSION in /var/www/html/includes/core.php on line 58\n"
|
||||||
|
.. "Warning: file_get_contents(): Filename cannot be empty in /var/www/html/vendor/autoload.php on line 23\n"
|
||||||
|
.. "Token: "
|
||||||
|
.. token
|
||||||
|
.. "\n"
|
||||||
|
elseif ctx.path:find("api") then
|
||||||
|
-- For API requests
|
||||||
|
local fake_api_key =
|
||||||
|
string.format("ak_%x%x%x", math.random(1000, 9999), math.random(1000, 9999), math.random(1000, 9999))
|
||||||
|
|
||||||
|
enhanced_content = enhanced_content
|
||||||
|
.. "{\n"
|
||||||
|
.. ' "status": "warning",\n'
|
||||||
|
.. ' "message": "Test API environment detected",\n'
|
||||||
|
.. ' "debug_token": "'
|
||||||
|
.. token
|
||||||
|
.. '",\n'
|
||||||
|
.. ' "api_key": "'
|
||||||
|
.. fake_api_key
|
||||||
|
.. '"\n'
|
||||||
|
.. "}\n"
|
||||||
|
else
|
||||||
|
-- For other requests
|
||||||
|
enhanced_content = enhanced_content
|
||||||
|
.. "\n<!-- Server: Apache/2.4.41 (Ubuntu) -->"
|
||||||
|
.. "\n<!-- Debug-Token: "
|
||||||
|
.. token
|
||||||
|
.. " -->"
|
||||||
|
.. "\n<!-- Environment: staging -->"
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Track which honeytokens were sent to which IP
|
||||||
|
local honeytokens = eris.get_state("honeytokens") or "{}"
|
||||||
|
local ht_table = {}
|
||||||
|
|
||||||
|
-- This is a simplistic approach - in a real script, you'd want to use
|
||||||
|
-- a proper JSON library to handle this correctly
|
||||||
|
if honeytokens ~= "{}" then
|
||||||
|
-- Simple parsing of the stored data
|
||||||
|
for ip, tok in honeytokens:gmatch('"([^"]+)":"([^"]+)"') do
|
||||||
|
ht_table[ip] = tok
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
ht_table[ctx.ip] = token
|
||||||
|
|
||||||
|
-- Convert back to a simple JSON-like string
|
||||||
|
local new_tokens = "{"
|
||||||
|
for ip, tok in pairs(ht_table) do
|
||||||
|
if new_tokens ~= "{" then
|
||||||
|
new_tokens = new_tokens .. ","
|
||||||
|
end
|
||||||
|
new_tokens = new_tokens .. '"' .. ip .. '":"' .. tok .. '"'
|
||||||
|
end
|
||||||
|
new_tokens = new_tokens .. "}"
|
||||||
|
|
||||||
|
eris.set_state("honeytokens", new_tokens)
|
||||||
|
|
||||||
|
return enhanced_content
|
||||||
|
end)
|
||||||
|
|
||||||
|
-- Called before sending each chunk of a response
|
||||||
|
eris.on("response_chunk", function(ctx)
|
||||||
|
-- This can be used to alter individual chunks for more deceptive behavior
|
||||||
|
-- For example, to simulate a slow, unreliable server
|
||||||
|
|
||||||
|
-- 5% chance of "corrupting" a chunk to confuse scanners
|
||||||
|
if math.random(1, 100) <= 5 then
|
||||||
|
local chunk = ctx.content
|
||||||
|
if #chunk > 10 then
|
||||||
|
local pos = math.random(1, #chunk - 5)
|
||||||
|
chunk = chunk:sub(1, pos) .. string.char(math.random(32, 126)) .. chunk:sub(pos + 2)
|
||||||
|
end
|
||||||
|
return chunk
|
||||||
|
end
|
||||||
|
|
||||||
|
return ctx.content
|
||||||
|
end)
|
||||||
|
|
||||||
|
-- Called when deciding whether to block an IP
|
||||||
|
eris.on("block_ip", function(ctx)
|
||||||
|
-- You can override the default blocking logic
|
||||||
|
|
||||||
|
-- Check for potential attackers using specific patterns
|
||||||
|
local banned_keywords = eris.get_state("banned_keywords") or ""
|
||||||
|
local user_agent = ctx.user_agent or ""
|
||||||
|
|
||||||
|
-- Check if user agent contains highly suspicious patterns
|
||||||
|
for keyword in banned_keywords:gmatch("[^,]+") do
|
||||||
|
if user_agent:lower():find(keyword:lower()) then
|
||||||
|
eris.info("Blocking IP " .. ctx.ip .. " due to suspicious user agent: " .. keyword)
|
||||||
|
eris.inc_counter("blocked_ips")
|
||||||
|
return true -- Force block
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- For demonstration, we'll be more lenient with 10.x IPs
|
||||||
|
if ctx.ip:match("^10%.") then
|
||||||
|
-- Only block if they've hit us many times
|
||||||
|
return ctx.hit_count >= 5
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Default to the system's threshold-based decision
|
||||||
|
return nil
|
||||||
|
end)
|
||||||
|
|
||||||
|
-- The enhance_response is now legacy, and I never liked it anyway. Though let's add it here
|
||||||
|
-- for the sake of backwards compatibility.
|
||||||
|
function enhance_response(text, response_type, path, token)
|
||||||
|
local enhanced = text
|
||||||
|
|
||||||
|
-- Add token as a comment
|
||||||
|
if response_type == "php_exploit" then
|
||||||
|
enhanced = enhanced .. "\n/* Token: " .. token .. " */\n"
|
||||||
|
elseif response_type == "wordpress" then
|
||||||
|
enhanced = enhanced .. "\n<!-- WordPress Debug Token: " .. token .. " -->\n"
|
||||||
|
elseif response_type == "api" then
|
||||||
|
enhanced = enhanced:gsub('"status": "[^"]+"', '"status": "warning"')
|
||||||
|
enhanced = enhanced:gsub('"message": "[^"]+"', '"message": "API token: ' .. token .. '"')
|
||||||
|
else
|
||||||
|
enhanced = enhanced .. "\n<!-- Debug token: " .. token .. " -->\n"
|
||||||
|
end
|
||||||
|
|
||||||
|
return enhanced
|
||||||
|
end
|
||||||
901
src/lua/mod.rs
Normal file
901
src/lua/mod.rs
Normal file
|
|
@ -0,0 +1,901 @@
|
||||||
|
use rlua::{Function, Lua, Table, Value};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::collections::hash_map::DefaultHasher;
|
||||||
|
use std::fs;
|
||||||
|
use std::hash::{Hash, Hasher};
|
||||||
|
use std::path::Path;
|
||||||
|
use std::sync::{Arc, Mutex, RwLock};
|
||||||
|
use std::time::{SystemTime, UNIX_EPOCH};
|
||||||
|
|
||||||
|
// Event types for the Lua scripting system
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
||||||
|
pub enum EventType {
|
||||||
|
Connection, // when a new connection is established
|
||||||
|
Request, // when a request is received
|
||||||
|
ResponseGen, // when generating a response
|
||||||
|
ResponseChunk, // before sending each response chunk
|
||||||
|
Disconnection, // when a connection is closed
|
||||||
|
BlockIP, // when an IP is being considered for blocking
|
||||||
|
Startup, // when the application starts
|
||||||
|
Shutdown, // when the application is shutting down
|
||||||
|
Periodic, // called periodically (e.g., every minute)
|
||||||
|
}
|
||||||
|
|
||||||
|
impl EventType {
|
||||||
|
/// Convert event type to string representation for Lua
|
||||||
|
const fn as_str(&self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
Self::Connection => "connection",
|
||||||
|
Self::Request => "request",
|
||||||
|
Self::ResponseGen => "response_gen",
|
||||||
|
Self::ResponseChunk => "response_chunk",
|
||||||
|
Self::Disconnection => "disconnection",
|
||||||
|
Self::BlockIP => "block_ip",
|
||||||
|
Self::Startup => "startup",
|
||||||
|
Self::Shutdown => "shutdown",
|
||||||
|
Self::Periodic => "periodic",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Convert from string to `EventType`
|
||||||
|
fn from_str(s: &str) -> Option<Self> {
|
||||||
|
match s {
|
||||||
|
"connection" => Some(Self::Connection),
|
||||||
|
"request" => Some(Self::Request),
|
||||||
|
"response_gen" => Some(Self::ResponseGen),
|
||||||
|
"response_chunk" => Some(Self::ResponseChunk),
|
||||||
|
"disconnection" => Some(Self::Disconnection),
|
||||||
|
"block_ip" => Some(Self::BlockIP),
|
||||||
|
"startup" => Some(Self::Startup),
|
||||||
|
"shutdown" => Some(Self::Shutdown),
|
||||||
|
"periodic" => Some(Self::Periodic),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Loaded Lua script with its metadata
|
||||||
|
struct ScriptInfo {
|
||||||
|
name: String,
|
||||||
|
enabled: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Script state and manage the Lua environment
|
||||||
|
pub struct ScriptManager {
|
||||||
|
lua: Mutex<Lua>,
|
||||||
|
scripts: Vec<ScriptInfo>,
|
||||||
|
hooks: HashMap<EventType, Vec<String>>,
|
||||||
|
state: Arc<RwLock<HashMap<String, String>>>,
|
||||||
|
counters: Arc<RwLock<HashMap<String, i64>>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Context passed to Lua event handlers
|
||||||
|
pub struct EventContext {
|
||||||
|
pub event_type: EventType,
|
||||||
|
pub ip: Option<String>,
|
||||||
|
pub path: Option<String>,
|
||||||
|
pub user_agent: Option<String>,
|
||||||
|
pub request_headers: Option<HashMap<String, String>>,
|
||||||
|
pub content: Option<String>,
|
||||||
|
pub timestamp: u64,
|
||||||
|
pub session_id: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make ScriptManager explicitly Send + Sync since we're using Mutex
|
||||||
|
unsafe impl Send for ScriptManager {}
|
||||||
|
unsafe impl Sync for ScriptManager {}
|
||||||
|
|
||||||
|
impl ScriptManager {
|
||||||
|
/// Create a new script manager and load scripts from the given directory
|
||||||
|
pub fn new(scripts_dir: &str) -> Self {
|
||||||
|
let mut manager = Self {
|
||||||
|
lua: Mutex::new(Lua::new()),
|
||||||
|
scripts: Vec::new(),
|
||||||
|
hooks: HashMap::new(),
|
||||||
|
state: Arc::new(RwLock::new(HashMap::new())),
|
||||||
|
counters: Arc::new(RwLock::new(HashMap::new())),
|
||||||
|
};
|
||||||
|
|
||||||
|
// Initialize Lua environment with our API
|
||||||
|
manager.init_lua_env();
|
||||||
|
|
||||||
|
// Load scripts from directory
|
||||||
|
manager.load_scripts_from_dir(scripts_dir);
|
||||||
|
|
||||||
|
// If no scripts were loaded, use default script
|
||||||
|
if manager.scripts.is_empty() {
|
||||||
|
log::info!("No Lua scripts found, loading default scripts");
|
||||||
|
manager.load_script(
|
||||||
|
"default",
|
||||||
|
include_str!("../../resources/default_script.lua"),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Trigger startup event
|
||||||
|
manager.trigger_event(&EventContext {
|
||||||
|
event_type: EventType::Startup,
|
||||||
|
ip: None,
|
||||||
|
path: None,
|
||||||
|
user_agent: None,
|
||||||
|
request_headers: None,
|
||||||
|
content: None,
|
||||||
|
timestamp: get_timestamp(),
|
||||||
|
session_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
manager
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize the Lua environment
|
||||||
|
fn init_lua_env(&self) {
|
||||||
|
let state_clone = self.state.clone();
|
||||||
|
let counters_clone = self.counters.clone();
|
||||||
|
|
||||||
|
if let Ok(lua) = self.lua.lock() {
|
||||||
|
// Create eris global table for our API
|
||||||
|
let eris_table = lua.create_table().unwrap();
|
||||||
|
|
||||||
|
self.register_utility_functions(&lua, &eris_table, state_clone, counters_clone);
|
||||||
|
self.register_event_functions(&lua, &eris_table);
|
||||||
|
self.register_logging_functions(&lua, &eris_table);
|
||||||
|
|
||||||
|
// Set the eris global table
|
||||||
|
lua.globals().set("eris", eris_table).unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Register utility functions for scripts to use
|
||||||
|
fn register_utility_functions(
|
||||||
|
&self,
|
||||||
|
lua: &Lua,
|
||||||
|
eris_table: &Table,
|
||||||
|
state: Arc<RwLock<HashMap<String, String>>>,
|
||||||
|
counters: Arc<RwLock<HashMap<String, i64>>>,
|
||||||
|
) {
|
||||||
|
// Store a key-value pair in persistent state
|
||||||
|
let state_for_set = state.clone();
|
||||||
|
let set_state = lua
|
||||||
|
.create_function(move |_, (key, value): (String, String)| {
|
||||||
|
let mut state_map = state_for_set.write().unwrap();
|
||||||
|
state_map.insert(key, value);
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("set_state", set_state).unwrap();
|
||||||
|
|
||||||
|
// Get a value from persistent state
|
||||||
|
let state_for_get = state;
|
||||||
|
let get_state = lua
|
||||||
|
.create_function(move |_, key: String| {
|
||||||
|
let state_map = state_for_get.read().unwrap();
|
||||||
|
let value = state_map.get(&key).cloned();
|
||||||
|
Ok(value)
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("get_state", get_state).unwrap();
|
||||||
|
|
||||||
|
// Increment a counter
|
||||||
|
let counters_for_inc = counters.clone();
|
||||||
|
let inc_counter = lua
|
||||||
|
.create_function(move |_, (key, amount): (String, Option<i64>)| {
|
||||||
|
let mut counters_map = counters_for_inc.write().unwrap();
|
||||||
|
let counter = counters_map.entry(key).or_insert(0);
|
||||||
|
*counter += amount.unwrap_or(1);
|
||||||
|
Ok(*counter)
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("inc_counter", inc_counter).unwrap();
|
||||||
|
|
||||||
|
// Get a counter value
|
||||||
|
let counters_for_get = counters;
|
||||||
|
let get_counter = lua
|
||||||
|
.create_function(move |_, key: String| {
|
||||||
|
let counters_map = counters_for_get.read().unwrap();
|
||||||
|
let value = counters_map.get(&key).copied().unwrap_or(0);
|
||||||
|
Ok(value)
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("get_counter", get_counter).unwrap();
|
||||||
|
|
||||||
|
// Generate a random token/string
|
||||||
|
let gen_token = lua
|
||||||
|
.create_function(move |_, prefix: Option<String>| {
|
||||||
|
let now = get_timestamp();
|
||||||
|
let random = rand::random::<u32>();
|
||||||
|
let token = format!("{}{:x}{:x}", prefix.unwrap_or_default(), now, random);
|
||||||
|
Ok(token)
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("gen_token", gen_token).unwrap();
|
||||||
|
|
||||||
|
// Get current timestamp
|
||||||
|
let timestamp = lua
|
||||||
|
.create_function(move |_, ()| Ok(get_timestamp()))
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("timestamp", timestamp).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Register event handling functions
|
||||||
|
fn register_event_functions(&self, lua: &Lua, eris_table: &Table) {
|
||||||
|
// Create a table to store event handlers
|
||||||
|
let handlers_table = lua.create_table().unwrap();
|
||||||
|
eris_table.set("handlers", handlers_table).unwrap();
|
||||||
|
|
||||||
|
// Function for scripts to register event handlers
|
||||||
|
let on_fn = lua
|
||||||
|
.create_function(move |lua, (event_name, handler): (String, Function)| {
|
||||||
|
let globals = lua.globals();
|
||||||
|
let eris: Table = globals.get("eris").unwrap();
|
||||||
|
let handlers: Table = eris.get("handlers").unwrap();
|
||||||
|
|
||||||
|
// Get or create a table for this event type
|
||||||
|
let event_handlers: Table = if let Ok(table) = handlers.get(&*event_name) {
|
||||||
|
table
|
||||||
|
} else {
|
||||||
|
let new_table = lua.create_table().unwrap();
|
||||||
|
handlers.set(&*event_name, new_table.clone()).unwrap();
|
||||||
|
new_table
|
||||||
|
};
|
||||||
|
|
||||||
|
// Add the handler to the table
|
||||||
|
let next_index = event_handlers.len().unwrap() + 1;
|
||||||
|
event_handlers.set(next_index, handler).unwrap();
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("on", on_fn).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Register logging functions
|
||||||
|
fn register_logging_functions(&self, lua: &Lua, eris_table: &Table) {
|
||||||
|
// Debug logging
|
||||||
|
let debug = lua
|
||||||
|
.create_function(|_, message: String| {
|
||||||
|
log::debug!("[Lua] {message}");
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("debug", debug).unwrap();
|
||||||
|
|
||||||
|
// Info logging
|
||||||
|
let info = lua
|
||||||
|
.create_function(|_, message: String| {
|
||||||
|
log::info!("[Lua] {message}");
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("info", info).unwrap();
|
||||||
|
|
||||||
|
// Warning logging
|
||||||
|
let warn = lua
|
||||||
|
.create_function(|_, message: String| {
|
||||||
|
log::warn!("[Lua] {message}");
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("warn", warn).unwrap();
|
||||||
|
|
||||||
|
// Error logging
|
||||||
|
let error = lua
|
||||||
|
.create_function(|_, message: String| {
|
||||||
|
log::error!("[Lua] {message}");
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
eris_table.set("error", error).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load all scripts from a directory
|
||||||
|
fn load_scripts_from_dir(&mut self, scripts_dir: &str) {
|
||||||
|
let script_dir = Path::new(scripts_dir);
|
||||||
|
if !script_dir.exists() {
|
||||||
|
log::warn!("Lua scripts directory does not exist: {scripts_dir}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
log::debug!("Loading Lua scripts from directory: {scripts_dir}");
|
||||||
|
if let Ok(entries) = fs::read_dir(script_dir) {
|
||||||
|
// Sort entries by filename to ensure consistent loading order
|
||||||
|
let mut sorted_entries: Vec<_> = entries.filter_map(Result::ok).collect();
|
||||||
|
sorted_entries.sort_by_key(std::fs::DirEntry::path);
|
||||||
|
|
||||||
|
for entry in sorted_entries {
|
||||||
|
let path = entry.path();
|
||||||
|
if path.extension().and_then(|ext| ext.to_str()) == Some("lua") {
|
||||||
|
if let Ok(content) = fs::read_to_string(&path) {
|
||||||
|
let script_name = path
|
||||||
|
.file_stem()
|
||||||
|
.and_then(|n| n.to_str())
|
||||||
|
.unwrap_or("unknown")
|
||||||
|
.to_string();
|
||||||
|
|
||||||
|
log::debug!("Loading Lua script: {} ({})", script_name, path.display());
|
||||||
|
self.load_script(&script_name, &content);
|
||||||
|
} else {
|
||||||
|
log::warn!("Failed to read Lua script: {}", path.display());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load a single script and register its event handlers
|
||||||
|
fn load_script(&mut self, name: &str, content: &str) {
|
||||||
|
// Store script info
|
||||||
|
self.scripts.push(ScriptInfo {
|
||||||
|
name: name.to_string(),
|
||||||
|
enabled: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Execute the script to register its event handlers
|
||||||
|
if let Ok(lua) = self.lua.lock() {
|
||||||
|
if let Err(e) = lua.load(content).set_name(name).exec() {
|
||||||
|
log::warn!("Error loading Lua script '{name}': {e}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collect registered event handlers
|
||||||
|
let globals = lua.globals();
|
||||||
|
let eris: Table = match globals.get("eris") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => return,
|
||||||
|
};
|
||||||
|
|
||||||
|
let handlers: Table = match eris.get("handlers") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => return,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Store the event handlers in our hooks map
|
||||||
|
let mut tmp: rlua::TablePairs<'_, String, Table<'_>> =
|
||||||
|
handlers.pairs::<String, Table>();
|
||||||
|
'l: loop {
|
||||||
|
if let Some(event_pair) = tmp.next() {
|
||||||
|
if let Ok((event_name, _)) = event_pair {
|
||||||
|
if let Some(event_type) = EventType::from_str(&event_name) {
|
||||||
|
self.hooks
|
||||||
|
.entry(event_type)
|
||||||
|
.or_default()
|
||||||
|
.push(name.to_string());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
break 'l;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log::info!("Loaded Lua script '{name}' successfully");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Check if a script is enabled
|
||||||
|
fn is_script_enabled(&self, name: &str) -> bool {
|
||||||
|
self.scripts
|
||||||
|
.iter()
|
||||||
|
.find(|s| s.name == name)
|
||||||
|
.is_some_and(|s| s.enabled)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Trigger an event, calling all registered handlers
|
||||||
|
pub fn trigger_event(&self, ctx: &EventContext) -> Option<String> {
|
||||||
|
// Check if we have any handlers for this event
|
||||||
|
if !self.hooks.contains_key(&ctx.event_type) {
|
||||||
|
return ctx.content.clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build the event data table to pass to Lua handlers
|
||||||
|
let mut result = ctx.content.clone();
|
||||||
|
|
||||||
|
if let Ok(lua) = self.lua.lock() {
|
||||||
|
// Create the event context table
|
||||||
|
let event_ctx = lua.create_table().unwrap();
|
||||||
|
|
||||||
|
// Add all the context fields
|
||||||
|
event_ctx.set("event", ctx.event_type.as_str()).unwrap();
|
||||||
|
if let Some(ip) = &ctx.ip {
|
||||||
|
event_ctx.set("ip", ip.clone()).unwrap();
|
||||||
|
}
|
||||||
|
if let Some(path) = &ctx.path {
|
||||||
|
event_ctx.set("path", path.clone()).unwrap();
|
||||||
|
}
|
||||||
|
if let Some(ua) = &ctx.user_agent {
|
||||||
|
event_ctx.set("user_agent", ua.clone()).unwrap();
|
||||||
|
}
|
||||||
|
event_ctx.set("timestamp", ctx.timestamp).unwrap();
|
||||||
|
if let Some(sid) = &ctx.session_id {
|
||||||
|
event_ctx.set("session_id", sid.clone()).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add request headers if available
|
||||||
|
if let Some(headers) = &ctx.request_headers {
|
||||||
|
let headers_table = lua.create_table().unwrap();
|
||||||
|
for (key, value) in headers {
|
||||||
|
headers_table
|
||||||
|
.set(key.to_string(), value.to_string())
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
event_ctx.set("headers", headers_table).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add content if available
|
||||||
|
if let Some(content) = &ctx.content {
|
||||||
|
event_ctx.set("content", content.clone()).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call all registered handlers for this event
|
||||||
|
if let Some(handler_scripts) = self.hooks.get(&ctx.event_type) {
|
||||||
|
for script_name in handler_scripts {
|
||||||
|
// Skip disabled scripts
|
||||||
|
if !self.is_script_enabled(script_name) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the globals and handlers table
|
||||||
|
let globals = lua.globals();
|
||||||
|
let eris: Table = match globals.get("eris") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
let handlers: Table = match eris.get("handlers") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Get handlers for this event
|
||||||
|
let event_handlers: Table = match handlers.get(ctx.event_type.as_str()) {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Call each handler
|
||||||
|
for pair in event_handlers.pairs::<i64, Function>() {
|
||||||
|
if let Ok((_, handler)) = pair {
|
||||||
|
let handler_result: rlua::Result<Option<String>> =
|
||||||
|
handler.call((event_ctx.clone(),));
|
||||||
|
if let Ok(Some(new_content)) = handler_result {
|
||||||
|
// For response events, allow handlers to modify the content
|
||||||
|
if matches!(
|
||||||
|
ctx.event_type,
|
||||||
|
EventType::ResponseGen | EventType::ResponseChunk
|
||||||
|
) {
|
||||||
|
result = Some(new_content);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Generate a deceptive response, calling all response_gen handlers
|
||||||
|
pub fn generate_response(
|
||||||
|
&self,
|
||||||
|
path: &str,
|
||||||
|
user_agent: &str,
|
||||||
|
ip: &str,
|
||||||
|
headers: &HashMap<String, String>,
|
||||||
|
markov_text: &str,
|
||||||
|
) -> String {
|
||||||
|
// Create event context
|
||||||
|
let ctx = EventContext {
|
||||||
|
event_type: EventType::ResponseGen,
|
||||||
|
ip: Some(ip.to_string()),
|
||||||
|
path: Some(path.to_string()),
|
||||||
|
user_agent: Some(user_agent.to_string()),
|
||||||
|
request_headers: Some(headers.clone()),
|
||||||
|
content: Some(markov_text.to_string()),
|
||||||
|
timestamp: get_timestamp(),
|
||||||
|
session_id: Some(generate_session_id(ip, user_agent)),
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Trigger the event and get the modified content
|
||||||
|
self.trigger_event(&ctx).unwrap_or_else(|| {
|
||||||
|
// Fallback to maintain backward compatibility
|
||||||
|
self.expand_response(
|
||||||
|
markov_text,
|
||||||
|
"generic",
|
||||||
|
path,
|
||||||
|
&generate_session_id(ip, user_agent),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Process a chunk before sending it to client
|
||||||
|
pub fn process_chunk(&self, chunk: &str, ip: &str, session_id: &str) -> String {
|
||||||
|
let ctx = EventContext {
|
||||||
|
event_type: EventType::ResponseChunk,
|
||||||
|
ip: Some(ip.to_string()),
|
||||||
|
path: None,
|
||||||
|
user_agent: None,
|
||||||
|
request_headers: None,
|
||||||
|
content: Some(chunk.to_string()),
|
||||||
|
timestamp: get_timestamp(),
|
||||||
|
session_id: Some(session_id.to_string()),
|
||||||
|
};
|
||||||
|
|
||||||
|
self.trigger_event(&ctx)
|
||||||
|
.unwrap_or_else(|| chunk.to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Called when a connection is established
|
||||||
|
pub fn on_connection(&self, ip: &str) -> bool {
|
||||||
|
let ctx = EventContext {
|
||||||
|
event_type: EventType::Connection,
|
||||||
|
ip: Some(ip.to_string()),
|
||||||
|
path: None,
|
||||||
|
user_agent: None,
|
||||||
|
request_headers: None,
|
||||||
|
content: None,
|
||||||
|
timestamp: get_timestamp(),
|
||||||
|
session_id: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
// If any handler returns false, reject the connection
|
||||||
|
let mut should_accept = true;
|
||||||
|
|
||||||
|
if let Ok(lua) = self.lua.lock() {
|
||||||
|
if let Some(handler_scripts) = self.hooks.get(&EventType::Connection) {
|
||||||
|
for script_name in handler_scripts {
|
||||||
|
// Skip disabled scripts
|
||||||
|
if !self.is_script_enabled(script_name) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let globals = lua.globals();
|
||||||
|
let eris: Table = match globals.get("eris") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
let handlers: Table = match eris.get("handlers") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
let event_handlers: Table = match handlers.get("connection") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
for pair in event_handlers.pairs::<i64, Function>() {
|
||||||
|
if let Ok((_, handler)) = pair {
|
||||||
|
let event_ctx = create_event_context(&lua, &ctx);
|
||||||
|
if let Ok(result) = handler.call::<_, Value>((event_ctx,)) {
|
||||||
|
if result == Value::Boolean(false) {
|
||||||
|
should_accept = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !should_accept {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
should_accept
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Called when deciding whether to block an IP
|
||||||
|
pub fn should_block_ip(&self, ip: &str, hit_count: u32) -> bool {
|
||||||
|
let ctx = EventContext {
|
||||||
|
event_type: EventType::BlockIP,
|
||||||
|
ip: Some(ip.to_string()),
|
||||||
|
path: None,
|
||||||
|
user_agent: None,
|
||||||
|
request_headers: None,
|
||||||
|
content: None,
|
||||||
|
timestamp: get_timestamp(),
|
||||||
|
session_id: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
// We should default to not modifying the blocking decision
|
||||||
|
let mut should_block = None;
|
||||||
|
|
||||||
|
if let Ok(lua) = self.lua.lock() {
|
||||||
|
if let Some(handler_scripts) = self.hooks.get(&EventType::BlockIP) {
|
||||||
|
for script_name in handler_scripts {
|
||||||
|
// Skip disabled scripts
|
||||||
|
if !self.is_script_enabled(script_name) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let globals = lua.globals();
|
||||||
|
let eris: Table = match globals.get("eris") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
let handlers: Table = match eris.get("handlers") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
let event_handlers: Table = match handlers.get("block_ip") {
|
||||||
|
Ok(table) => table,
|
||||||
|
Err(_) => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
for pair in event_handlers.pairs::<i64, Function>() {
|
||||||
|
if let Ok((_, handler)) = pair {
|
||||||
|
let event_ctx = create_event_context(&lua, &ctx);
|
||||||
|
// Add hit count for the block_ip event
|
||||||
|
event_ctx.set("hit_count", hit_count).unwrap();
|
||||||
|
|
||||||
|
if let Ok(result) = handler.call::<_, Value>((event_ctx,)) {
|
||||||
|
if let Value::Boolean(block) = result {
|
||||||
|
should_block = Some(block);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if should_block.is_some() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the script's decision, or default to the system behavior
|
||||||
|
should_block.unwrap_or(hit_count >= 3)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Maintains backward compatibility with the old API
|
||||||
|
// XXX: I never liked expand_response, should probably be removeedf
|
||||||
|
// in the future.
|
||||||
|
pub fn expand_response(
|
||||||
|
&self,
|
||||||
|
text: &str,
|
||||||
|
response_type: &str,
|
||||||
|
path: &str,
|
||||||
|
token: &str,
|
||||||
|
) -> String {
|
||||||
|
if let Ok(lua) = self.lua.lock() {
|
||||||
|
let globals = lua.globals();
|
||||||
|
match globals.get::<_, Function>("enhance_response") {
|
||||||
|
Ok(enhance_func) => {
|
||||||
|
match enhance_func.call::<_, String>((text, response_type, path, token)) {
|
||||||
|
Ok(result) => result,
|
||||||
|
Err(e) => {
|
||||||
|
log::warn!("Error calling Lua function enhance_response: {e}");
|
||||||
|
format!("{text}\n<!-- Error calling Lua enhance_response -->")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(_) => format!("{text}\n<!-- Token: {token} -->"),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
format!("{text}\n<!-- Token: {token} -->")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get current timestamp in seconds
|
||||||
|
fn get_timestamp() -> u64 {
|
||||||
|
SystemTime::now()
|
||||||
|
.duration_since(UNIX_EPOCH)
|
||||||
|
.unwrap_or_default()
|
||||||
|
.as_secs()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a unique session ID for tracking a connection
|
||||||
|
fn generate_session_id(ip: &str, user_agent: &str) -> String {
|
||||||
|
let timestamp = get_timestamp();
|
||||||
|
let random = rand::random::<u32>();
|
||||||
|
|
||||||
|
// Use std::hash instead of xxhash_rust
|
||||||
|
let mut hasher = DefaultHasher::new();
|
||||||
|
format!("{ip}_{user_agent}_{timestamp}").hash(&mut hasher);
|
||||||
|
let hash = hasher.finish();
|
||||||
|
|
||||||
|
format!("SID_{hash:x}_{random:x}")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create an event context table in Lua
|
||||||
|
fn create_event_context<'a>(lua: &'a Lua, event_ctx: &EventContext) -> Table<'a> {
|
||||||
|
let table = lua.create_table().unwrap();
|
||||||
|
|
||||||
|
table.set("event", event_ctx.event_type.as_str()).unwrap();
|
||||||
|
if let Some(ip) = &event_ctx.ip {
|
||||||
|
table.set("ip", ip.clone()).unwrap();
|
||||||
|
}
|
||||||
|
if let Some(path) = &event_ctx.path {
|
||||||
|
table.set("path", path.clone()).unwrap();
|
||||||
|
}
|
||||||
|
if let Some(ua) = &event_ctx.user_agent {
|
||||||
|
table.set("user_agent", ua.clone()).unwrap();
|
||||||
|
}
|
||||||
|
table.set("timestamp", event_ctx.timestamp).unwrap();
|
||||||
|
if let Some(sid) = &event_ctx.session_id {
|
||||||
|
table.set("session_id", sid.clone()).unwrap();
|
||||||
|
}
|
||||||
|
if let Some(content) = &event_ctx.content {
|
||||||
|
table.set("content", content.clone()).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
table
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
use std::fs;
|
||||||
|
|
||||||
|
use tempfile::TempDir;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_event_registration() {
|
||||||
|
let temp_dir = TempDir::new().unwrap();
|
||||||
|
let script_path = temp_dir.path().join("test_events.lua");
|
||||||
|
let script_content = r#"
|
||||||
|
-- Example script with event handlers
|
||||||
|
eris.info("Registering event handlers")
|
||||||
|
|
||||||
|
-- Connection event handler
|
||||||
|
eris.on("connection", function(ctx)
|
||||||
|
eris.debug("Connection from " .. ctx.ip)
|
||||||
|
return true -- accept the connection
|
||||||
|
end)
|
||||||
|
|
||||||
|
-- Response generation handler
|
||||||
|
eris.on("response_gen", function(ctx)
|
||||||
|
eris.debug("Generating response for " .. ctx.path)
|
||||||
|
return ctx.content .. "<!-- Enhanced by Lua -->"
|
||||||
|
end)
|
||||||
|
"#;
|
||||||
|
|
||||||
|
fs::write(&script_path, script_content).unwrap();
|
||||||
|
|
||||||
|
let script_manager = ScriptManager::new(temp_dir.path().to_str().unwrap());
|
||||||
|
|
||||||
|
// Verify hooks were registered
|
||||||
|
assert!(script_manager.hooks.contains_key(&EventType::Connection));
|
||||||
|
assert!(script_manager.hooks.contains_key(&EventType::ResponseGen));
|
||||||
|
assert!(!script_manager.hooks.contains_key(&EventType::BlockIP));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_generate_response() {
|
||||||
|
let temp_dir = TempDir::new().unwrap();
|
||||||
|
let script_path = temp_dir.path().join("response_test.lua");
|
||||||
|
let script_content = r#"
|
||||||
|
eris.on("response_gen", function(ctx)
|
||||||
|
return ctx.content .. " - Modified by " .. ctx.user_agent
|
||||||
|
end)
|
||||||
|
"#;
|
||||||
|
|
||||||
|
fs::write(&script_path, script_content).unwrap();
|
||||||
|
|
||||||
|
let script_manager = ScriptManager::new(temp_dir.path().to_str().unwrap());
|
||||||
|
|
||||||
|
let headers = HashMap::new();
|
||||||
|
let result = script_manager.generate_response(
|
||||||
|
"/test/path",
|
||||||
|
"TestBot",
|
||||||
|
"127.0.0.1",
|
||||||
|
&headers,
|
||||||
|
"Original content",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(result.contains("Original content"));
|
||||||
|
assert!(result.contains("Modified by TestBot"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_process_chunk() {
|
||||||
|
let temp_dir = TempDir::new().unwrap();
|
||||||
|
let script_path = temp_dir.path().join("chunk_test.lua");
|
||||||
|
let script_content = r#"
|
||||||
|
eris.on("response_chunk", function(ctx)
|
||||||
|
return ctx.content:gsub("secret", "REDACTED")
|
||||||
|
end)
|
||||||
|
"#;
|
||||||
|
|
||||||
|
fs::write(&script_path, script_content).unwrap();
|
||||||
|
|
||||||
|
let script_manager = ScriptManager::new(temp_dir.path().to_str().unwrap());
|
||||||
|
|
||||||
|
let result = script_manager.process_chunk(
|
||||||
|
"This contains a secret password",
|
||||||
|
"127.0.0.1",
|
||||||
|
"test_session",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(result.contains("This contains a REDACTED password"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_should_block_ip() {
|
||||||
|
let temp_dir = TempDir::new().unwrap();
|
||||||
|
let script_path = temp_dir.path().join("block_test.lua");
|
||||||
|
let script_content = r#"
|
||||||
|
eris.on("block_ip", function(ctx)
|
||||||
|
-- Block any IP with "192.168.1" prefix regardless of hit count
|
||||||
|
if string.match(ctx.ip, "^192%.168%.1%.") then
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Don't block IPs with "10.0" prefix even if they hit the threshold
|
||||||
|
if string.match(ctx.ip, "^10%.0%.") then
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Default behavior for other IPs (nil = use system default)
|
||||||
|
return nil
|
||||||
|
end)
|
||||||
|
"#;
|
||||||
|
|
||||||
|
fs::write(&script_path, script_content).unwrap();
|
||||||
|
|
||||||
|
let script_manager = ScriptManager::new(temp_dir.path().to_str().unwrap());
|
||||||
|
|
||||||
|
// Should be blocked based on IP pattern
|
||||||
|
assert!(script_manager.should_block_ip("192.168.1.50", 1));
|
||||||
|
|
||||||
|
// Should not be blocked despite high hit count
|
||||||
|
assert!(!script_manager.should_block_ip("10.0.0.5", 10));
|
||||||
|
|
||||||
|
// Should use default behavior (block if >= 3 hits)
|
||||||
|
assert!(!script_manager.should_block_ip("172.16.0.1", 2));
|
||||||
|
assert!(script_manager.should_block_ip("172.16.0.1", 3));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_state_and_counters() {
|
||||||
|
let temp_dir = TempDir::new().unwrap();
|
||||||
|
let script_path = temp_dir.path().join("state_test.lua");
|
||||||
|
let script_content = r#"
|
||||||
|
eris.on("startup", function(ctx)
|
||||||
|
eris.set_state("test_key", "test_value")
|
||||||
|
eris.inc_counter("visits", 0)
|
||||||
|
end)
|
||||||
|
|
||||||
|
eris.on("connection", function(ctx)
|
||||||
|
local count = eris.inc_counter("visits")
|
||||||
|
eris.debug("Visit count: " .. count)
|
||||||
|
|
||||||
|
-- Store last visitor
|
||||||
|
eris.set_state("last_visitor", ctx.ip)
|
||||||
|
return true
|
||||||
|
end)
|
||||||
|
|
||||||
|
eris.on("response_gen", function(ctx)
|
||||||
|
local last_visitor = eris.get_state("last_visitor") or "unknown"
|
||||||
|
local visits = eris.get_counter("visits")
|
||||||
|
return ctx.content .. "<!-- Last visitor: " .. last_visitor ..
|
||||||
|
", Total visits: " .. visits .. " -->"
|
||||||
|
end)
|
||||||
|
"#;
|
||||||
|
|
||||||
|
fs::write(&script_path, script_content).unwrap();
|
||||||
|
|
||||||
|
let script_manager = ScriptManager::new(temp_dir.path().to_str().unwrap());
|
||||||
|
|
||||||
|
// Simulate connections
|
||||||
|
script_manager.on_connection("192.168.1.100");
|
||||||
|
script_manager.on_connection("10.0.0.50");
|
||||||
|
|
||||||
|
// Check response includes state
|
||||||
|
let headers = HashMap::new();
|
||||||
|
let result = script_manager.generate_response(
|
||||||
|
"/test",
|
||||||
|
"test-agent",
|
||||||
|
"8.8.8.8",
|
||||||
|
&headers,
|
||||||
|
"Response",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(result.contains("Last visitor: 10.0.0.50"));
|
||||||
|
assert!(result.contains("Total visits: 2"));
|
||||||
|
}
|
||||||
|
}
|
||||||
565
src/main.rs
565
src/main.rs
|
|
@ -1,11 +1,11 @@
|
||||||
use actix_web::{App, HttpResponse, HttpServer, web};
|
use actix_web::{App, HttpResponse, HttpServer, web};
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
use ipnetwork::IpNetwork;
|
use ipnetwork::IpNetwork;
|
||||||
use rlua::{Function, Lua};
|
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use std::collections::{HashMap, HashSet};
|
use std::collections::{HashMap, HashSet};
|
||||||
use std::env;
|
use std::env;
|
||||||
use std::fs;
|
use std::fs;
|
||||||
|
use std::hash::Hasher;
|
||||||
use std::io::Write;
|
use std::io::Write;
|
||||||
use std::net::IpAddr;
|
use std::net::IpAddr;
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
|
|
@ -17,9 +17,11 @@ use tokio::process::Command;
|
||||||
use tokio::sync::RwLock;
|
use tokio::sync::RwLock;
|
||||||
use tokio::time::sleep;
|
use tokio::time::sleep;
|
||||||
|
|
||||||
|
mod lua;
|
||||||
mod markov;
|
mod markov;
|
||||||
mod metrics;
|
mod metrics;
|
||||||
|
|
||||||
|
use lua::{EventContext, EventType, ScriptManager};
|
||||||
use markov::MarkovGenerator;
|
use markov::MarkovGenerator;
|
||||||
use metrics::{
|
use metrics::{
|
||||||
ACTIVE_CONNECTIONS, BLOCKED_IPS, HITS_COUNTER, PATH_HITS, UA_HITS, metrics_handler,
|
ACTIVE_CONNECTIONS, BLOCKED_IPS, HITS_COUNTER, PATH_HITS, UA_HITS, metrics_handler,
|
||||||
|
|
@ -356,10 +358,11 @@ impl BotState {
|
||||||
}
|
}
|
||||||
|
|
||||||
let hit_cache_file = format!("{}/hit_counters.json", self.cache_dir);
|
let hit_cache_file = format!("{}/hit_counters.json", self.cache_dir);
|
||||||
let mut hit_map = HashMap::new();
|
let hit_map: HashMap<String, u32> = self
|
||||||
for (ip, count) in &self.hits {
|
.hits
|
||||||
hit_map.insert(ip.to_string(), *count);
|
.iter()
|
||||||
}
|
.map(|(ip, count)| (ip.to_string(), *count))
|
||||||
|
.collect();
|
||||||
|
|
||||||
match fs::File::create(&hit_cache_file) {
|
match fs::File::create(&hit_cache_file) {
|
||||||
Ok(file) => {
|
Ok(file) => {
|
||||||
|
|
@ -376,116 +379,11 @@ impl BotState {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Lua scripts for response generation and customization
|
|
||||||
struct ScriptManager {
|
|
||||||
script_content: String,
|
|
||||||
scripts_loaded: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ScriptManager {
|
|
||||||
fn new(scripts_dir: &str) -> Self {
|
|
||||||
let mut script_content = String::new();
|
|
||||||
let mut scripts_loaded = false;
|
|
||||||
|
|
||||||
// Try to load scripts from directory
|
|
||||||
let script_dir = Path::new(scripts_dir);
|
|
||||||
if script_dir.exists() {
|
|
||||||
log::debug!("Loading Lua scripts from directory: {scripts_dir}");
|
|
||||||
if let Ok(entries) = fs::read_dir(script_dir) {
|
|
||||||
for entry in entries {
|
|
||||||
if let Ok(entry) = entry {
|
|
||||||
let path = entry.path();
|
|
||||||
if path.extension().and_then(|ext| ext.to_str()) == Some("lua") {
|
|
||||||
if let Ok(content) = fs::read_to_string(&path) {
|
|
||||||
log::debug!("Loaded Lua script: {}", path.display());
|
|
||||||
script_content.push_str(&content);
|
|
||||||
script_content.push('\n');
|
|
||||||
scripts_loaded = true;
|
|
||||||
} else {
|
|
||||||
log::warn!("Failed to read Lua script: {}", path.display());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
log::warn!("Lua scripts directory does not exist: {scripts_dir}");
|
|
||||||
}
|
|
||||||
|
|
||||||
// If no scripts were loaded, use a default script
|
|
||||||
if !scripts_loaded {
|
|
||||||
log::info!("No Lua scripts found, loading default scripts");
|
|
||||||
script_content = r#"
|
|
||||||
function generate_honeytoken(token)
|
|
||||||
local token_types = {"API_KEY", "AUTH_TOKEN", "SESSION_ID", "SECRET_KEY"}
|
|
||||||
local prefix = token_types[math.random(#token_types)]
|
|
||||||
local suffix = string.format("%08x", math.random(0xffffff))
|
|
||||||
return prefix .. "_" .. token .. "_" .. suffix
|
|
||||||
end
|
|
||||||
|
|
||||||
function enhance_response(text, response_type, path, token)
|
|
||||||
local result = text
|
|
||||||
local honeytoken = generate_honeytoken(token)
|
|
||||||
|
|
||||||
-- Add some fake sensitive data
|
|
||||||
result = result .. "\n<!-- DEBUG: " .. honeytoken .. " -->"
|
|
||||||
result = result .. "\n<div style='display:none'>Server ID: " .. token .. "</div>"
|
|
||||||
|
|
||||||
return result
|
|
||||||
end
|
|
||||||
"#
|
|
||||||
.to_string();
|
|
||||||
scripts_loaded = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
Self {
|
|
||||||
script_content,
|
|
||||||
scripts_loaded,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Lua is a powerful configuration language we can use to expand functionality of
|
|
||||||
// Eris, e.g., with fake tokens or honeytrap content.
|
|
||||||
fn expand_response(&self, text: &str, response_type: &str, path: &str, token: &str) -> String {
|
|
||||||
if !self.scripts_loaded {
|
|
||||||
return format!("{text}\n<!-- Token: {token} -->");
|
|
||||||
}
|
|
||||||
|
|
||||||
let lua = Lua::new();
|
|
||||||
if let Err(e) = lua.load(&self.script_content).exec() {
|
|
||||||
log::warn!("Error loading Lua script: {e}");
|
|
||||||
return format!("{text}\n<!-- Error: Failed to load Lua script -->");
|
|
||||||
}
|
|
||||||
|
|
||||||
let globals = lua.globals();
|
|
||||||
match globals.get::<_, Function>("enhance_response") {
|
|
||||||
Ok(enhance_func) => {
|
|
||||||
match enhance_func.call::<_, String>((text, response_type, path, token)) {
|
|
||||||
Ok(result) => result,
|
|
||||||
Err(e) => {
|
|
||||||
log::warn!("Error calling Lua function enhance_response: {e}");
|
|
||||||
format!("{text}\n<!-- Error calling Lua enhance_response -->")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
log::warn!("Lua enhance_response function not found: {e}");
|
|
||||||
format!("{text}\n<!-- Lua enhance_response function not found -->")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Find end of HTTP headers
|
// Find end of HTTP headers
|
||||||
// XXX: I'm sure this could be made less fragile.
|
|
||||||
fn find_header_end(data: &[u8]) -> Option<usize> {
|
fn find_header_end(data: &[u8]) -> Option<usize> {
|
||||||
for i in 0..data.len().saturating_sub(3) {
|
data.windows(4)
|
||||||
if data[i] == b'\r' && data[i + 1] == b'\n' && data[i + 2] == b'\r' && data[i + 3] == b'\n'
|
.position(|window| window == b"\r\n\r\n")
|
||||||
{
|
.map(|pos| pos + 4)
|
||||||
return Some(i + 4);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
None
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extract path from raw request data
|
// Extract path from raw request data
|
||||||
|
|
@ -516,6 +414,67 @@ fn extract_header_value(data: &[u8], header_name: &str) -> Option<String> {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Extract all headers from request data
|
||||||
|
fn extract_all_headers(data: &[u8]) -> HashMap<String, String> {
|
||||||
|
let mut headers = HashMap::new();
|
||||||
|
|
||||||
|
if let Ok(data_str) = std::str::from_utf8(data) {
|
||||||
|
let mut lines = data_str.lines();
|
||||||
|
|
||||||
|
// Skip the request line
|
||||||
|
let _ = lines.next();
|
||||||
|
|
||||||
|
// Parse headers until empty line
|
||||||
|
for line in lines {
|
||||||
|
if line.is_empty() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(colon_pos) = line.find(':') {
|
||||||
|
let key = line[..colon_pos].trim().to_lowercase();
|
||||||
|
let value = line[colon_pos + 1..].trim().to_string();
|
||||||
|
headers.insert(key, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
headers
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine response type based on request path
|
||||||
|
fn choose_response_type(path: &str) -> &'static str {
|
||||||
|
if path.contains("phpunit") || path.contains("eval") {
|
||||||
|
"php_exploit"
|
||||||
|
} else if path.contains("wp-") {
|
||||||
|
"wordpress"
|
||||||
|
} else if path.contains("api") {
|
||||||
|
"api"
|
||||||
|
} else {
|
||||||
|
"generic"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper function to get current timestamp in seconds
|
||||||
|
fn get_timestamp() -> u64 {
|
||||||
|
std::time::SystemTime::now()
|
||||||
|
.duration_since(std::time::UNIX_EPOCH)
|
||||||
|
.unwrap_or_default()
|
||||||
|
.as_secs()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a unique session ID for tracking a connection
|
||||||
|
fn generate_session_id(ip: &str, user_agent: &str) -> String {
|
||||||
|
let timestamp = get_timestamp();
|
||||||
|
let random = rand::random::<u32>();
|
||||||
|
|
||||||
|
// Use std::hash instead of xxhash_rust
|
||||||
|
let mut hasher = std::collections::hash_map::DefaultHasher::new();
|
||||||
|
std::hash::Hash::hash(&format!("{ip}_{user_agent}_{timestamp}"), &mut hasher);
|
||||||
|
let hash = hasher.finish();
|
||||||
|
|
||||||
|
format!("SID_{hash:x}_{random:x}")
|
||||||
|
}
|
||||||
|
|
||||||
// Main connection handler.
|
// Main connection handler.
|
||||||
// Decides whether to tarpit or proxy
|
// Decides whether to tarpit or proxy
|
||||||
async fn handle_connection(
|
async fn handle_connection(
|
||||||
|
|
@ -541,6 +500,13 @@ async fn handle_connection(
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check if Lua scripts allow this connection
|
||||||
|
if !script_manager.on_connection(&peer_addr.to_string()) {
|
||||||
|
log::debug!("Connection rejected by Lua script: {peer_addr}");
|
||||||
|
let _ = stream.shutdown().await;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Pre-check for whitelisted IPs to bypass heavy processing
|
// Pre-check for whitelisted IPs to bypass heavy processing
|
||||||
let mut whitelisted = false;
|
let mut whitelisted = false;
|
||||||
for network_str in &config.whitelist_networks {
|
for network_str in &config.whitelist_networks {
|
||||||
|
|
@ -615,14 +581,30 @@ async fn handle_connection(
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Extract request headers for Lua scripts
|
||||||
|
let headers = extract_all_headers(&request_data);
|
||||||
|
|
||||||
|
// Extract user agent for logging and decision making
|
||||||
|
let user_agent =
|
||||||
|
extract_header_value(&request_data, "user-agent").unwrap_or_else(|| "unknown".to_string());
|
||||||
|
|
||||||
|
// Trigger request event for Lua scripts
|
||||||
|
let request_ctx = EventContext {
|
||||||
|
event_type: EventType::Request,
|
||||||
|
ip: Some(peer_addr.to_string()),
|
||||||
|
path: Some(path.to_string()),
|
||||||
|
user_agent: Some(user_agent.clone()),
|
||||||
|
request_headers: Some(headers.clone()),
|
||||||
|
content: None,
|
||||||
|
timestamp: get_timestamp(),
|
||||||
|
session_id: Some(generate_session_id(&peer_addr.to_string(), &user_agent)),
|
||||||
|
};
|
||||||
|
script_manager.trigger_event(&request_ctx);
|
||||||
|
|
||||||
// Check if this request matches our tarpit patterns
|
// Check if this request matches our tarpit patterns
|
||||||
let should_tarpit = should_tarpit(path, &peer_addr, &config).await;
|
let should_tarpit = should_tarpit(path, &peer_addr, &config).await;
|
||||||
|
|
||||||
if should_tarpit {
|
if should_tarpit {
|
||||||
// Extract minimal info needed for tarpit
|
|
||||||
let user_agent = extract_header_value(&request_data, "user-agent")
|
|
||||||
.unwrap_or_else(|| "unknown".to_string());
|
|
||||||
|
|
||||||
log::info!("Tarpit triggered: {path} from {peer_addr} (UA: {user_agent})");
|
log::info!("Tarpit triggered: {path} from {peer_addr} (UA: {user_agent})");
|
||||||
|
|
||||||
// Update metrics
|
// Update metrics
|
||||||
|
|
@ -639,8 +621,11 @@ async fn handle_connection(
|
||||||
*state.hits.entry(peer_addr).or_insert(0) += 1;
|
*state.hits.entry(peer_addr).or_insert(0) += 1;
|
||||||
let hit_count = state.hits[&peer_addr];
|
let hit_count = state.hits[&peer_addr];
|
||||||
|
|
||||||
|
// Use Lua to decide whether to block this IP
|
||||||
|
let should_block = script_manager.should_block_ip(&peer_addr.to_string(), hit_count);
|
||||||
|
|
||||||
// Block IPs that hit tarpits too many times
|
// Block IPs that hit tarpits too many times
|
||||||
if hit_count >= config.block_threshold && !state.blocked.contains(&peer_addr) {
|
if should_block && !state.blocked.contains(&peer_addr) {
|
||||||
log::info!("Blocking IP {peer_addr} after {hit_count} hits");
|
log::info!("Blocking IP {peer_addr} after {hit_count} hits");
|
||||||
state.blocked.insert(peer_addr);
|
state.blocked.insert(peer_addr);
|
||||||
BLOCKED_IPS.set(state.blocked.len() as f64);
|
BLOCKED_IPS.set(state.blocked.len() as f64);
|
||||||
|
|
@ -682,20 +667,116 @@ async fn handle_connection(
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate a deceptive response using Markov chains and Lua
|
// Generate a deceptive response using Markov chains and Lua
|
||||||
let response =
|
let response = generate_deceptive_response(
|
||||||
generate_deceptive_response(path, &user_agent, &markov_generator, &script_manager)
|
path,
|
||||||
.await;
|
&user_agent,
|
||||||
|
&peer_addr,
|
||||||
|
&headers,
|
||||||
|
&markov_generator,
|
||||||
|
&script_manager,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
|
||||||
|
// Generate a session ID for tracking this tarpit session
|
||||||
|
let session_id = generate_session_id(&peer_addr.to_string(), &user_agent);
|
||||||
|
|
||||||
// Send the response with the tarpit delay strategy
|
// Send the response with the tarpit delay strategy
|
||||||
tarpit_connection(
|
{
|
||||||
stream,
|
let mut stream = stream;
|
||||||
response,
|
let peer_addr = peer_addr;
|
||||||
peer_addr,
|
let state = state.clone();
|
||||||
state.clone(),
|
let min_delay = config.min_delay;
|
||||||
config.min_delay,
|
let max_delay = config.max_delay;
|
||||||
config.max_delay,
|
let max_tarpit_time = config.max_tarpit_time;
|
||||||
config.max_tarpit_time,
|
let script_manager = script_manager.clone();
|
||||||
)
|
async move {
|
||||||
|
let start_time = Instant::now();
|
||||||
|
let mut chars = response.chars().collect::<Vec<_>>();
|
||||||
|
for i in (0..chars.len()).rev() {
|
||||||
|
if i > 0 && rand::random::<f32>() < 0.1 {
|
||||||
|
chars.swap(i, i - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log::debug!(
|
||||||
|
"Starting tarpit for {} with {} chars, min_delay={}ms, max_delay={}ms",
|
||||||
|
peer_addr,
|
||||||
|
chars.len(),
|
||||||
|
min_delay,
|
||||||
|
max_delay
|
||||||
|
);
|
||||||
|
let mut position = 0;
|
||||||
|
let mut chunks_sent = 0;
|
||||||
|
let mut total_delay = 0;
|
||||||
|
while position < chars.len() {
|
||||||
|
// Check if we've exceeded maximum tarpit time
|
||||||
|
let elapsed_secs = start_time.elapsed().as_secs();
|
||||||
|
if elapsed_secs > max_tarpit_time {
|
||||||
|
log::info!(
|
||||||
|
"Tarpit maximum time ({max_tarpit_time} sec) reached for {peer_addr}"
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Decide how many chars to send in this chunk (usually 1, sometimes more)
|
||||||
|
let chunk_size = if rand::random::<f32>() < 0.9 {
|
||||||
|
1
|
||||||
|
} else {
|
||||||
|
(rand::random::<f32>() * 3.0).floor() as usize + 1
|
||||||
|
};
|
||||||
|
|
||||||
|
let end = (position + chunk_size).min(chars.len());
|
||||||
|
let chunk: String = chars[position..end].iter().collect();
|
||||||
|
|
||||||
|
// Process chunk through Lua before sending
|
||||||
|
let processed_chunk =
|
||||||
|
script_manager.process_chunk(&chunk, &peer_addr.to_string(), &session_id);
|
||||||
|
|
||||||
|
// Try to write processed chunk
|
||||||
|
if stream.write_all(processed_chunk.as_bytes()).await.is_err() {
|
||||||
|
log::debug!("Connection closed by client during tarpit: {peer_addr}");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if stream.flush().await.is_err() {
|
||||||
|
log::debug!("Failed to flush stream during tarpit: {peer_addr}");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
position = end;
|
||||||
|
chunks_sent += 1;
|
||||||
|
|
||||||
|
// Apply random delay between min and max configured values
|
||||||
|
let delay_ms =
|
||||||
|
(rand::random::<f32>() * (max_delay - min_delay) as f32) as u64 + min_delay;
|
||||||
|
total_delay += delay_ms;
|
||||||
|
sleep(Duration::from_millis(delay_ms)).await;
|
||||||
|
}
|
||||||
|
log::debug!(
|
||||||
|
"Tarpit stats for {}: sent {} chunks, {}% of data, total delay {}ms over {}s",
|
||||||
|
peer_addr,
|
||||||
|
chunks_sent,
|
||||||
|
position * 100 / chars.len(),
|
||||||
|
total_delay,
|
||||||
|
start_time.elapsed().as_secs()
|
||||||
|
);
|
||||||
|
let disconnection_ctx = EventContext {
|
||||||
|
event_type: EventType::Disconnection,
|
||||||
|
ip: Some(peer_addr.to_string()),
|
||||||
|
path: None,
|
||||||
|
user_agent: None,
|
||||||
|
request_headers: None,
|
||||||
|
content: None,
|
||||||
|
timestamp: get_timestamp(),
|
||||||
|
session_id: Some(session_id),
|
||||||
|
};
|
||||||
|
script_manager.trigger_event(&disconnection_ctx);
|
||||||
|
if let Ok(mut state) = state.try_write() {
|
||||||
|
state.active_connections.remove(&peer_addr);
|
||||||
|
ACTIVE_CONNECTIONS.set(state.active_connections.len() as f64);
|
||||||
|
}
|
||||||
|
let _ = stream.shutdown().await;
|
||||||
|
}
|
||||||
|
}
|
||||||
.await;
|
.await;
|
||||||
} else {
|
} else {
|
||||||
log::debug!("Proxying request: {path} from {peer_addr}");
|
log::debug!("Proxying request: {path} from {peer_addr}");
|
||||||
|
|
@ -816,134 +897,25 @@ async fn should_tarpit(path: &str, ip: &IpAddr, config: &Config) -> bool {
|
||||||
async fn generate_deceptive_response(
|
async fn generate_deceptive_response(
|
||||||
path: &str,
|
path: &str,
|
||||||
user_agent: &str,
|
user_agent: &str,
|
||||||
|
peer_addr: &IpAddr,
|
||||||
|
headers: &HashMap<String, String>,
|
||||||
markov: &MarkovGenerator,
|
markov: &MarkovGenerator,
|
||||||
script_manager: &ScriptManager,
|
script_manager: &ScriptManager,
|
||||||
) -> String {
|
) -> String {
|
||||||
// Choose response type based on path to seem more realistic
|
|
||||||
let response_type = if path.contains("phpunit") || path.contains("eval") {
|
|
||||||
"php_exploit"
|
|
||||||
} else if path.contains("wp-") {
|
|
||||||
"wordpress"
|
|
||||||
} else if path.contains("api") {
|
|
||||||
"api"
|
|
||||||
} else {
|
|
||||||
"generic"
|
|
||||||
};
|
|
||||||
|
|
||||||
log::debug!("Generating {response_type} response for path: {path}");
|
|
||||||
|
|
||||||
// Generate tracking token for this interaction
|
|
||||||
let tracking_token = format!(
|
|
||||||
"BOT_{}_{}",
|
|
||||||
user_agent
|
|
||||||
.chars()
|
|
||||||
.filter(|c| c.is_alphanumeric())
|
|
||||||
.collect::<String>(),
|
|
||||||
chrono::Utc::now().timestamp()
|
|
||||||
);
|
|
||||||
|
|
||||||
// Generate base response using Markov chain text generator
|
// Generate base response using Markov chain text generator
|
||||||
|
let response_type = choose_response_type(path);
|
||||||
let markov_text = markov.generate(response_type, 30);
|
let markov_text = markov.generate(response_type, 30);
|
||||||
|
|
||||||
// Use Lua to enhance with honeytokens and other deceptive content
|
// Use Lua scripts to enhance with honeytokens and other deceptive content
|
||||||
let response_expanded =
|
script_manager.generate_response(
|
||||||
script_manager.expand_response(&markov_text, response_type, path, &tracking_token);
|
path,
|
||||||
|
user_agent,
|
||||||
// Return full HTTP response with appropriate headers
|
&peer_addr.to_string(),
|
||||||
format!(
|
headers,
|
||||||
"HTTP/1.1 200 OK\r\nContent-Type: text/html\r\nX-Powered-By: PHP/7.4.3\r\nConnection: keep-alive\r\n\r\n{response_expanded}"
|
&markov_text,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Slowly feed a response to the client with random delays to waste attacker time
|
|
||||||
async fn tarpit_connection(
|
|
||||||
mut stream: TcpStream,
|
|
||||||
response: String,
|
|
||||||
peer_addr: IpAddr,
|
|
||||||
state: Arc<RwLock<BotState>>,
|
|
||||||
min_delay: u64,
|
|
||||||
max_delay: u64,
|
|
||||||
max_tarpit_time: u64,
|
|
||||||
) {
|
|
||||||
let start_time = Instant::now();
|
|
||||||
let mut chars = response.chars().collect::<Vec<_>>();
|
|
||||||
|
|
||||||
// Randomize the char order slightly to confuse automated tools
|
|
||||||
for i in (0..chars.len()).rev() {
|
|
||||||
if i > 0 && rand::random::<f32>() < 0.1 {
|
|
||||||
chars.swap(i, i - 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
log::debug!(
|
|
||||||
"Starting tarpit for {} with {} chars, min_delay={}ms, max_delay={}ms",
|
|
||||||
peer_addr,
|
|
||||||
chars.len(),
|
|
||||||
min_delay,
|
|
||||||
max_delay
|
|
||||||
);
|
|
||||||
|
|
||||||
let mut position = 0;
|
|
||||||
let mut chunks_sent = 0;
|
|
||||||
let mut total_delay = 0;
|
|
||||||
|
|
||||||
// Send the response character by character with random delays
|
|
||||||
while position < chars.len() {
|
|
||||||
// Check if we've exceeded maximum tarpit time
|
|
||||||
let elapsed_secs = start_time.elapsed().as_secs();
|
|
||||||
if elapsed_secs > max_tarpit_time {
|
|
||||||
log::info!("Tarpit maximum time ({max_tarpit_time} sec) reached for {peer_addr}");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Decide how many chars to send in this chunk (usually 1, sometimes more)
|
|
||||||
let chunk_size = if rand::random::<f32>() < 0.9 {
|
|
||||||
1
|
|
||||||
} else {
|
|
||||||
(rand::random::<f32>() * 3.0).floor() as usize + 1
|
|
||||||
};
|
|
||||||
|
|
||||||
let end = (position + chunk_size).min(chars.len());
|
|
||||||
let chunk: String = chars[position..end].iter().collect();
|
|
||||||
|
|
||||||
// Try to write chunk
|
|
||||||
if stream.write_all(chunk.as_bytes()).await.is_err() {
|
|
||||||
log::debug!("Connection closed by client during tarpit: {peer_addr}");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if stream.flush().await.is_err() {
|
|
||||||
log::debug!("Failed to flush stream during tarpit: {peer_addr}");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
position = end;
|
|
||||||
chunks_sent += 1;
|
|
||||||
|
|
||||||
// Apply random delay between min and max configured values
|
|
||||||
let delay_ms = (rand::random::<f32>() * (max_delay - min_delay) as f32) as u64 + min_delay;
|
|
||||||
total_delay += delay_ms;
|
|
||||||
sleep(Duration::from_millis(delay_ms)).await;
|
|
||||||
}
|
|
||||||
|
|
||||||
log::debug!(
|
|
||||||
"Tarpit stats for {}: sent {} chunks, {}% of data, total delay {}ms over {}s",
|
|
||||||
peer_addr,
|
|
||||||
chunks_sent,
|
|
||||||
position * 100 / chars.len(),
|
|
||||||
total_delay,
|
|
||||||
start_time.elapsed().as_secs()
|
|
||||||
);
|
|
||||||
|
|
||||||
// Remove from active connections
|
|
||||||
if let Ok(mut state) = state.try_write() {
|
|
||||||
state.active_connections.remove(&peer_addr);
|
|
||||||
ACTIVE_CONNECTIONS.set(state.active_connections.len() as f64);
|
|
||||||
}
|
|
||||||
|
|
||||||
let _ = stream.shutdown().await;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set up nftables firewall rules for IP blocking
|
// Set up nftables firewall rules for IP blocking
|
||||||
async fn setup_firewall() -> Result<(), String> {
|
async fn setup_firewall() -> Result<(), String> {
|
||||||
log::info!("Setting up firewall rules");
|
log::info!("Setting up firewall rules");
|
||||||
|
|
@ -1162,6 +1134,8 @@ async fn main() -> std::io::Result<()> {
|
||||||
// Initialize Lua script manager
|
// Initialize Lua script manager
|
||||||
log::info!("Loading Lua scripts from {}", config.lua_scripts_dir);
|
log::info!("Loading Lua scripts from {}", config.lua_scripts_dir);
|
||||||
let script_manager = Arc::new(ScriptManager::new(&config.lua_scripts_dir));
|
let script_manager = Arc::new(ScriptManager::new(&config.lua_scripts_dir));
|
||||||
|
let script_manager_for_tarpit = script_manager.clone();
|
||||||
|
let script_manager_for_periodic = script_manager.clone();
|
||||||
|
|
||||||
// Clone config for metrics server
|
// Clone config for metrics server
|
||||||
let metrics_config = config.clone();
|
let metrics_config = config.clone();
|
||||||
|
|
@ -1186,7 +1160,7 @@ async fn main() -> std::io::Result<()> {
|
||||||
|
|
||||||
let state_clone = tarpit_state.clone();
|
let state_clone = tarpit_state.clone();
|
||||||
let markov_clone = markov_generator.clone();
|
let markov_clone = markov_generator.clone();
|
||||||
let script_manager_clone = script_manager.clone();
|
let script_manager_clone = script_manager_for_tarpit.clone();
|
||||||
let config_clone = config.clone();
|
let config_clone = config.clone();
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
|
|
@ -1241,6 +1215,27 @@ async fn main() -> std::io::Result<()> {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Setup periodic task runner for Lua scripts
|
||||||
|
tokio::spawn(async move {
|
||||||
|
let mut interval = tokio::time::interval(Duration::from_secs(60));
|
||||||
|
loop {
|
||||||
|
interval.tick().await;
|
||||||
|
|
||||||
|
// Trigger periodic event
|
||||||
|
let ctx = EventContext {
|
||||||
|
event_type: EventType::Periodic,
|
||||||
|
ip: None,
|
||||||
|
path: None,
|
||||||
|
user_agent: None,
|
||||||
|
request_headers: None,
|
||||||
|
content: None,
|
||||||
|
timestamp: get_timestamp(),
|
||||||
|
session_id: None,
|
||||||
|
};
|
||||||
|
script_manager_for_periodic.trigger_event(&ctx);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// Run both servers concurrently if metrics server is enabled
|
// Run both servers concurrently if metrics server is enabled
|
||||||
if let Some(metrics_server) = metrics_server {
|
if let Some(metrics_server) = metrics_server {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
|
|
@ -1379,18 +1374,6 @@ mod tests {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_script_manager_default_script() {
|
|
||||||
let script_manager = ScriptManager::new("/nonexistent_directory");
|
|
||||||
assert!(script_manager.scripts_loaded);
|
|
||||||
assert!(
|
|
||||||
script_manager
|
|
||||||
.script_content
|
|
||||||
.contains("generate_honeytoken")
|
|
||||||
);
|
|
||||||
assert!(script_manager.script_content.contains("enhance_response"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::test]
|
#[tokio::test]
|
||||||
async fn test_bot_state() {
|
async fn test_bot_state() {
|
||||||
let state = BotState::new("/tmp/eris_test", "/tmp/eris_test_cache");
|
let state = BotState::new("/tmp/eris_test", "/tmp/eris_test_cache");
|
||||||
|
|
@ -1432,37 +1415,6 @@ mod tests {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test]
|
|
||||||
async fn test_generate_deceptive_response() {
|
|
||||||
// Create a simple markov generator for testing
|
|
||||||
let markov = MarkovGenerator::new("/nonexistent/path");
|
|
||||||
let script_manager = ScriptManager::new("/nonexistent/path");
|
|
||||||
|
|
||||||
// Test different path types
|
|
||||||
let resp1 = generate_deceptive_response(
|
|
||||||
"/vendor/phpunit/exec",
|
|
||||||
"TestBot/1.0",
|
|
||||||
&markov,
|
|
||||||
&script_manager,
|
|
||||||
)
|
|
||||||
.await;
|
|
||||||
assert!(resp1.contains("HTTP/1.1 200 OK"));
|
|
||||||
assert!(resp1.contains("X-Powered-By: PHP"));
|
|
||||||
|
|
||||||
let resp2 =
|
|
||||||
generate_deceptive_response("/wp-admin/", "TestBot/1.0", &markov, &script_manager)
|
|
||||||
.await;
|
|
||||||
assert!(resp2.contains("HTTP/1.1 200 OK"));
|
|
||||||
|
|
||||||
let resp3 =
|
|
||||||
generate_deceptive_response("/api/users", "TestBot/1.0", &markov, &script_manager)
|
|
||||||
.await;
|
|
||||||
assert!(resp3.contains("HTTP/1.1 200 OK"));
|
|
||||||
|
|
||||||
// Verify tracking token is included
|
|
||||||
assert!(resp1.contains("BOT_TestBot"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_find_header_end() {
|
fn test_find_header_end() {
|
||||||
let data = b"GET / HTTP/1.1\r\nHost: example.com\r\nUser-Agent: test\r\n\r\nBody content";
|
let data = b"GET / HTTP/1.1\r\nHost: example.com\r\nUser-Agent: test\r\n\r\nBody content";
|
||||||
|
|
@ -1494,4 +1446,31 @@ mod tests {
|
||||||
);
|
);
|
||||||
assert_eq!(extract_header_value(data, "nonexistent"), None);
|
assert_eq!(extract_header_value(data, "nonexistent"), None);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_extract_all_headers() {
|
||||||
|
let data = b"GET / HTTP/1.1\r\nHost: example.com\r\nUser-Agent: TestBot/1.0\r\nAccept: */*\r\n\r\n";
|
||||||
|
let headers = extract_all_headers(data);
|
||||||
|
|
||||||
|
assert_eq!(headers.len(), 3);
|
||||||
|
assert_eq!(headers.get("host").unwrap(), "example.com");
|
||||||
|
assert_eq!(headers.get("user-agent").unwrap(), "TestBot/1.0");
|
||||||
|
assert_eq!(headers.get("accept").unwrap(), "*/*");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_choose_response_type() {
|
||||||
|
assert_eq!(
|
||||||
|
choose_response_type("/vendor/phpunit/whatever"),
|
||||||
|
"php_exploit"
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
choose_response_type("/path/to/eval-stdin.php"),
|
||||||
|
"php_exploit"
|
||||||
|
);
|
||||||
|
assert_eq!(choose_response_type("/wp-admin/login.php"), "wordpress");
|
||||||
|
assert_eq!(choose_response_type("/wp-login.php"), "wordpress");
|
||||||
|
assert_eq!(choose_response_type("/api/v1/users"), "api");
|
||||||
|
assert_eq!(choose_response_type("/index.html"), "generic");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -103,7 +103,7 @@ impl MarkovGenerator {
|
||||||
let path = Path::new(corpus_dir);
|
let path = Path::new(corpus_dir);
|
||||||
if path.exists() && path.is_dir() {
|
if path.exists() && path.is_dir() {
|
||||||
if let Ok(entries) = fs::read_dir(path) {
|
if let Ok(entries) = fs::read_dir(path) {
|
||||||
for entry in entries {
|
entries.for_each(|entry| {
|
||||||
if let Ok(entry) = entry {
|
if let Ok(entry) = entry {
|
||||||
let file_path = entry.path();
|
let file_path = entry.path();
|
||||||
if let Some(file_name) = file_path.file_stem() {
|
if let Some(file_name) = file_path.file_stem() {
|
||||||
|
|
@ -120,7 +120,7 @@ impl MarkovGenerator {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue