mirror of
https://github.com/elicpeter/nyx.git
synced 2026-06-09 19:45:13 +02:00
* refactor: Update comments for clarity and add expectations.json files for performance metrics * feat: Implement FP guard for JS/TS local-collection receivers to suppress missing ownership checks * feat: Enhance Rust parameter handling to classify local collections and prevent false ownership checks * refactor: Simplify code formatting for better readability in multiple files * refactor: Improve UTF-8 sequence length handling and enhance clarity in loop iteration * feat: Update Java and Python patterns to include new security rules * refactor: Improve comment clarity and consistency across multiple Rust files * refactor: Simplify code formatting for improved readability in integration tests and module files * refactor: Improve comment formatting and enhance clarity in assertions across multiple files
247 lines
8.5 KiB
Rust
247 lines
8.5 KiB
Rust
use crate::labels::{Cap, DataLabel, Kind, LabelRule, ParamConfig, RuntimeLabelRule};
|
|
use crate::utils::project::{DetectedFramework, FrameworkContext};
|
|
use phf::{Map, phf_map};
|
|
|
|
pub static RULES: &[LabelRule] = &[
|
|
// ─────────── Sources ───────────
|
|
LabelRule {
|
|
matchers: &["System.getenv"],
|
|
label: DataLabel::Source(Cap::all()),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &[
|
|
"getParameter",
|
|
"getInputStream",
|
|
"getHeader",
|
|
"getCookies",
|
|
"getReader",
|
|
"getQueryString",
|
|
"getPathInfo",
|
|
"getRequestURI",
|
|
"getRequestURL",
|
|
"getServletPath",
|
|
"getContextPath",
|
|
],
|
|
label: DataLabel::Source(Cap::all()),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &["readObject", "readLine", "ObjectMapper.readValue"],
|
|
label: DataLabel::Source(Cap::all()),
|
|
case_sensitive: false,
|
|
},
|
|
// ───────── Sanitizers ──────────
|
|
LabelRule {
|
|
matchers: &["HtmlUtils.htmlEscape", "StringEscapeUtils.escapeHtml4"],
|
|
label: DataLabel::Sanitizer(Cap::HTML_ESCAPE),
|
|
case_sensitive: false,
|
|
},
|
|
// OWASP ESAPI encoders
|
|
LabelRule {
|
|
matchers: &["Encoder.encodeForHTML", "Encoder.encodeForJavaScript"],
|
|
label: DataLabel::Sanitizer(Cap::HTML_ESCAPE),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &["Encoder.encodeForSQL"],
|
|
label: DataLabel::Sanitizer(Cap::SQL_QUERY),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &["Encoder.encodeForURL"],
|
|
label: DataLabel::Sanitizer(Cap::URL_ENCODE),
|
|
case_sensitive: false,
|
|
},
|
|
// OWASP ESAPI input validator, validates and canonicalizes input
|
|
LabelRule {
|
|
matchers: &["Validator.getValidInput"],
|
|
label: DataLabel::Sanitizer(Cap::all()),
|
|
case_sensitive: false,
|
|
},
|
|
// Type-check sanitizers, parsing to a primitive erases taint
|
|
LabelRule {
|
|
matchers: &[
|
|
"Integer.parseInt",
|
|
"Long.parseLong",
|
|
"Short.parseShort",
|
|
"Double.parseDouble",
|
|
"Integer.valueOf",
|
|
"Boolean.parseBoolean",
|
|
],
|
|
label: DataLabel::Sanitizer(Cap::all()),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &["URLEncoder.encode"],
|
|
label: DataLabel::Sanitizer(Cap::URL_ENCODE),
|
|
case_sensitive: false,
|
|
},
|
|
// Parameterized queries prevent SQL injection
|
|
LabelRule {
|
|
matchers: &["prepareStatement"],
|
|
label: DataLabel::Sanitizer(Cap::SQL_QUERY),
|
|
case_sensitive: false,
|
|
},
|
|
// ─────────── Sinks ─────────────
|
|
LabelRule {
|
|
matchers: &["Runtime.exec", "ProcessBuilder"],
|
|
label: DataLabel::Sink(Cap::SHELL_ESCAPE),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &["executeQuery", "executeUpdate"],
|
|
label: DataLabel::Sink(Cap::SQL_QUERY),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &["Class.forName"],
|
|
label: DataLabel::Sink(Cap::CODE_EXEC),
|
|
case_sensitive: false,
|
|
},
|
|
// HTTP response sinks, println/print are broad (also match System.out)
|
|
// but necessary to catch response.getWriter().println() via suffix matching.
|
|
LabelRule {
|
|
matchers: &["println", "print"],
|
|
label: DataLabel::Sink(Cap::HTML_ESCAPE),
|
|
case_sensitive: false,
|
|
},
|
|
// openConnection() is the standard java.net.URL API for initiating a connection.
|
|
// It is the correct interception point, the URL is already set on the object.
|
|
LabelRule {
|
|
matchers: &[
|
|
"openConnection",
|
|
"HttpClient.send",
|
|
"HttpClient.sendAsync",
|
|
"getForObject",
|
|
"RestTemplate.exchange",
|
|
"postForObject",
|
|
"postForEntity",
|
|
],
|
|
label: DataLabel::Sink(Cap::SSRF),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &[
|
|
"readObject",
|
|
"readUnshared",
|
|
"XMLDecoder.readObject",
|
|
"ObjectMapper.readValue",
|
|
],
|
|
label: DataLabel::Sink(Cap::DESERIALIZE),
|
|
case_sensitive: false,
|
|
},
|
|
// ─── Spring / JPA / Hibernate SQL sinks ───
|
|
LabelRule {
|
|
matchers: &[
|
|
"jdbcTemplate.query",
|
|
"jdbcTemplate.update",
|
|
"jdbcTemplate.execute",
|
|
"jdbcTemplate.queryForObject",
|
|
"jdbcTemplate.queryForList",
|
|
],
|
|
label: DataLabel::Sink(Cap::SQL_QUERY),
|
|
case_sensitive: false,
|
|
},
|
|
LabelRule {
|
|
matchers: &[
|
|
"entityManager.createNativeQuery",
|
|
"entityManager.createQuery",
|
|
"session.createQuery",
|
|
"session.createSQLQuery",
|
|
],
|
|
label: DataLabel::Sink(Cap::SQL_QUERY),
|
|
case_sensitive: true,
|
|
},
|
|
// NOTE: Java logging (logger.info, log.warn, etc.) removed as sinks ,
|
|
// logging format injection is not a real security vulnerability in Java.
|
|
// String.format also removed, it builds strings in memory (not a sink);
|
|
// the real sink is wherever the formatted string is used (SQL, HTTP, etc.).
|
|
// ─── JNDI injection sinks ───
|
|
LabelRule {
|
|
matchers: &[
|
|
"InitialContext.lookup",
|
|
"ctx.lookup",
|
|
"context.lookup",
|
|
"dirContext.lookup",
|
|
],
|
|
label: DataLabel::Sink(Cap::CODE_EXEC),
|
|
case_sensitive: false,
|
|
},
|
|
];
|
|
|
|
pub static KINDS: Map<&'static str, Kind> = phf_map! {
|
|
// control-flow
|
|
"if_statement" => Kind::If,
|
|
"while_statement" => Kind::While,
|
|
"for_statement" => Kind::For,
|
|
"enhanced_for_statement" => Kind::For,
|
|
"do_statement" => Kind::While,
|
|
|
|
"return_statement" => Kind::Return,
|
|
"throw_statement" => Kind::Throw,
|
|
"break_statement" => Kind::Break,
|
|
"continue_statement" => Kind::Continue,
|
|
|
|
// structure
|
|
"program" => Kind::SourceFile,
|
|
"block" => Kind::Block,
|
|
"class_declaration" => Kind::Block,
|
|
"class_body" => Kind::Block,
|
|
"interface_body" => Kind::Block,
|
|
"method_declaration" => Kind::Function,
|
|
"constructor_declaration" => Kind::Function,
|
|
"switch_expression" => Kind::Switch,
|
|
"switch_block" => Kind::Block,
|
|
"switch_block_statement_group" => Kind::Block,
|
|
"try_statement" => Kind::Try,
|
|
"try_with_resources_statement" => Kind::Try,
|
|
"resource_specification" => Kind::Block,
|
|
"resource" => Kind::CallWrapper,
|
|
"catch_clause" => Kind::Block,
|
|
"finally_clause" => Kind::Block,
|
|
"lambda_expression" => Kind::Function,
|
|
"constructor_body" => Kind::Block,
|
|
"static_initializer" => Kind::Block,
|
|
|
|
// data-flow
|
|
"method_invocation" => Kind::CallMethod,
|
|
"object_creation_expression" => Kind::CallFn,
|
|
"assignment_expression" => Kind::Assignment,
|
|
"local_variable_declaration" => Kind::CallWrapper,
|
|
"expression_statement" => Kind::CallWrapper,
|
|
"cast_expression" => Kind::Seq,
|
|
|
|
// trivia
|
|
"line_comment" => Kind::Trivia,
|
|
"block_comment" => Kind::Trivia,
|
|
";" => Kind::Trivia, "," => Kind::Trivia,
|
|
"(" => Kind::Trivia, ")" => Kind::Trivia,
|
|
"{" => Kind::Trivia, "}" => Kind::Trivia,
|
|
"\n" => Kind::Trivia,
|
|
"import_declaration" => Kind::Trivia,
|
|
"package_declaration" => Kind::Trivia,
|
|
};
|
|
|
|
pub static PARAM_CONFIG: ParamConfig = ParamConfig {
|
|
params_field: "parameters",
|
|
param_node_kinds: &["formal_parameter", "spread_parameter"],
|
|
self_param_kinds: &[],
|
|
ident_fields: &["name"],
|
|
};
|
|
|
|
/// Framework-conditional rules for Java.
|
|
pub fn framework_rules(ctx: &FrameworkContext) -> Vec<RuntimeLabelRule> {
|
|
let mut rules = Vec::new();
|
|
|
|
if ctx.has(DetectedFramework::Spring) {
|
|
// When Spring is detected, bare "send" is likely HttpClient.send()
|
|
rules.push(RuntimeLabelRule {
|
|
matchers: vec!["send".into()],
|
|
label: DataLabel::Sink(Cap::SSRF),
|
|
case_sensitive: false,
|
|
});
|
|
}
|
|
|
|
rules
|
|
}
|