High-performance Nikkei NEEDS tick data processing with Polars
Project description
tse_tick
A Python library for parsing, filtering, and querying Nikkei NEEDS tick data from the Tokyo Stock Exchange.
Who it's for: Researchers working with NEEDS tick data who need to convert thousands of zipped CSVs into queryable Parquet stores, filter by ticker or event windows, and handle format changes across historical eras.
What it solves: NEEDS data is delivered as daily ZIP files (1–27 parts per day) with era-dependent schemas — 2016 used fixed-width records for indices, 2017+ switched to CSV, and individual stocks have 95 columns with complex quote-book nesting. This library detects the format automatically, validates for security, parses everything into clean DataFrames, and writes Hive-partitioned Parquet.
Data access required: This tool does NOT provide NEEDS data itself. You must have an institutional subscription (Nikkei NEEDS) and access to the raw TICST120/TICSS110/TICIT110/TICIS110 ZIP files. If your data is shared via Google Drive, see the rclone download guide for mirroring it to local disk.
Features
- 4 data types — TICST120 (individual stock ticks, 95 cols), TICSS110 (daily stock summary, 82 cols), TICIT110 (index ticks, 10 cols), TICIS110 (daily index summary, 17 cols)
- Multi-era format support — 2016 fixed-width (TICIT010/TICIS010) and 2017-2025 CSV, auto-detected from the ZIP filename
- Polars backend — fast CSV parsing, vectorized cleaning, memory-efficient
- CLI batch ingestion —
tse-tick ingestconverts entire years/months/date ranges to partitioned Parquet - Ticker filtering (
--tickers) — keep only specific stock codes at read time - Event-window extraction (
--filter-csv) — extract ±N minute windows around corporate events with automatic after-hours reaction-anchor shifting - Bilingual columns — English and Japanese column names via
--language en|jp - One-shot reader (
read_ticks) — raw ZIPs → a ticker/time-filtered DataFrame with no Parquet store to build first - Name translation (
translate) — look up thetse_tickequivalent of a yfinance / Polygon / ccxt call (tables intse_tick/data/translations.json; override withTSE_TICK_TRANSLATIONS) - Typed enums (
DataType,Language) — autocomplete-friendly and accepted anywhere the magic strings are - Security guards — ZIP bomb detection (5 GB max decompressed, 100:1 compression ratio cap, max 5 entries), path traversal prevention, query row limits (10M)
Installation
pip install tse-tick # from PyPI: core (polars, pyarrow)
pip install "tse-tick[query]" # + DuckDB-powered Parquet queries
To work from the latest (unreleased) source instead, install in editable mode:
git clone https://github.com/tse-tick/tse_tick.git
cd tse_tick
pip install -e . # core: polars, pyarrow
pip install -e ".[query]" # + DuckDB-powered Parquet queries
pip install -e ".[dev]" # + everything for development (tests, linters, jupyter)
Requires Python ≥3.9. Core dependencies are polars and pyarrow; the query extra adds DuckDB (see pyproject.toml).
Quick Start
Python API — load a single ZIP
import tse_tick
# Load individual stock tick data (auto-detects data type and year)
df = tse_tick.create_df("path/to/HTICST120.20230104.1.zip", language="en")
# Load with Japanese column names
df_jp = tse_tick.create_df("path/to/HTICST120.20230104.1.zip", language="jp")
# Sample first 1000 rows only
df_sample = tse_tick.create_df("path/to/HTICST120.20230104.1.zip", rows=1000)
# Explicit data type and year (skip auto-detection)
df = tse_tick.create_df(
"path/to/file.zip",
auto_detect=False,
data_type="individual_stock",
year=2023,
)
CLI — batch ingest to Parquet
# Ingest a date range
tse-tick ingest \
--data-type individual_stock \
--period 20240201-20240205 \
--input-root /path/to/TSE_DATA \
--output-root /path/to/PARQUET_STORE
# Ingest a full year
tse-tick ingest \
--data-type individual_stock \
--period 2024 \
--input-root /path/to/TSE_DATA \
--output-root /path/to/PARQUET_STORE
# Ticker-filtered ingest (keep only specified stocks)
tse-tick ingest \
--data-type individual_stock \
--period 2024 \
--input-root /path/to/TSE_DATA \
--output-root /path/to/PARQUET_STORE \
--tickers 7203,6758,9984
# Ticker filter from file (one ticker per line)
tse-tick ingest \
--data-type individual_stock \
--period 2024 \
--input-root /path/to/TSE_DATA \
--output-root /path/to/PARQUET_STORE \
--tickers @ticker_list.txt
# Event-window filtered ingest (±120 min around each event)
tse-tick ingest \
--data-type individual_stock \
--period 20250106-20250131 \
--input-root /path/to/TSE_DATA \
--output-root /path/to/PARQUET_STORE \
--filter-csv event_filter_list.csv \
--window 120
CLI — export one ticker to CSV or Parquet (no store)
For a quick slice straight from the raw ZIPs — ideal if you don't write Python. Reads every part of each day, so the result is complete:
tse-tick export \
--data-type individual_stock \
--tickers 7203 \
--period 20240201-20240205 \
--input-root /path/to/TSE_DATA \
--output toyota.csv # .csv or .parquet, chosen by extension
Query the Parquet store
Note: the query functions (
query_ticks,query_sql,get_available_*) require the[query]extra —pip install "tse-tick[query]"(DuckDB). On the core install, use the DuckDB-freeread_parquet_partition(store, "individual_stock", date=..., ticker=...)instead.
import tse_tick
# Query specific ticker and date
df = tse_tick.query_ticks(
"/path/to/PARQUET_STORE",
data_type="individual_stock",
ticker=7203,
date="20240201",
start_time="09:00:00",
end_time="11:30:00",
)
# Get available dates and tickers
dates = tse_tick.get_available_dates("/path/to/PARQUET_STORE")
tickers = tse_tick.get_available_tickers("/path/to/PARQUET_STORE", date="20240201")
Feature extraction
import tse_tick
df = tse_tick.query_ticks("/store", ticker=7203, date="20220201")
# Bid-ask spread
spread = tse_tick.compute_spread(df)
# Order-book depth (10 levels per side)
depth = tse_tick.compute_depth(df, levels=5, side="both")
# Order flow imbalance over rolling window
ofi = tse_tick.compute_flow_imbalance(df, window="5min")
# All features in one pass
features = tse_tick.compute_all_features(df)
Two access patterns
tse_tick gives you a filtered DataFrame two ways:
- Two-stage (scale / repeated work) —
ingestthe raw ZIPs into a Hive-partitioned Parquet store once, thenquery_ticksit repeatedly. Querying the store prunes by date/ticker and is far faster than re-reading raw files (~694× vs a pandas CSV scan; see Performance). - One-shot (quick, targeted exploration) —
read_ticks(...)reads straight from raw ZIPs to a ticker/time-filtered DataFrame with no store to build first. It reads every ZIP part of each day (complete multi-part data) and accepts a date range (date="20240201-20240205"); best for one or a few tickers over a bounded window. Thetse-tick exportCLI wraps it to CSV/Parquet for non-coders.
import tse_tick
# Toyota (7203) over a date range — straight from the raw ZIPs, no store.
# read_ticks reads EVERY part of each day, so the result is complete.
df = tse_tick.read_ticks(
"/path/to/TSE_DATA", # a .zip, a flat folder, or ANY folder above the data (located by type+date)
ticker_filter={"7203"},
date="20240201-20240205", # single day "20240201", a month "202402", a year "2024", or a range
start_time="09:00:00",
end_time="11:30:00",
)
Data Types
| Code | Internal Name | Output Fields | Description |
|---|---|---|---|
| TICST120 | individual_stock |
95 | Tick-level executions, 10-level bid/ask quotes, volume |
| TICSS110 | stock_summary |
82 (83 raw) | Daily OHLC, VWAP, session splits, quote statistics |
| TICIT110 | indices |
10 (23 raw, 15 in 2016) | Index tick updates (Nikkei 225, TOPIX, etc.) |
| TICIS110 | indices_summary |
17 (83 raw) | Daily index summary prices |
Multi-Era Format Support
The format changed only once, after 2016, and only for the index types (fixed-width to CSV); individual stock and stock summary files were CSV throughout. The library detects the era automatically from the ZIP filename (the year) and applies the correct parser.
| Era | Individual Stocks | Stock Summary | Index Ticks | Index Summary |
|---|---|---|---|---|
| 2016 | CSV, 95 cols | CSV, 83 cols | Fixed-width (69 bytes) | Fixed-width (hybrid) |
| 2017-2025 | CSV, 95 cols | CSV, 83 cols | CSV, 23 cols | CSV, 83 cols |
No user action needed — if your ZIP filename contains 2016, the fixed-width parser is used automatically for index data.
Performance
tse_tick is built on Polars (CSV parsing, vectorized cleaning) and DuckDB over Hive-partitioned Parquet (queries). Measured on one day of HTICST120 (4.78 M rows, 95 columns, 2.16 GB raw CSV) on an Intel Core i5-14400F (10-core / 16-thread) with 32 GB RAM, Python 3.11, Polars 1.40, pandas 2.2.
| Comparison | Speedup | Source |
|---|---|---|
| Polars (16T) vs pandas (Python engine) | 55.5× | benchmarks/results_engine_summary.csv |
| Polars (16T) vs pandas (C engine, fair baseline) | 22.8× | benchmarks/results_engine_summary.csv |
| Polars (1 thread) vs pandas (C engine) | 6.2× | benchmarks/results_engine_summary.csv |
| DuckDB + Hive Parquet vs pandas CSV scan (single-ticker hour slice) | 694.1× | benchmarks/results_query.csv |
| Parquet (Snappy) storage size vs raw CSV | 22× smaller (100 MB vs 2.2 GB) | benchmarks/results_format.csv |
The three Polars speedup numbers are deliberately reported together: against the original pandas Python-engine prototype, against a fair C-engine baseline (all-string dtypes, forced column count), and at single-thread parity to isolate the contribution of threading from the engine itself. Polars wins on all three.
tse_tick defaults to Polars because the ingest workload (multi-GB daily CSVs, mostly columnar transformations) hits exactly the case where lazy expression planning and parallel CSV parsing dominate; pandas-on-DataFrame's row-oriented model leaves throughput on the table even with the C engine. For querying, the Parquet store + DuckDB combination converts repeated single-ticker / single-date filters from full file scans into partition pruning, which is the source of the ~700× query speedup.
To reproduce: python benchmarks/run_all.py (see benchmarks/ENVIRONMENT.md).
Expected Input Layout
The CLI expects NEEDS data organized as delivered by Nikkei:
{input_root}/
2016/
201601/
HTICST120.20160104.1.zip
HTICST120.20160104.2.zip
...
201602/
...
2017/
201701/
...
Real NEEDS deliveries are often nested — e.g. 個別株式{year}/TICST120/{yyyymm}/HTICST120.*.zip
(a Japanese-named year folder, then the data-type code, then the month). You don't have to match the
strict layout above: point --input-root (or read_ticks(...) / tse-tick export) at any folder
that contains the data — files are located by type + date, regardless of folder names or depth.
Tip: aim at the common parent (e.g. G:\NEEDS) to cover several years at once.
Parquet Output Layout
Standard ingest produces Hive-partitioned Parquet per ticker per date:
{output_root}/
individual_stock/
date=20230104/
ticker=7203.parquet
ticker=6758.parquet
...
Event-window filtered ingest writes per-date files:
{output_root}/
year=2025/
month=01/
20250106.parquet
20250107.parquet
...
CLI Reference
| Flag | Description |
|---|---|
--data-type (required) |
individual_stock, stock_summary, indices, or indices_summary |
--input-root (required) |
Root directory with NEEDS ZIPs in {year}/{yearmonth}/ layout |
--output-root (required) |
Root directory for Parquet output |
--period |
Date range: YYYY, YYYYMM-YYYYMM, or YYYYMMDD-YYYYMMDD |
--language |
Column name language: en (default) or jp |
--parallel |
Number of parallel workers (default 1, max 8) |
--no-resume |
Disable resume (reprocess dates even if output exists) |
--tickers |
Comma-separated codes or @file.txt with one per line. Keeps only these stocks. |
--filter-csv |
Path to event filter CSV. Enables event-window mode. Overrides --tickers. |
--window |
Window minutes around each event's reaction anchor (default 120). Only with --filter-csv. |
--flat |
Treat input-root as a flat directory (no year/month subdirectories) |
--years / --year |
Legacy flags for specifying year(s) directly |
Event Filter CSV Format
When using --filter-csv, the file must have these columns:
| Column | Description |
|---|---|
ticker |
4-digit stock code (string) |
event_date |
Original event date YYYY-MM-DD |
event_time |
Original event time HH:MM (JST) |
event_type |
Category (earnings, buyback, dividend, etc.) |
session_type |
intraday or after_hours |
reaction_anchor_dt |
Datetime to center the window on YYYY-MM-DD HH:MM (JST) |
zip_date |
TICST120 date YYYYMMDD whose ZIP contains the relevant ticks |
For after-hours events, reaction_anchor_dt shifts to the next trading day's 09:00 open, and zip_date points to that next day's ZIP file. This is critical: centering on the event time (e.g., 15:30) would produce empty windows because the market is closed.
Python API Reference
create_df(folder_path, language="en", rows=None, auto_detect=True, data_type=None, year=None, ticker_filter=None)
Load and clean tick data from a ZIP file or directory of ZIP files.
folder_path— path to a.zipfile or directory of.zipfileslanguage—"en"or"jp"for column namesrows— max rows to returnauto_detect— ifTrue, detect data type and year from path. IfFalse, must providedata_typeandyeardata_type—"individual_stock","stock_summary","indices", or"indices_summary"year— data year (e.g., 2023)ticker_filter— optionalsetof 4-digit stock codes to pre-filter at line level
Returns a Polars DataFrame with English or Japanese column names.
export_to_csv(folder_path, output_path=None, language="en", rows=None)
Load and export to CSV. If output_path is None, generates a filename.
Security
Built-in protections for local data processing:
| Guard | Value |
|---|---|
| ZIP bomb detection (max decompressed) | 5 GB |
| ZIP compression ratio cap | 100:1 |
| Max ZIP entries | 5 |
| Max parallel workers | 8 |
| Query row limit | 10,000,000 |
| Path traversal prevention | Resolved path validation |
| SQL injection prevention | Identifier/date/time format validation |
What's New in 0.9.0
tse_tick 0.9.0 — pip install -U tse-tick. Two real-data defect fixes from a sixth run.
stock_summarynumbers are numeric again. Every measure column (OHLC, VWAP, volumes, amounts, counts) was returned asString, so.mean()/ arithmetic silently producednull; they're nowFloat64like the other three types. Re-ingeststock_summarystores to refresh the dtypes.- Time-filtering
individual_stockkeeps the whole order book. Quote-only book updates (no trade) have a blankExecution Timebut a realUpdate Time; a session window keyed only onExecution Timewas silently dropping ~94% of a liquid day (only trade-coincident snapshots).read_ticksandquery_ticksnow fall back toUpdate Timefor those rows, socompute_depth/compute_spread/compute_flow_imbalancesee the full in-window book (theExecution Timecolumn itself is unchanged in the output). - Docs:
read_ticksnow notes typical one-shot timing — it opens every ZIP part of each day, so useingest_*+query_ticksfor faster repeated/narrow work.
Earlier highlights (0.8.0): capturable NoDataWarning, fixed-width index Execution Time, string get_available_tickers codes, UTF-8 Windows stdout. (0.7.0): all-four-types correctness — summary query_ticks, jp/ingest ticker_filter, the 2016 index era, raw Index Code. (0.6.0): Windows-safe print(df), missing-date warnings + typed-empty reads.
See CHANGELOG.md for the full list.
Notes for library users
- Quiet by default.
create_df,read_ticks, and theingest_*functions emit diagnostics vialogging, notprint, so they never write to stdout (or crash on non-ASCII paths) unless you opt in withlogging.basicConfig(level=logging.INFO). Thetse-tickCLI still prints progress. - Windows-friendly
print. On Windows, importingtse_tickswitches Polars to ASCII table borders and reconfiguresstdout/stderrto UTF-8, so a bareprint(df)no longer raisesUnicodeEncodeErroron a cp1252 console — neither the box-drawing borders nor the content glyphs (datetime[μs],≤in column names,—in exchange values). Opt out of both withTSE_TICK_ASCII_TABLES=0;tse_tick.display(df)prints any DataFrame as UTF-8 on any platform regardless. - Discovery round-trips.
get_available_tickers(...)returns string codes (e.g.["6758", "7203"]) you can pass straight toread_ticks(ticker_filter=...); alphanumeric codes (e.g."130A") are preserved rather than dropped. (read_ticks/query_ticksalso acceptintcodes.) - Flexible discovery. Structured-root
read_ticks/discover_zipsfind ZIPs under the documented{year}/{yearmonth}/layout, a{yearmonth}/folder directly under the root (e.g. a…/TICST120type folder), and — as a fallback — recursively under nested delivery trees such as個別株式{year}/TICST120/{yyyymm}/. - One numbered ZIP is part of a day. NEEDS splits each day across parts by ascending code, so
filtering a lone
HTICST120.<date>.N.zipby ticker can return 0 rows (Toyota 7203 is in a later part) — pass the day's directory or a structured root for complete coverage. - Numeric dtypes. Price/quote columns (
Execution Price,Sell Quote 1 Best, …) areFloat64, and allstock_summarymeasures (OHLC, VWAP, volumes, amounts, counts) areFloat64too — so.mean()and arithmetic work without manual casting. (Stores ingested before the relevant change held these asString— re-ingest to refresh.) - Time filters keep the whole order book. For
individual_stock, quote-only book updates have a blankExecution Timebut a realUpdate Time;read_ticks/query_tickstime windows fall back toUpdate Timefor those rows, so a session filter retains in-window quote updates (not just trade-coincident snapshots) — whatcompute_depth/compute_spread/compute_flow_imbalanceneed. - Index codes are raw codes.
indicesandindices_summaryboth returnIndex Codeas the raw numeric code (e.g."101"), matching what you pass toticker_filterand theticker=partition;ticker_filteralso accepts the display name ("Nikkei 225"). (Stores written before 0.7.0 held decoded names forindices— re-ingest to refresh.) - Empty results keep their schema and warn. A read that matches nothing — a date with no ZIPs (e.g.
a market holiday), an unknown ticker/index code, or an over-tight filter — returns an empty but
fully-typed DataFrame (all columns present), so chained access like
df["Exchange Code"]won't raise.read_ticksalso emits a capturabletse_tick.NoDataWarning(aUserWarning) for every zero-row result across all four types, so "no data" is never silent — trap it withwarnings.catch_warnings()or silence it withwarnings.filterwarnings("ignore", category=tse_tick.NoDataWarning). - Ingestion entry points are the functions
ingest_period,ingest_single_zip,ingest_year_from_root, … —tse_tick.ingestitself is the submodule.
Contributing
Contributions are welcome. Please open an issue or submit a pull request.
- Fork the repository
- Create a feature branch (
git checkout -b feature/your-feature) - Commit your changes (
git commit -m 'Add your feature') - Push to the branch (
git push origin feature/your-feature) - Open a Pull Request
Development setup:
pip install -e ".[dev]"
pytest tests/ -v
Testing
pytest tests/ -v
The suite collects 282 tests. Without a local NEEDS store, 234 pass and 48 skip; with a complete NEEDS store, all 282 pass. Stage-1
(ingestion) and Stage-2 (query, order-book features, and
event-window-from-Parquet) both run with no proprietary data — a session-scoped
pytest fixture builds a tiny Hive-partitioned Parquet store at test time by
feeding synthetic, obviously-fake individual_stock (TICST120) ZIPs through the
real ingest pipeline (tests/synthetic_data.py, tests/conftest.py).
The 48 skips load real NEEDS files from local paths
(test_real_data.py and the real-ZIP cases in test_ingest.py), plus a handful
of fixtures outside the synthetic store's scope. They run automatically once a
local NEEDS store is present.
Citation
If you use this software in your research, please cite it using the CITATION.cff file in the repository. A technical paper describing the library is in preparation.
License
Authors
- Kazumi Li — Schema definitions, package architecture, current maintainer
- Masataka Hayashi — Initial pandas-based prototype
- Peter Romero — Original concept and initial project design
Developed at Keio University, Nakatsuma Seminar.
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 tse_tick-0.9.0.tar.gz.
File metadata
- Download URL: tse_tick-0.9.0.tar.gz
- Upload date:
- Size: 98.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
359d7902e77f4fefd192bf8b7297f97f4be70d7fb12f40b93a0375f84dcf58ea
|
|
| MD5 |
322395fb5969a813a2833c5cbdb80046
|
|
| BLAKE2b-256 |
68c3ae0a51491a0b07c3d42acf07266d71cc9d3cf7ab0b18af8447ad489e89c5
|
Provenance
The following attestation bundles were made for tse_tick-0.9.0.tar.gz:
Publisher:
publish.yml on tse-tick/tse_tick
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
tse_tick-0.9.0.tar.gz -
Subject digest:
359d7902e77f4fefd192bf8b7297f97f4be70d7fb12f40b93a0375f84dcf58ea - Sigstore transparency entry: 1859840941
- Sigstore integration time:
-
Permalink:
tse-tick/tse_tick@75f58dce74b0de863ff8c2af2450267e82600372 -
Branch / Tag:
refs/tags/v0.9.0 - Owner: https://github.com/tse-tick
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@75f58dce74b0de863ff8c2af2450267e82600372 -
Trigger Event:
release
-
Statement type:
File details
Details for the file tse_tick-0.9.0-py3-none-any.whl.
File metadata
- Download URL: tse_tick-0.9.0-py3-none-any.whl
- Upload date:
- Size: 64.6 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 |
5a7819e277d09864807d2b83d8dd31fb8829f29e26ad9a4b38a58c88aae77588
|
|
| MD5 |
5038b01856988b1cf5835ca9b97777f5
|
|
| BLAKE2b-256 |
3f4c93c2781b239518cc75444f82b60f6708645ac36374b8566e3ad9c940c894
|
Provenance
The following attestation bundles were made for tse_tick-0.9.0-py3-none-any.whl:
Publisher:
publish.yml on tse-tick/tse_tick
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
tse_tick-0.9.0-py3-none-any.whl -
Subject digest:
5a7819e277d09864807d2b83d8dd31fb8829f29e26ad9a4b38a58c88aae77588 - Sigstore transparency entry: 1859840988
- Sigstore integration time:
-
Permalink:
tse-tick/tse_tick@75f58dce74b0de863ff8c2af2450267e82600372 -
Branch / Tag:
refs/tags/v0.9.0 - Owner: https://github.com/tse-tick
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@75f58dce74b0de863ff8c2af2450267e82600372 -
Trigger Event:
release
-
Statement type: