Non-parametric trend analysis for unequally spaced time series with censored data
Project description
MannKS
(Mann-Kendall Sen)
Robust Trend Analysis in Python
📦 Installation
pip install mannks
Requirements: Python 3.7+, NumPy, Pandas, SciPy, Matplotlib, Piecewise-Regression
✨ What is MannKS?
MannKS (Mann-Kendall Sen) is a Python package for detecting trends in time series data using non-parametric methods. It's specifically designed for environmental monitoring, water quality analysis, and other fields where data is messy, irregular, or contains detection limits.
When to Use MannKS
Use this package when your data has:
- Irregular sampling intervals (daily → monthly → quarterly)
- Censored values (measurements like
<5or>100) - Seasonal patterns you need to account for
- No normal distribution (non-parametric methods don't require it)
- Small to moderate sample sizes (n < 5,000 recommended)
Don't use if you need n > 46,340 observations.
NEW IN V0.4.0: Segmented Trend Analysis. The segmented_trend_test function performs a hybrid segmented regression analysis. It uses Piecewise Regression (OLS) to automatically identify structural breakpoints in the time series, followed by robust Mann-Kendall / Sen's Slope estimation on each identified segment. This allows you to detect distinct phases in a trend (e.g., "Stable" -> "Rapid Decrease" -> "Stable").
NEW IN V0.3.0: Rolling Trend Analysis. The rolling_trend_test function allows you to perform a rolling window analysis, calculating the Sen's slope, Mann-Kendall score, and confidence intervals over time. This enables the detection of when a trend started, stopped, or changed direction, rather than just providing a single global summary. The feature includes:
- Flexible window sizes (numeric or time-based, e.g., '10 years')
compare_periodsutility to statistically test for changes in trend between two time periods (e.g., before vs. after an intervention).
NEW IN V0.2.0: The trend_test and seasonal_trend_test functions now support a Block Bootstrap method (autocorr_method='block_bootstrap'). This feature provides robust trend testing for data with serial correlation (autocorrelation) by resampling blocks of data rather than individual points, preserving the internal dependency structure.
See Statistical Methodology: Bootstrap for a detailed explanation of the hybrid methodology:
- Hypothesis Testing (P-values): Uses Detrended Residual Block Bootstrap to generate a null distribution while preserving autocorrelation.
- Confidence Intervals (Sen's Slope): Uses Pairs Block Bootstrap to avoid bias when "reconstructing" censored data values from residuals.
🚀 Quick Start
import pandas as pd
from MannKS import prepare_censored_data, trend_test
# 1. Prepare data with censored values
# Converts strings like '<5' into a structured format
values = [10, 12, '<5', 14, 15, 18, 20, '<5', 25, 30]
dates = pd.date_range(start='2020-01-01', periods=len(values), freq='ME')
data = prepare_censored_data(values)
# 2. Run trend test
# slope_scaling converts slope from "per second" to "per year"
result = trend_test(
x=data,
t=dates,
slope_scaling='year',
x_unit='mg/L',
plot_path='trend.png'
)
# 3. Interpret results
print(f"Trend: {result.classification}")
print(f"Slope: {result.slope:.2f} {result.slope_units}")
print(f"Confidence: {result.C:.2%}")
Output:
Trend: Highly Likely Increasing
Slope: 24.57 mg/L per year
Confidence: 98.47%
🎯 Key Features
Core Functionality
- Mann-Kendall Trend Test: Detect monotonic trends with statistical significance
- Sen's Slope Estimator: Calculate trend magnitude with confidence intervals
- Seasonal Analysis: Separate seasonal signals from long-term trends
- Regional Aggregation: Combine results across multiple monitoring sites
Data Handling
- Censored Data Support: Native handling of detection limits (
<5,>100)- Three methods: Standard, LWP-compatible, Akritas-Theil-Sen (ATS)
- Handles left-censored, right-censored, and mixed censoring
- Unequal Spacing: Uses actual time differences (not just rank order)
- Missing Data: Automatically handles NaN values and missing seasons
- Temporal Aggregation: Multiple strategies for high-frequency data
Statistical Features
- Continuous Confidence: Reports likelihood ("Highly Likely Increasing") not just p-values
- Data Quality Checks: Automatic warnings for tied values, long runs, insufficient data
- Robust Methods: ATS estimator for heavily censored data
- Flexible Testing: Kendall's Tau-a or Tau-b, custom significance levels
- Rolling Trends (New in v0.3.0): Analyze how trends evolve over time with
rolling_trend_test. See Example 31. - Segmented Trends (New in v0.4.0): Automatically detect breakpoints and analyze trends in segments with
segmented_trend_test. See Example 32. - Block Bootstrap (New in v0.2.0): Robust trend testing for autocorrelated data with automatic ACF-based block size selection. See bootstrap.md for details and Example 29.
📊 Example Use Cases
Seasonal Water Quality Trend
from MannKS import seasonal_trend_test, check_seasonality
# Check if seasonality exists (period=12 is inferred from season_type='month')
seasonality = check_seasonality(x=data, t=dates, season_type='month')
print(f"Seasonal pattern detected: {seasonality.is_seasonal}")
# Run seasonal trend test
result = seasonal_trend_test(
x=data,
t=dates,
season_type='month', # Infers period=12 automatically
agg_method='robust_median', # Aggregates multiple samples per month
slope_scaling='year'
)
Regional Analysis Across Sites
from MannKS import regional_test
# Run trend tests for each site
site_results = []
for site in ['Site_A', 'Site_B', 'Site_C']:
result = trend_test(x=site_data[site], t=dates)
site_results.append({
'site': site,
's': result.s,
'C': result.C
})
# Aggregate regional trend
regional = regional_test(
trend_results=pd.DataFrame(site_results),
time_series_data=all_site_data,
site_col='site'
)
print(f"Regional trend: {regional.DT}, confidence: {regional.CT:.2%}")
⚠️ Important Limitations
Sample Size
- Recommended maximum: n = 5,000 (triggers memory warning)
- Hard limit: n = 46,340 (prevents integer overflow)
- For larger datasets, use
regional_test()to aggregate multiple smaller sites
Statistical Assumptions
- Independence: Data points must be serially independent
- Autocorrelation violates this and causes spurious significance
- Pre-test with ACF or use block bootstrap methods if autocorrelated
- Monotonic trend: Cannot detect U-shaped or cyclical patterns
- Homogeneous variance: Most powerful when variance is constant over time
📚 Documentation
Detailed Guides
- Trend Test Parameters - Full parameter reference
- Seasonal Analysis - Season types and aggregation
- Regional Tests - Multi-site aggregation
- Analysis Notes - Interpreting data quality warnings
- Trend Classification - Understanding confidence levels
- Bootstrap Methodology - Block bootstrap for autocorrelated data
- Rolling Trend Analysis - Moving window analysis
- Segmented Trend Analysis - Structural breakpoint detection
Examples
The Examples folder contains step-by-step tutorials from basic to advanced usage.
🔬 Validation
Extensively validated against:
- LWP-TRENDS R script (34 test cases, 99%+ agreement)
- NADA2 R package (censored data methods)
- Edge cases: missing data, tied values, all-censored data, insufficient samples
See validation/ for detailed comparison reports.
🙏 Acknowledgments
This package is heavily inspired by the excellent work of LandWaterPeople (LWP). The robust censored data handling and regional aggregation methods are based on their R scripts and methodologies.
📖 References
- Helsel, D.R. (2012). Statistics for Censored Environmental Data Using Minitab and R (2nd ed.). Wiley.
- Gilbert, R.O. (1987). Statistical Methods for Environmental Pollution Monitoring. Wiley.
- Hirsch, R.M., Slack, J.R., & Smith, R.A. (1982). Techniques of trend analysis for monthly water quality data. Water Resources Research, 18(1), 107-121.
- Mann, H.B. (1945). Nonparametric tests against trend. Econometrica, 13(3), 245-259.
- Sen, P.K. (1968). Estimates of the regression coefficient based on a particular kind of rank correlation. Journal of the American Statistical Association, 63(324), 1379-1389.
- Fraser, C., & Whitehead, A. L. (2022). Continuous measures of confidence in direction of environmental trends at site and other spatial scales. Environmental Challenges, 9, 100601.
- Fraser, C., Snelder, T., & Matthews, A. (2018). State and trends of river water quality in the Manawatu-Whanganui region. Report for Horizons Regional Council.
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file mannks-0.4.1.tar.gz.
File metadata
- Download URL: mannks-0.4.1.tar.gz
- Upload date:
- Size: 70.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.7.1 CPython/3.10.6 Linux/4.15.0-142-generic
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
398d0acf31804f50a38c6411e7006f1d88b727c4171883f0966b876e450a3577
|
|
| MD5 |
4c554c4f3bee8b2687d3d222eafd6cd6
|
|
| BLAKE2b-256 |
d46fa5ce7f37e2cfc8cd313a0dee67ca40af0d552cc5483ac6ced2ad67d7d27f
|
File details
Details for the file mannks-0.4.1-py3-none-any.whl.
File metadata
- Download URL: mannks-0.4.1-py3-none-any.whl
- Upload date:
- Size: 81.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.7.1 CPython/3.10.6 Linux/4.15.0-142-generic
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
9be3a74878da40802793a38ccea07b53d4e96cb4d3665202e233e069363e8dd7
|
|
| MD5 |
a599e34e5ab57932ac8fa9bc19b233a6
|
|
| BLAKE2b-256 |
672452ed5722685c717b7ca3677aeafab711e0c9373bae9f6f93d5ef2c3a9b29
|