#![allow(unused_imports)]
mod base;
mod engine;
mod circular_dependency;
mod god_class;
mod long_parameter;
mod data_clumps;
mod dead_code;
mod feature_envy;
mod inappropriate_intimacy;
mod lazy_class;
mod message_chain;
mod middle_man;
mod refused_bequest;
mod ai_boilerplate;
mod ai_churn;
mod ai_complexity_spike;
mod ai_duplicate_block;
mod ai_missing_tests;
mod ai_naming_pattern;
mod architectural_bottleneck;
mod core_utility;
mod degree_centrality;
mod influential_code;
mod module_cohesion;
mod shotgun_surgery;
mod eval_detector;
mod pickle_detector;
mod sql_injection;
mod taint_detector;
mod unsafe_template;
pub mod taint;
pub mod framework_detection;
mod generator_misuse;
mod infinite_loop;
mod unused_imports;
mod health_delta;
mod incremental_cache;
mod query_cache;
mod risk_analyzer;
mod root_cause_analyzer;
mod voting_engine;
pub mod external_tool;
mod bandit;
mod eslint;
mod gh_actions;
mod mypy;
mod npm_audit;
mod radon;
mod ruff;
mod semgrep;
mod tsc;
mod vulture;
mod secrets;
mod empty_catch;
mod todo_scanner;
mod deep_nesting;
mod magic_numbers;
mod large_files;
mod path_traversal;
mod command_injection;
mod ssrf;
mod missing_docstrings;
mod regex_dos;
mod sync_in_async;
mod n_plus_one;
mod insecure_crypto;
mod xss;
mod hardcoded_ips;
mod insecure_random;
mod cors_misconfig;
mod debug_code;
mod commented_code;
mod long_methods;
mod duplicate_code;
mod unreachable_code;
mod string_concat_loop;
mod xxe;
mod insecure_deserialize;
mod cleartext_credentials;
mod wildcard_imports;
mod mutable_default_args;
mod global_variables;
mod implicit_coercion;
mod single_char_names;
mod missing_await;
mod unhandled_promise;
mod callback_hell;
mod test_in_production;
mod insecure_cookie;
mod jwt_weak;
mod prototype_pollution;
mod nosql_injection;
mod log_injection;
mod broad_exception;
mod boolean_trap;
mod inconsistent_returns;
mod dead_store;
mod hardcoded_timeout;
mod regex_in_loop;
mod react_hooks;
mod django_security;
mod express_security;
pub use base::{
DetectionSummary,
Detector,
DetectorConfig,
DetectorResult,
ProgressCallback,
};
pub use engine::{
DetectorEngine,
DetectorEngineBuilder,
};
pub use circular_dependency::CircularDependencyDetector;
pub use god_class::{GodClassDetector, GodClassThresholds};
pub use long_parameter::{LongParameterListDetector, LongParameterThresholds};
pub use data_clumps::DataClumpsDetector;
pub use dead_code::DeadCodeDetector;
pub use feature_envy::FeatureEnvyDetector;
pub use inappropriate_intimacy::InappropriateIntimacyDetector;
pub use lazy_class::LazyClassDetector;
pub use message_chain::MessageChainDetector;
pub use middle_man::MiddleManDetector;
pub use refused_bequest::RefusedBequestDetector;
pub use ai_boilerplate::AIBoilerplateDetector;
pub use ai_churn::AIChurnDetector;
pub use ai_complexity_spike::AIComplexitySpikeDetector;
pub use ai_duplicate_block::AIDuplicateBlockDetector;
pub use ai_missing_tests::AIMissingTestsDetector;
pub use ai_naming_pattern::AINamingPatternDetector;
pub use architectural_bottleneck::ArchitecturalBottleneckDetector;
pub use core_utility::CoreUtilityDetector;
pub use degree_centrality::DegreeCentralityDetector;
pub use influential_code::InfluentialCodeDetector;
pub use module_cohesion::ModuleCohesionDetector;
pub use shotgun_surgery::ShotgunSurgeryDetector;
pub use eval_detector::EvalDetector;
pub use pickle_detector::PickleDeserializationDetector;
pub use sql_injection::SQLInjectionDetector;
pub use taint_detector::TaintDetector;
pub use unsafe_template::UnsafeTemplateDetector;
pub use generator_misuse::GeneratorMisuseDetector;
pub use infinite_loop::InfiniteLoopDetector;
pub use unused_imports::UnusedImportsDetector;
pub use health_delta::{
estimate_batch_fix_impact, estimate_fix_impact, BatchHealthScoreDelta, HealthScoreDelta,
HealthScoreDeltaCalculator, ImpactLevel, MetricsBreakdown,
};
pub use incremental_cache::{CacheStats, IncrementalCache};
pub use query_cache::{ClassData, FileData, FunctionData, QueryCache};
pub use risk_analyzer::{
analyze_compound_risks, RiskAnalyzer, RiskAssessment, RiskFactor,
};
pub use root_cause_analyzer::{RootCauseAnalysis, RootCauseAnalyzer, RootCauseSummary};
pub use voting_engine::{
ConfidenceMethod, ConsensusResult, DetectorWeight, SeverityResolution, VotingEngine,
VotingStats, VotingStrategy,
};
pub use bandit::BanditDetector;
pub use eslint::ESLintDetector;
pub use gh_actions::GHActionsInjectionDetector;
pub use mypy::MypyDetector;
pub use npm_audit::NpmAuditDetector;
pub use radon::RadonDetector;
pub use ruff::{RuffImportDetector, RuffLintDetector};
pub use semgrep::SemgrepDetector;
pub use tsc::TscDetector;
pub use vulture::VultureDetector;
pub use secrets::SecretDetector;
pub use empty_catch::EmptyCatchDetector;
pub use todo_scanner::TodoScanner;
pub use deep_nesting::DeepNestingDetector;
pub use magic_numbers::MagicNumbersDetector;
pub use large_files::LargeFilesDetector;
pub use path_traversal::PathTraversalDetector;
pub use command_injection::CommandInjectionDetector;
pub use ssrf::SsrfDetector;
pub use missing_docstrings::MissingDocstringsDetector;
pub use regex_dos::RegexDosDetector;
pub use sync_in_async::SyncInAsyncDetector;
pub use n_plus_one::NPlusOneDetector;
pub use insecure_crypto::InsecureCryptoDetector;
pub use xss::XssDetector;
pub use hardcoded_ips::HardcodedIpsDetector;
pub use insecure_random::InsecureRandomDetector;
pub use cors_misconfig::CorsMisconfigDetector;
pub use debug_code::DebugCodeDetector;
pub use commented_code::CommentedCodeDetector;
pub use long_methods::LongMethodsDetector;
pub use duplicate_code::DuplicateCodeDetector;
pub use unreachable_code::UnreachableCodeDetector;
pub use string_concat_loop::StringConcatLoopDetector;
pub use xxe::XxeDetector;
pub use insecure_deserialize::InsecureDeserializeDetector;
pub use cleartext_credentials::CleartextCredentialsDetector;
pub use wildcard_imports::WildcardImportsDetector;
pub use mutable_default_args::MutableDefaultArgsDetector;
pub use global_variables::GlobalVariablesDetector;
pub use implicit_coercion::ImplicitCoercionDetector;
pub use single_char_names::SingleCharNamesDetector;
pub use missing_await::MissingAwaitDetector;
pub use unhandled_promise::UnhandledPromiseDetector;
pub use callback_hell::CallbackHellDetector;
pub use test_in_production::TestInProductionDetector;
pub use insecure_cookie::InsecureCookieDetector;
pub use jwt_weak::JwtWeakDetector;
pub use prototype_pollution::PrototypePollutionDetector;
pub use nosql_injection::NosqlInjectionDetector;
pub use log_injection::LogInjectionDetector;
pub use broad_exception::BroadExceptionDetector;
pub use boolean_trap::BooleanTrapDetector;
pub use inconsistent_returns::InconsistentReturnsDetector;
pub use dead_store::DeadStoreDetector;
pub use hardcoded_timeout::HardcodedTimeoutDetector;
pub use regex_in_loop::RegexInLoopDetector;
pub use react_hooks::ReactHooksDetector;
pub use django_security::DjangoSecurityDetector;
pub use express_security::ExpressSecurityDetector;
pub use external_tool::{
ExternalToolResult,
GraphContext,
JsRuntime,
batch_get_graph_context,
get_graph_context,
get_js_exec_command,
get_js_runtime,
is_python_tool_installed,
is_tool_installed,
run_external_tool,
run_js_tool,
};
use std::path::Path;
use std::sync::Arc;
use crate::config::ProjectConfig;
pub fn default_detectors(repository_path: &Path) -> Vec<Arc<dyn Detector>> {
default_detectors_with_config(repository_path, &ProjectConfig::default())
}
pub fn default_detectors_with_config(repository_path: &Path, project_config: &ProjectConfig) -> Vec<Arc<dyn Detector>> {
vec![
Arc::new(CircularDependencyDetector::new()),
Arc::new(GodClassDetector::with_config(
DetectorConfig::from_project_config("GodClassDetector", project_config)
)),
Arc::new(LongParameterListDetector::with_config(
DetectorConfig::from_project_config("LongParameterListDetector", project_config)
)),
Arc::new(DataClumpsDetector::new()),
Arc::new(DeadCodeDetector::new()),
Arc::new(FeatureEnvyDetector::new()),
Arc::new(InappropriateIntimacyDetector::new()),
Arc::new(LazyClassDetector::new()),
Arc::new(MessageChainDetector::new()),
Arc::new(MiddleManDetector::new()),
Arc::new(RefusedBequestDetector::new()),
Arc::new(AIBoilerplateDetector::new()),
Arc::new(AIChurnDetector::new()),
Arc::new(AIComplexitySpikeDetector::new()),
Arc::new(AIDuplicateBlockDetector::new()),
Arc::new(AIMissingTestsDetector::new()),
Arc::new(AINamingPatternDetector::new()),
Arc::new(ArchitecturalBottleneckDetector::new()),
Arc::new(CoreUtilityDetector::new()),
Arc::new(DegreeCentralityDetector::new()),
Arc::new(InfluentialCodeDetector::new()),
Arc::new(ModuleCohesionDetector::new()),
Arc::new(ShotgunSurgeryDetector::new()),
Arc::new(EvalDetector::with_repository_path(repository_path.to_path_buf())),
Arc::new(PickleDeserializationDetector::with_repository_path(repository_path.to_path_buf())),
Arc::new(SQLInjectionDetector::with_repository_path(repository_path.to_path_buf())),
Arc::new(TaintDetector::with_repository_path(repository_path.to_path_buf())),
Arc::new(UnsafeTemplateDetector::with_repository_path(repository_path.to_path_buf())),
Arc::new(GeneratorMisuseDetector::new()),
Arc::new(InfiniteLoopDetector::new()),
Arc::new(UnusedImportsDetector::new()),
Arc::new(SecretDetector::new(repository_path)),
Arc::new(PathTraversalDetector::new(repository_path)),
Arc::new(CommandInjectionDetector::new(repository_path)),
Arc::new(SsrfDetector::new(repository_path)),
Arc::new(RegexDosDetector::new(repository_path)),
Arc::new(EmptyCatchDetector::new(repository_path)),
Arc::new(TodoScanner::new(repository_path)),
Arc::new(DeepNestingDetector::new(repository_path)),
Arc::new(MagicNumbersDetector::new(repository_path)),
Arc::new(LargeFilesDetector::new(repository_path)),
Arc::new(MissingDocstringsDetector::new(repository_path)),
Arc::new(SyncInAsyncDetector::new(repository_path)),
Arc::new(NPlusOneDetector::new(repository_path)),
Arc::new(InsecureCryptoDetector::new(repository_path)),
Arc::new(XssDetector::new(repository_path)),
Arc::new(HardcodedIpsDetector::new(repository_path)),
Arc::new(InsecureRandomDetector::new(repository_path)),
Arc::new(CorsMisconfigDetector::new(repository_path)),
Arc::new(DebugCodeDetector::new(repository_path)),
Arc::new(CommentedCodeDetector::new(repository_path)),
Arc::new(LongMethodsDetector::with_config(
repository_path,
DetectorConfig::from_project_config("long-methods", project_config)
)),
Arc::new(DuplicateCodeDetector::new(repository_path)),
Arc::new(UnreachableCodeDetector::new(repository_path)),
Arc::new(StringConcatLoopDetector::new(repository_path)),
Arc::new(XxeDetector::new(repository_path)),
Arc::new(InsecureDeserializeDetector::new(repository_path)),
Arc::new(CleartextCredentialsDetector::new(repository_path)),
Arc::new(WildcardImportsDetector::new(repository_path)),
Arc::new(MutableDefaultArgsDetector::new(repository_path)),
Arc::new(GlobalVariablesDetector::new(repository_path)),
Arc::new(ImplicitCoercionDetector::new(repository_path)),
Arc::new(SingleCharNamesDetector::new(repository_path)),
Arc::new(MissingAwaitDetector::new(repository_path)),
Arc::new(UnhandledPromiseDetector::new(repository_path)),
Arc::new(CallbackHellDetector::new(repository_path)),
Arc::new(TestInProductionDetector::new(repository_path)),
Arc::new(InsecureCookieDetector::new(repository_path)),
Arc::new(JwtWeakDetector::new(repository_path)),
Arc::new(PrototypePollutionDetector::new(repository_path)),
Arc::new(NosqlInjectionDetector::new(repository_path)),
Arc::new(LogInjectionDetector::new(repository_path)),
Arc::new(BroadExceptionDetector::new(repository_path)),
Arc::new(BooleanTrapDetector::new(repository_path)),
Arc::new(InconsistentReturnsDetector::new(repository_path)),
Arc::new(DeadStoreDetector::new(repository_path)),
Arc::new(HardcodedTimeoutDetector::new(repository_path)),
Arc::new(RegexInLoopDetector::new(repository_path)),
Arc::new(ReactHooksDetector::new(repository_path)),
Arc::new(DjangoSecurityDetector::new(repository_path)),
Arc::new(ExpressSecurityDetector::new(repository_path)),
]
}
pub fn python_detectors(repository_path: &Path) -> Vec<Arc<dyn Detector>> {
vec![
Arc::new(BanditDetector::new(repository_path)),
Arc::new(RuffLintDetector::new(repository_path)),
Arc::new(RuffImportDetector::new(repository_path)),
Arc::new(MypyDetector::new(repository_path)),
Arc::new(RadonDetector::new(repository_path)),
Arc::new(VultureDetector::new(repository_path)),
]
}
pub fn javascript_detectors(repository_path: &Path) -> Vec<Arc<dyn Detector>> {
vec![
Arc::new(ESLintDetector::new(repository_path)),
Arc::new(TscDetector::new(repository_path)),
Arc::new(NpmAuditDetector::new(repository_path)),
]
}
pub fn security_detectors(repository_path: &Path) -> Vec<Arc<dyn Detector>> {
vec![
Arc::new(BanditDetector::new(repository_path)),
Arc::new(SemgrepDetector::new(repository_path)),
Arc::new(NpmAuditDetector::new(repository_path)),
Arc::new(GHActionsInjectionDetector::new(repository_path)),
Arc::new(SecretDetector::new(repository_path)),
Arc::new(PathTraversalDetector::new(repository_path)),
Arc::new(CommandInjectionDetector::new(repository_path)),
Arc::new(SsrfDetector::new(repository_path)),
Arc::new(RegexDosDetector::new(repository_path)),
]
}
pub fn all_external_detectors(repository_path: &Path) -> Vec<Arc<dyn Detector>> {
vec![
Arc::new(BanditDetector::new(repository_path)),
Arc::new(RuffLintDetector::new(repository_path)),
Arc::new(RuffImportDetector::new(repository_path)),
Arc::new(MypyDetector::new(repository_path)),
Arc::new(RadonDetector::new(repository_path)),
Arc::new(VultureDetector::new(repository_path)),
Arc::new(ESLintDetector::new(repository_path)),
Arc::new(TscDetector::new(repository_path)),
Arc::new(NpmAuditDetector::new(repository_path)),
Arc::new(SemgrepDetector::new(repository_path)),
Arc::new(GHActionsInjectionDetector::new(repository_path)),
]
}
pub fn create_default_engine(workers: usize, repository_path: &Path) -> DetectorEngine {
DetectorEngineBuilder::new()
.workers(workers)
.detectors(default_detectors(repository_path))
.build()
}
pub fn create_full_engine(workers: usize, repository_path: &Path) -> DetectorEngine {
let mut detectors = default_detectors(repository_path);
detectors.extend(all_external_detectors(repository_path));
DetectorEngineBuilder::new()
.workers(workers)
.detectors(detectors)
.build()
}
pub fn walk_source_files<'a>(
repository_path: &'a Path,
extensions: Option<&'a [&'a str]>,
) -> impl Iterator<Item = std::path::PathBuf> + 'a {
use ignore::WalkBuilder;
let mut builder = WalkBuilder::new(repository_path);
builder
.hidden(true) .git_ignore(true) .git_global(true) .git_exclude(true) .require_git(false) .add_custom_ignore_filename(".repotoireignore");
builder.build().filter_map(move |entry| {
let entry = entry.ok()?;
let path = entry.path();
if !path.is_file() {
return None;
}
if let Some(exts) = extensions {
let ext = path.extension()?.to_str()?;
if !exts.contains(&ext) {
return None;
}
}
Some(path.to_path_buf())
})
}
pub fn is_line_suppressed(line: &str, prev_line: Option<&str>) -> bool {
let suppression_pattern = "repotoire: ignore";
let suppression_pattern_alt = "repotoire:ignore";
let line_lower = line.to_lowercase();
if line_lower.contains(suppression_pattern) || line_lower.contains(suppression_pattern_alt) {
return true;
}
if let Some(prev) = prev_line {
let prev_lower = prev.trim().to_lowercase();
if (prev_lower.starts_with('#') || prev_lower.starts_with("//") ||
prev_lower.starts_with("--") || prev_lower.starts_with("/*")) &&
(prev_lower.contains(suppression_pattern) || prev_lower.contains(suppression_pattern_alt)) {
return true;
}
}
false
}