| language |
|
|||||
|---|---|---|---|---|---|---|
| license | mit | |||||
| tags |
|
|||||
| base_model | Qwen/Qwen2.5-0.5B | |||||
| datasets |
|
|||||
| library_name | transformers | |||||
| pipeline_tag | text-generation |
Meridian.AI is a finance-specialized language model that trains itself continuously, every hour, entirely on free GitHub Actions infrastructure. It continuously fine-tunes a Qwen2.5-0.5B backbone on 25+ finance and math datasets using Elastic Weight Consolidation (EWC) to prevent catastrophic forgetting across training sessions.
Model checkpoints: huggingface.co/meridianal/FinAI
What this is, clearly: Qwen2.5-0.5B (a 494M-parameter causal LM from Alibaba) continuously fine-tuned on finance data via hourly GitHub Actions CI. The
meridian/module in this repo contains a custom Sparse MoE research architecture used for experiments and smoke tests — it is not what is deployed in the HuggingFace checkpoint.
- Why Meridian.AI
- Key Design Decisions
- Model Specifications
- Quick Start
- Inference
- Local Training
- Environment Variables Reference
- CI/CD Training Pipeline
- Dataset Curriculum
- Repository Structure
- Troubleshooting
- Changelog
- Disclaimer
Standard LLMs have a static knowledge cutoff. For finance — where earnings reports, Fed decisions, and market conditions change daily — this is a critical limitation.
Meridian.AI solves this with an automated continuous training pipeline: every hour, GitHub Actions pulls the latest checkpoint from HuggingFace, trains on fresh financial data, and pushes the updated checkpoint back. No GPUs. No cloud bills. No manual intervention.
Key constraints this design respects:
- 16 GB RAM ceiling of free GitHub Actions ubuntu-latest runners
- 2000 CI minutes/month free tier (public repos: unlimited)
- No persistent storage between runs — all state is round-tripped through HuggingFace Hub
Rather than training a model from scratch, Meridian.AI continuously fine-tunes Qwen/Qwen2.5-0.5B — a production-quality base model with strong pre-training on code, math, and multilingual text. This gives the model strong priors out of the box, allowing hourly fine-tuning to specialize it without training from zero.
After each hourly training run, the model computes the diagonal Fisher Information Matrix — a measure of which parameters were most important for tasks learned so far. The next run adds a regularization penalty for changing those parameters. This prevents the model from "forgetting" financial knowledge from earlier training sessions while absorbing new data.
Every component is designed around the 16 GB RAM constraint:
- AdaFactor optimizer: eliminates the 2×-parameter-size optimizer state of Adam
- Gradient checkpointing: trades compute for activation memory
- Soft RAM throttle: dynamically truncates sequence length when memory pressure rises
- Hard RAM guard: emergency checkpoint + clean exit before OOM
- Fisher threshold pruning: only stores EWC state for parameters with Fisher value above threshold
The meridian/ module contains a from-scratch Sparse Mixture-of-Experts Transformer (MeridianForCausalLM) with:
- SMoE (8 experts, top-2 per token) on alternating layers
- Grouped Query Attention (12 Q heads, 4 KV heads)
- RoPE position embeddings (theta=500,000)
- SwiGLU feed-forward blocks
- Financial Numeracy Encoding
This module is used for smoke tests and architecture experiments. It is not the model in the HuggingFace checkpoint — the deployed model is Qwen2.5-0.5B fine-tuned via AutoModelForCausalLM.
| Specification | Value |
|---|---|
| Base Model | Qwen2.5-0.5B |
| Architecture | Qwen2ForCausalLM |
| Parameters | ~494M |
| Layers | 24 |
| Hidden Size | 896 |
| Attention Heads | 14 Q / 2 KV (GQA) |
| Vocabulary | 151,643 tokens |
| Context Window | 32,768 tokens (Qwen2.5 default) |
| Training dtype | bfloat16 |
| Continual Learning | Elastic Weight Consolidation (EWC) |
| Specification | Value |
|---|---|
| Architecture | Sparse MoE Transformer |
| Layers | 14 (alternating Dense ↔ MoE) |
| Attention | GQA: 12 Q heads, 4 KV heads |
| Position Encoding | RoPE (theta=500,000) |
| Feed-Forward | SwiGLU |
| Normalization | RMSNorm |
| MoE Experts | 8 per layer, top-2 active per token |
| Vocabulary | 151,665 tokens (Qwen2.5 tokenizer) |
| Context Window | 2,048 tokens (configurable) |
| Continual Learning | Elastic Weight Consolidation |
- Python 3.10+
- ~2 GB disk (for model weights)
git clone https://github.com/MeridianAlgo/FinAI.git
cd FinAI
pip install -r requirements.txtRuns a tiny in-memory model to confirm the custom architecture works without downloading anything:
SMOKE_TEST=1 FAST_MODE=1 python train.pyExpected output: [OK] Smoke test passed!
import torch
from transformers import AutoModelForCausalLM, AutoTokenizer
repo_id = "meridianal/FinAI"
subfolder = "checkpoint"
tokenizer = AutoTokenizer.from_pretrained(repo_id, subfolder=subfolder)
model = AutoModelForCausalLM.from_pretrained(
repo_id,
subfolder=subfolder,
torch_dtype=torch.float32,
low_cpu_mem_usage=True,
)
model.eval()
prompt = """### Instruction:
Explain the difference between a bond's yield to maturity and its coupon rate.
### Response:
"""
inputs = tokenizer(prompt, return_tensors="pt")
with torch.no_grad():
output = model.generate(
**inputs,
max_new_tokens=200,
do_sample=True,
temperature=0.8,
top_p=0.92,
repetition_penalty=1.3,
no_repeat_ngram_size=3,
pad_token_id=tokenizer.pad_token_id,
eos_token_id=tokenizer.eos_token_id,
)
print(tokenizer.decode(output[0], skip_special_tokens=True))Note:
trust_remote_code=Trueis not required for the deployed checkpoint — it is a standard Qwen2 model and loads withAutoModelForCausalLMdirectly.
Download the checkpoint and run locally:
python scripts/download_and_save_hf.pyThen:
from transformers import AutoModelForCausalLM, AutoTokenizer
tokenizer = AutoTokenizer.from_pretrained("./checkpoint")
model = AutoModelForCausalLM.from_pretrained("./checkpoint")See examples/01_inference.py for a complete, annotated script.
export HF_TOKEN=your_huggingface_token
python train.pyThe script will:
- Pull the latest checkpoint from HuggingFace (if
HF_TOKENis set) - Load the Qwen2.5-0.5B base model (or resume from checkpoint)
- Stream finance datasets and train for
MAX_STEPSsteps - Save checkpoint locally (and upload if token is present)
Runs with minimal settings (no dataset streaming, 5 steps, tiny sequences):
FAST_MODE=1 python train.pyMAX_STEPS=300 BATCH_SIZE=1 python train.pyCompare your checkpoint against base Qwen2.5-0.5B:
python scripts/evaluate_model.pyAll variables are optional. CI defaults are shown in .github/workflows/train.yml.
| Variable | CI Default (v6) | Description |
|---|---|---|
MAX_STEPS |
150 |
Gradient update steps per run |
TOTAL_STEPS |
100000 |
Cumulative steps across all runs (for LR schedule) |
BATCH_SIZE |
1 |
Samples per micro-step |
GRAD_ACCUM |
4 |
Micro-steps before each optimizer update |
LEARNING_RATE |
5e-5 |
Peak learning rate |
BLOCK_SIZE |
512 |
Token sequence length |
DTYPE |
bfloat16 |
Model dtype (bfloat16 or float32) |
OPTIMIZER |
adafactor |
Optimizer (adafactor or adamw) |
| Variable | CI Default | Description |
|---|---|---|
HARD_RAM_GUARD |
1 |
Enable emergency save + stop at RAM ceiling |
MAX_RAM_GB |
14.5 |
Hard RAM limit in GB |
SOFT_RAM_GB |
12.5 |
Soft limit — begins sequence truncation |
SOFT_RAM_PCT |
80 |
Soft limit as % of total RAM |
MIN_THROTTLE_SEQ_LEN |
64 |
Minimum sequence length during throttle |
GRADIENT_CHECKPOINTING |
1 |
Trade compute for activation memory |
SKIP_OPTIMIZER_SAVE |
1 |
Omit optimizer state from checkpoint |
| Variable | CI Default | Description |
|---|---|---|
MAX_BYTES |
26214400 (25 MB) |
Max training data per run |
USE_LIGHT_DATASETS |
0 |
Restrict to small/fast datasets only |
| Variable | CI Default (v6) | Description |
|---|---|---|
USE_EWC |
1 |
Enable Elastic Weight Consolidation |
EWC_LAMBDA |
75.0 |
EWC regularization strength (reduced from 500) |
EWC_SAMPLES |
20 |
Batches used to estimate Fisher matrix (increased from 5) |
SKIP_FISHER |
0 |
Skip Fisher computation (disables EWC next run) |
FREE_OPTIMIZER_BEFORE_FISHER |
1 |
Free optimizer RAM before Fisher computation |
FISHER_SEQ_LEN |
64 |
Sequence length used during Fisher estimation |
FISHER_THRESHOLD |
5e-4 |
Drop Fisher entries below this value (raised from 1e-4 to reduce EWC file size) |
| Variable | Default | Description |
|---|---|---|
CHECKPOINT_PATH |
./checkpoint |
Local checkpoint directory |
TOKENIZER_ID |
Qwen/Qwen2.5-0.5B |
HuggingFace tokenizer ID |
FAST_MODE |
0 |
Minimal config for quick local debugging |
SMOKE_TEST |
0 |
Run tiny in-memory architecture test |
GC_EVERY_STEPS |
5 |
Python GC frequency (steps) |
DEBUG_STEPS |
0 |
Print verbose per-step debug info |
COMET_API_KEY |
(unset) | Comet ML experiment tracking key |
Every hour (GitHub Actions cron: '0 * * * *')
│
├── Pull checkpoint from HuggingFace Hub
│ meridianal/FinAI → ./checkpoint/
│
├── Train (timeout: 90 minutes)
│ • Load Qwen2.5-0.5B (or resume checkpoint)
│ • Stream finance datasets (25+ sources, weighted curriculum mix)
│ • 150 AdaFactor steps with gradient checkpointing
│ • EWC regularization (lambda=75, 20 Fisher samples)
│ • Auto-throttle sequence length if RAM > 12.5 GB
│ • Emergency save + exit if RAM > 14.5 GB
│
├── Upload checkpoint to HuggingFace Hub
│ ./checkpoint/ → meridianal/FinAI/checkpoint/
│
└── Sync dataset state to git
dataset_state.json → main branch
If training encounters >50 [ERROR] lines or any fatal pattern (OOM, NaN explosion), the CI workflow automatically opens a GitHub Issue with the error details and a diagnostic checklist.
From the GitHub Actions tab, click Meridian.AI Train → Run workflow. You can override MAX_STEPS at dispatch time.
To wipe the HuggingFace checkpoint and restart training from a fresh Qwen2.5-0.5B:
Run workflow with force_seed: true. This runs scripts/seed_hf_repo.py before training.
Training data is a weighted mix of finance-focused HuggingFace datasets, streamed in real-time (no full downloads):
| Dataset | Weight | Focus |
|---|---|---|
gbharti/finance-alpaca |
26% | Financial Q&A instructions |
sujet-ai/Sujet-Finance-Instruct-177k |
18% | High-quality finance instruction pairs |
nvidia/OpenMathInstruct-2 |
15% | Math reasoning (quantitative finance) |
HuggingFaceFW/fineweb-edu |
12% | General knowledge foundation |
mhenrichsen/alpaca_data_cleaned |
5% | General instruction format |
FinGPT/fingpt-sentiment-train |
4% | Financial news sentiment |
FinanceMTEB/financial_phrasebank |
1% | Sentiment classification |
FinanceMTEB/FinQA |
1% | Financial QA pairs |
FinanceMTEB/TATQA |
1% | Table-and-text QA |
FinanceMTEB/FOMC |
0.8% | FOMC meeting transcripts |
| Various FinanceMTEB | ~16% | Sentiment, ESG, fraud, FLS, events, and more |
All text is formatted into the ### Instruction: / ### Response: template before tokenization.
See docs/training_pipeline.md for full dataset details.
FinAI/
├── meridian/ # Python package
│ ├── model/
│ │ ├── configuration.py # MeridianConfig — research arch config
│ │ └── modeling.py # MeridianForCausalLM — custom SMoE (research only)
│ ├── data/
│ │ └── pipeline.py # Streaming dataset curriculum + DataLoader
│ └── training/
│ ├── trainer.py # MeridianTrainer (AdaFactor, EWC, RAM guards)
│ └── ewc.py # Elastic Weight Consolidation
│
├── scripts/
│ ├── seed_hf_repo.py # Nuke & reseed HuggingFace repo
│ ├── migrate_legacy_and_seed.py # Copy checkpoint → legacy/ and seed fresh model
│ ├── cleanup_hf_checkpoint.py # Remove stale pytorch_model.bin from HF
│ ├── evaluate_model.py # Evaluation: perplexity + generation quality
│ ├── diagnose_and_test.py # Full diagnostic report (download + test)
│ ├── download_and_save_hf.py # Download checkpoint to local directory
│ ├── hf_download_and_test.py # Download + quick generation test
│ └── count_params.py # Parameter counting utility
│
├── examples/
│ ├── 01_inference.py # HuggingFace inference example
│ ├── 02_dataset_pipeline.py # Dataset streaming walkthrough
│ └── 03_model_config.py # Custom arch instantiation
│
├── docs/
│ ├── architecture.md # Detailed architecture spec (custom SMoE)
│ ├── training_pipeline.md # Pipeline, env vars, memory management
│ └── setup_and_usage.md # Setup guide and inference examples
│
├── tests/
│ ├── test_model.py # Architecture unit tests
│ └── test_training.py # Trainer and EWC tests
│
├── .github/
│ └── workflows/
│ ├── train.yml # Hourly training CI
│ ├── lint.yml # Ruff + Black linting
│ └── dependency-cache.yml
│
├── train.py # Main training entry point
├── CHANGELOG.md # Version history and training audit
├── requirements.txt # Python dependencies
└── pyproject.toml # Ruff + Black + mypy config
Reduce memory usage:
BATCH_SIZE=1 GRAD_ACCUM=4 BLOCK_SIZE=256 SOFT_RAM_GB=10.0 python train.pyIf you see [WARN] Checkpoint architecture mismatch (old model), the saved config.json has a model_type that doesn't match qwen2. The checkpoint will be discarded and training restarts from the base model. This is expected when switching base architectures.
[WARN] EWC: Dropped N params due to shape/name mismatch appears when the model architecture changed between runs. EWC state for mismatched layers is safely dropped. Training continues normally.
Usually caused by extreme learning rates or corrupted data batches. The trainer automatically skips batches with NaN loss or NaN gradients. If persistent:
LEARNING_RATE=1e-5 USE_EWC=0 python train.pyRaise the Fisher threshold to prune more aggressively:
FISHER_THRESHOLD=1e-3 python train.pyExpected — these are CPU-only runners. With BATCH_SIZE=1 BLOCK_SIZE=512 MAX_STEPS=200, expect ~50–80 minutes per run. This fits the 90-minute CI timeout.
- Run the test suite before submitting:
pytest tests/ -v - Format with black:
black . - Lint with ruff:
ruff check . --fix - Keep all code pure Python — no unnecessary system dependencies
- Submit PRs against
mainwith clear commit messages (conventional commits format)
See CHANGELOG.md for the full version history, training audit, and issue tracker.
Meridian.AI is an experimental research project on continual learning for financial NLP. All model outputs are strictly for academic and research purposes. Nothing generated by this model constitutes financial advice. Do not use outputs to make real financial decisions or execute trades.