A scikit-style recommender systems library
Project description
scikit-rec
A composable, scikit-style recommender systems library.
scikit-rec provides a 3-layer architecture that cleanly separates business logic, scoring strategy, and ML models. Any recommender works with any compatible scorer and estimator, giving you a mix-and-match toolkit for building recommendation systems.
Recommender (business logic) --> Scorer (item scoring) --> Estimator (ML model)
Why scikit-rec?
Composable by design. Each layer is independently extensible. Swap XGBoost for a Two-Tower model without changing your recommender. Add a new bandit strategy without touching the scorer. The library spans XGBoost, LightGBM, and scikit-learn alongside deep learning models (NCF, Two-Tower, DeepFM, SASRec, HRNN), with GPU optional — a pure-NumPy matrix factorization (ALS/SGD) requires no PyTorch. The composable architecture also accommodates novel research: a Goal-Conditioned Supervised Learning (GCSL) recommender for multi-objective recommendation was implemented as a single Recommender subclass — no new scorer or estimator required. Contributions welcome: implement one abstract class and it works with everything else.
Beyond ranking. Contextual bandits (epsilon-greedy, static-action) and heterogeneous treatment effect estimation (T/S/X-Learner) are first-class paradigms, not afterthoughts. All share the same evaluation infrastructure, so you can directly compare a ranking policy against a bandit or uplift policy on the same logged data.
Production-grade evaluation. The most complete offline policy evaluation suite in any recommendation library: IPS, Doubly Robust, SNIPS, Direct Method, Policy-Weighted, and Replay Match, paired with ten ranking, classification, and regression metrics (Precision, Recall, MAP, MRR, NDCG, ROC-AUC, PR-AUC, Expected Reward, RMSE, MAE) — enabling counterfactual policy comparison from logged data with a single call. Multi-label classification and multi-target regression workloads (wide-format MultioutputScorer) get per-label diagnostics and macro-averaged metrics out of the same evaluate() API.
Production readiness. Config-driven pipeline factory with Optuna HPO, low-latency single-user inference (recommend_online), two-stage retrieval-then-ranking, and batch training.
Agent-friendly. Optionally pair with scikit-rec-agent so an LLM agent can build, train, and tune models against this library's contracts.
Learn by example. Ten end-to-end Jupyter notebooks on MovieLens 1M cover ranking, bandits, uplift, sequential recommendations, multi-objective optimization, hyperparameter tuning, two-stage retrieval, and contextual two-tower models. Our SASRec achieves HR@10 = 0.8953 and NDCG@10 = 0.6331 on MovieLens-1M (leave-last-out, 1 positive + 100 negatives). Each notebook downloads data, trains, evaluates, and shows sample recommendations — ready to run.
Installation
pip install scikit-rec
Optional extras:
pip install scikit-rec[torch] # Deep learning models (DeepFM, NCF, SASRec, HRNN, Two-Tower)
pip install scikit-rec[aws] # S3 data loading
pip install scikit-rec[torch,aws] # All optional extras
Notes:
scikit-rec[torch]is required only for PyTorch-based estimators and deep learning pipelines.scikit-rec[aws]adds S3 dataset loading support without pulling in Torch.
Quick Start
from skrec.estimator.classification.xgb_classifier import XGBClassifierEstimator
from skrec.scorer.universal import UniversalScorer
from skrec.recommender.ranking.ranking_recommender import RankingRecommender
from skrec.examples.datasets import (
sample_binary_reward_interactions,
sample_binary_reward_users,
sample_binary_reward_items,
)
# Build the pipeline: Estimator -> Scorer -> Recommender
estimator = XGBClassifierEstimator({"learning_rate": 0.1, "max_depth": 5})
scorer = UniversalScorer(estimator)
recommender = RankingRecommender(scorer)
# Train
recommender.train(
interactions_ds=sample_binary_reward_interactions,
users_ds=sample_binary_reward_users,
items_ds=sample_binary_reward_items,
)
# Recommend
interactions_df = sample_binary_reward_interactions.fetch_data()
users_df = sample_binary_reward_users.fetch_data()
recommendations = recommender.recommend(interactions=interactions_df, users=users_df, top_k=5)
Config-driven quick start
The orchestrator API lets you build a pipeline from a configuration dictionary instead of wiring classes manually.
from skrec.orchestrator import create_recommender_pipeline
config = {
"recommender_type": "ranking",
"scorer_type": "universal",
"estimator_config": {
"ml_task": "classification",
"xgboost": {
"n_estimators": 100,
"max_depth": 5,
"learning_rate": 0.1,
},
},
}
recommender = create_recommender_pipeline(config)
recommender.train(
interactions_ds=sample_binary_reward_interactions,
users_ds=sample_binary_reward_users,
items_ds=sample_binary_reward_items,
)
Agent integration
scikit-rec is designed to be consumed by LLM agents via its config-driven factory and a capability_matrix() introspection API. The companion repo scikit-rec-agent provides a ready-made conversational agent that uses this surface to guide a user through dataset preparation, model selection, training, and evaluation — no hand-coded class wiring required.
How the agent surface works
capability_matrix() returns the authoritative set of valid types for every factory dimension. The factory validates its inputs against these same tuples, so the agent's knowledge is always in lockstep with what the library actually supports:
from skrec.orchestrator import capability_matrix
cm = capability_matrix()
cm["recommender_types"] # ('ranking', 'bandits', 'sequential', 'hierarchical_sequential', 'uplift', 'gcsl')
cm["scorer_types"] # ('universal', 'independent', 'multiclass', 'multioutput', 'sequential', 'hierarchical')
cm["estimator_types"] # ('tabular', 'embedding', 'sequential')
cm["embedding_model_types"] # ('matrix_factorization', 'ncf', 'two_tower', ...)
cm["scorer_config_keys"] # per-scorer accepted kwargs, e.g. {"multioutput": ("on_degenerate_target",)}
An agent can call capability_matrix() to populate its system prompt, then pass the user's choices directly to create_recommender_pipeline(). The factory validates compatibility upfront and raises a clear ValueError for any invalid combination before training begins.
Components
Recommenders
| Recommender | Description |
|---|---|
RankingRecommender |
Rank items by predicted score |
ContextualBanditsRecommender |
Exploration-exploitation strategies (epsilon-greedy, static action) |
UpliftRecommender |
Uplift modeling (S-Learner, T-Learner, X-Learner) |
SequentialRecommender |
Sequence-aware recommendations |
HierarchicalSequentialRecommender |
Session-aware hierarchical sequences (HRNN) |
GcslRecommender |
Multi-objective goal-conditioned supervised learning |
Scorers
| Scorer | Description |
|---|---|
UniversalScorer |
Single global model using item features (auto-dispatches tabular vs. embedding) |
IndependentScorer |
Separate model per item |
MulticlassScorer |
Items as competing classes |
MultioutputScorer |
Wide-format multi-label binary classification or multi-target regression (one ITEM_<name> column per target); accepts scorer_config={"on_degenerate_target": "constant"} to handle single-class training columns |
MixedTypeMultiTargetScorer |
Wide-format scorer for heterogeneous per-target types (binary + regression + multiclass + multilabel groups) in one model. Three estimator families: joint MLP, joint Transformer (FT-Transformer-style), independent (per-target sub-estimators). Per-target metric dispatch returns Dict[str, float]. See scorers guide |
SequentialScorer |
For sequential estimators (SASRec) |
HierarchicalScorer |
For HRNN estimators |
Estimators
| Type | Models |
|---|---|
| Tabular | XGBoost, LightGBM, Logistic Regression, sklearn classifiers/regressors, DeepFM |
| Embedding | Matrix Factorization, NCF, Two-Tower, Deep Cross Network, Neural Factorization Machine |
| Sequential | SASRec, HRNN |
Evaluators
| Evaluator | Description |
|---|---|
SimpleEvaluator |
Standard offline evaluation on held-out data |
IPSEvaluator |
Inverse Propensity Scoring for counterfactual evaluation |
DREvaluator |
Doubly Robust — combines direct estimation with IPS |
SNIPSEvaluator |
Self-Normalized IPS — reduces variance of IPS |
DirectMethodEvaluator |
Uses a reward model to estimate policy value |
PolicyWeightedEvaluator |
Weights logged rewards by policy/logging probability ratio |
ReplayMatchEvaluator |
Unbiased evaluation using only matching logged actions |
Metrics
Precision@k, Recall@k, MAP, MRR, NDCG, ROC-AUC, PR-AUC, Expected Reward, RMSE, MAE (the last two for multi-target regression via MultioutputScorer).
Retrievers
Two-stage retrieval: Popularity, Content-Based, Embedding-Based.
Example Notebooks
Notebooks are grouped by dataset under examples/:
MovieLens-1M (examples/movielens-1m/)
| Notebook | What it demonstrates |
|---|---|
| Ranking with XGBoost | Feature-based ranking with demographics and genre features |
| GCSL Multi-Objective | Goal-conditioned recommendations — steer quality vs. novelty |
| HPO with Optuna | Hyperparameter tuning with TPE, GP, and CMA-ES samplers |
| Two-Tower Models | Three context modes: user_tower, trilinear, scoring_layer |
| SASRec (Positives) | Self-attentive sequential recommendation on positive interactions |
| SASRec (Ratings) | SASRec with explicit ratings as soft labels |
| SASRec (MSE) | SASRec regressor with MSE loss |
| HRNN | Hierarchical RNN for session-aware recommendations |
Amazon Books (examples/amazon-books/)
| Notebook | What it demonstrates |
|---|---|
| LightGBM | Fast tabular ranking on Amazon Books with categorical book metadata |
| DeepFM | Sparse-feature interaction learning with FM + MLP + cross network |
| NCF | Neural Collaborative Filtering with NeuMF + embedding-based two-stage retrieval |
| SASRec (Positives) | Sequential recommendation — mirrors the original SASRec paper's Amazon protocol |
Generic (examples/generic/) — dataset-agnostic library mechanics
| Notebook | What it demonstrates |
|---|---|
| Factory Pipeline | Config-driven recommender construction on the shipped sample dataset |
| Two-Stage Retrieval | Popularity, content-based, and embedding retrieval + ranking on synthetic data |
| Uplift Modeling | S-Learner, T-Learner, X-Learner treatment effect estimation on synthetic data |
The MovieLens-1M notebooks download data automatically from GroupLens. The four Amazon Books
notebooks share a cached download from McAuley-Lab's Amazon Reviews 2023 via HuggingFace
datasets (the first run pays the download cost; subsequent runs reuse the cache).
All notebooks include training, evaluation, and sample recommendations.
Documentation
Full documentation is available at intuit.github.io/scikit-rec.
Development
git clone https://github.com/intuit/scikit-rec.git
cd scikit-rec
python -m venv .venv && source .venv/bin/activate
pip install -e ".[dev]"
pytest tests/
Contributing
Contributions are welcome! See CONTRIBUTING.md for guidance on feature requests, bug reports, and pull request workflow.
Code of Conduct
This project follows a Code of Conduct to ensure a respectful, inclusive community.
License
Project details
Release history Release notifications | RSS feed
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 scikit_rec-0.4.tar.gz.
File metadata
- Download URL: scikit_rec-0.4.tar.gz
- Upload date:
- Size: 2.1 MB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2ecc96433e5c3f7e710d577bc126d06e1f574a297d828ab1d8c47c91c034e034
|
|
| MD5 |
d50806369af1284f7b97f864d250df9f
|
|
| BLAKE2b-256 |
86bb728892fec29bfce3252486a35b1bf2b8592d58ed7d8da035f67207cdeaf7
|
Provenance
The following attestation bundles were made for scikit_rec-0.4.tar.gz:
Publisher:
publish.yml on intuit/scikit-rec
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
scikit_rec-0.4.tar.gz -
Subject digest:
2ecc96433e5c3f7e710d577bc126d06e1f574a297d828ab1d8c47c91c034e034 - Sigstore transparency entry: 1633412383
- Sigstore integration time:
-
Permalink:
intuit/scikit-rec@a8eff135c582027f3d547ddde41251b0b462cf98 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/intuit
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@a8eff135c582027f3d547ddde41251b0b462cf98 -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file scikit_rec-0.4-py3-none-any.whl.
File metadata
- Download URL: scikit_rec-0.4-py3-none-any.whl
- Upload date:
- Size: 556.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
fb2c2715de51f65436b0779b28eef8021e9f1e9c1acfab13de699c2c43e77d2f
|
|
| MD5 |
751b7904a3dbc722e7f1b9410d7c30ce
|
|
| BLAKE2b-256 |
bf8d50cc4ac89dde9e77e31b53e62b6a8ccc0c6cc5146c78228a879267396593
|
Provenance
The following attestation bundles were made for scikit_rec-0.4-py3-none-any.whl:
Publisher:
publish.yml on intuit/scikit-rec
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
scikit_rec-0.4-py3-none-any.whl -
Subject digest:
fb2c2715de51f65436b0779b28eef8021e9f1e9c1acfab13de699c2c43e77d2f - Sigstore transparency entry: 1633412403
- Sigstore integration time:
-
Permalink:
intuit/scikit-rec@a8eff135c582027f3d547ddde41251b0b462cf98 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/intuit
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@a8eff135c582027f3d547ddde41251b0b462cf98 -
Trigger Event:
workflow_dispatch
-
Statement type: