#![allow(unused_imports)]
pub mod base;
pub mod content_classifier;
mod engine;
pub mod streaming_engine;
pub mod context_hmm;
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 ml_smells;
mod rust_smells;
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 data_flow;
pub mod ssa_flow;
pub mod taint;
pub mod function_context;
pub mod class_context;
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;
mod boolean_trap;
mod broad_exception;
mod callback_hell;
mod cleartext_credentials;
mod command_injection;
mod commented_code;
mod cors_misconfig;
mod dead_store;
mod debug_code;
mod deep_nesting;
mod dep_audit;
mod django_security;
mod duplicate_code;
mod empty_catch;
mod express_security;
mod gh_actions;
mod global_variables;
mod hardcoded_ips;
mod hardcoded_timeout;
mod implicit_coercion;
mod inconsistent_returns;
mod insecure_cookie;
mod insecure_crypto;
mod insecure_deserialize;
mod insecure_random;
mod insecure_tls;
mod jwt_weak;
mod large_files;
mod log_injection;
mod long_methods;
mod magic_numbers;
mod missing_await;
mod missing_docstrings;
mod mutable_default_args;
mod n_plus_one;
mod nosql_injection;
mod path_traversal;
mod prototype_pollution;
mod react_hooks;
mod regex_dos;
mod regex_in_loop;
mod secrets;
mod single_char_names;
mod ssrf;
mod string_concat_loop;
mod sync_in_async;
mod test_in_production;
mod todo_scanner;
mod unhandled_promise;
mod unreachable_code;
mod wildcard_imports;
mod xss;
mod xxe;
pub use base::{
DetectionSummary, Detector, DetectorConfig, DetectorResult, DetectorScope, 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 ml_smells::{
ChainIndexingDetector, DeprecatedTorchApiDetector, ForwardMethodDetector,
MissingRandomSeedDetector, MissingZeroGradDetector, NanEqualityDetector,
RequireGradTypoDetector, TorchLoadUnsafeDetector,
};
pub use rust_smells::{
BoxDynTraitDetector, CloneInHotPathDetector, MissingMustUseDetector,
MutexPoisoningRiskDetector, UnsafeWithoutSafetyCommentDetector, UnwrapWithoutContextDetector,
};
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, CachedScoreResult, 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 gh_actions::GHActionsInjectionDetector;
pub use boolean_trap::BooleanTrapDetector;
pub use broad_exception::BroadExceptionDetector;
pub use callback_hell::CallbackHellDetector;
pub use cleartext_credentials::CleartextCredentialsDetector;
pub use command_injection::CommandInjectionDetector;
pub use commented_code::CommentedCodeDetector;
pub use cors_misconfig::CorsMisconfigDetector;
pub use dead_store::DeadStoreDetector;
pub use debug_code::DebugCodeDetector;
pub use deep_nesting::DeepNestingDetector;
pub use dep_audit::DepAuditDetector;
pub use django_security::DjangoSecurityDetector;
pub use duplicate_code::DuplicateCodeDetector;
pub use empty_catch::EmptyCatchDetector;
pub use express_security::ExpressSecurityDetector;
pub use global_variables::GlobalVariablesDetector;
pub use hardcoded_ips::HardcodedIpsDetector;
pub use hardcoded_timeout::HardcodedTimeoutDetector;
pub use implicit_coercion::ImplicitCoercionDetector;
pub use inconsistent_returns::InconsistentReturnsDetector;
pub use insecure_cookie::InsecureCookieDetector;
pub use insecure_crypto::InsecureCryptoDetector;
pub use insecure_deserialize::InsecureDeserializeDetector;
pub use insecure_random::InsecureRandomDetector;
pub use insecure_tls::InsecureTlsDetector;
pub use jwt_weak::JwtWeakDetector;
pub use large_files::LargeFilesDetector;
pub use log_injection::LogInjectionDetector;
pub use long_methods::LongMethodsDetector;
pub use magic_numbers::MagicNumbersDetector;
pub use missing_await::MissingAwaitDetector;
pub use missing_docstrings::MissingDocstringsDetector;
pub use mutable_default_args::MutableDefaultArgsDetector;
pub use n_plus_one::NPlusOneDetector;
pub use nosql_injection::NosqlInjectionDetector;
pub use path_traversal::PathTraversalDetector;
pub use prototype_pollution::PrototypePollutionDetector;
pub use react_hooks::ReactHooksDetector;
pub use regex_dos::RegexDosDetector;
pub use regex_in_loop::RegexInLoopDetector;
pub use secrets::SecretDetector;
pub use single_char_names::SingleCharNamesDetector;
pub use ssrf::SsrfDetector;
pub use string_concat_loop::StringConcatLoopDetector;
pub use sync_in_async::SyncInAsyncDetector;
pub use test_in_production::TestInProductionDetector;
pub use todo_scanner::TodoScanner;
pub use unhandled_promise::UnhandledPromiseDetector;
pub use unreachable_code::UnreachableCodeDetector;
pub use wildcard_imports::WildcardImportsDetector;
pub use xss::XssDetector;
pub use xxe::XxeDetector;
pub use function_context::{
FunctionContext, FunctionContextBuilder, FunctionContextMap, FunctionRole,
};
use crate::config::ProjectConfig;
use std::path::Path;
use std::sync::Arc;
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>> {
default_detectors_with_profile(repository_path, project_config, None)
}
pub fn default_detectors_with_profile(
repository_path: &Path,
project_config: &ProjectConfig,
style_profile: Option<&crate::calibrate::StyleProfile>,
) -> Vec<Arc<dyn Detector>> {
let project_type = project_config.get_project_type(repository_path);
tracing::info!(
"Detected project type: {:?} (coupling multiplier: {:.1}x)",
project_type,
project_type.coupling_multiplier()
);
let resolver = crate::calibrate::ThresholdResolver::new(style_profile.cloned());
let make_config = |name: &str| -> DetectorConfig {
DetectorConfig::from_project_config_with_type(name, project_config, repository_path)
.with_adaptive(resolver.clone())
};
vec![
Arc::new(CircularDependencyDetector::new()),
Arc::new(GodClassDetector::with_config(make_config("GodClassDetector"))),
Arc::new(LongParameterListDetector::with_config(make_config("LongParameterListDetector"))),
Arc::new(DataClumpsDetector::with_config(make_config("DataClumpsDetector"))),
Arc::new(DeadCodeDetector::new()),
Arc::new(FeatureEnvyDetector::with_config(make_config("FeatureEnvyDetector"))),
Arc::new(InappropriateIntimacyDetector::new()),
Arc::new(LazyClassDetector::new()),
Arc::new(MessageChainDetector::new(repository_path)),
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(TorchLoadUnsafeDetector::new(repository_path)),
Arc::new(NanEqualityDetector::new(repository_path)),
Arc::new(MissingZeroGradDetector::new(repository_path)),
Arc::new(ForwardMethodDetector::new(repository_path)),
Arc::new(MissingRandomSeedDetector::new(repository_path)),
Arc::new(ChainIndexingDetector::new(repository_path)),
Arc::new(RequireGradTypoDetector::new(repository_path)),
Arc::new(DeprecatedTorchApiDetector::new(repository_path)),
Arc::new(ArchitecturalBottleneckDetector::with_config(make_config("ArchitecturalBottleneckDetector"))),
Arc::new(CoreUtilityDetector::new()),
Arc::new(DegreeCentralityDetector::with_config(make_config("DegreeCentralityDetector"))),
Arc::new(InfluentialCodeDetector::with_config(make_config("InfluentialCodeDetector"))),
Arc::new(ModuleCohesionDetector::with_config(make_config("ModuleCohesionDetector"))),
Arc::new(ShotgunSurgeryDetector::with_config(make_config("ShotgunSurgeryDetector"))),
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(UnsafeTemplateDetector::with_repository_path(
repository_path.to_path_buf(),
)),
Arc::new(GeneratorMisuseDetector::with_path(repository_path)),
Arc::new(InfiniteLoopDetector::with_path(repository_path)),
Arc::new(UnusedImportsDetector::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)),
Arc::new(EmptyCatchDetector::new(repository_path)),
Arc::new(TodoScanner::new(repository_path)),
Arc::new(DeepNestingDetector::with_resolver(repository_path, &resolver)),
Arc::new(MagicNumbersDetector::new(repository_path)),
Arc::new(LargeFilesDetector::with_resolver(repository_path, &resolver)),
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)),
Arc::new(UnwrapWithoutContextDetector::new(repository_path)),
Arc::new(UnsafeWithoutSafetyCommentDetector::new(repository_path)),
Arc::new(CloneInHotPathDetector::new(repository_path)),
Arc::new(MissingMustUseDetector::new(repository_path)),
Arc::new(BoxDynTraitDetector::new(repository_path)),
Arc::new(MutexPoisoningRiskDetector::new(repository_path)),
Arc::new(GHActionsInjectionDetector::new(repository_path)),
Arc::new(InsecureTlsDetector::new(repository_path)),
Arc::new(DepAuditDetector::new(repository_path)),
]
}
#[allow(dead_code)] pub fn create_default_engine(workers: usize, repository_path: &Path) -> DetectorEngine {
DetectorEngineBuilder::new()
.workers(workers)
.detectors(default_detectors(repository_path))
.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
}