
citracer
๐ Description
A paper cites 50+ references, but which ones actually discuss the concept you care about? And which papers those cite? And the ones after that?
citracer answers this recursively. Give it a PDF and a keyword: it finds every sentence where the keyword appears, identifies the references cited nearby, downloads those papers, and repeats the process N levels deep. The output is an interactive citation graph you can explore in your browser. A 5-depth trace starting from a single paper typically surfaces 50-150 relevant papers in minutes.
With --semantic, matching goes beyond literal keywords: a sentence-transformer embedding model catches passages that express the same concept with different vocabulary (e.g. "univariate processing" matches a trace for "channel-independent").
With --reverse, citracer walks the other direction: "which papers cite this paper while mentioning a given concept?" This is useful for tracing how an idea spread forward through the literature.
Supported sources. citracer resolves cited papers through arXiv, Semantic Scholar, OpenReview, Sci-Hub, and Semantic Scholar's open-access PDF links (which cover PMC, publisher OA pages, and more). Preprint servers bioRxiv, medRxiv, ChemRxiv, SSRN, PsyArXiv, AgriXiv, and engrXiv are also supported. Papers that still can't be downloaded appear as
unavailablenodes, but can be enriched with metadata via OpenAlex using the--enrichflag.

Full documentation available at marcpinet.fr/citracer with detailed guides for each feature, export formats, and the pipeline internals.
โ๏ธ Installation
Requirements: Python 3.10+ and Docker.
From PyPI (recommended)
# Optional: enable semantic matching (adds ~500MB for sentence-transformers + PyTorch)
After pip install citracer, the citracer command is available globally on your PATH. You can then run it from anywhere in your terminal:
From source (for development)
GROBID must be reachable on http://localhost:8070. Verify with curl http://localhost:8070/api/isalive.
A Semantic Scholar API key is optional but recommended. Without one the public endpoint is throttled to ~3.5s between calls. With a key, the throttle drops to ~1.1s. Get a free key at semanticscholar.org/product/api.
--s2-api-key <key>as a CLI flagS2_API_KEYenvironment variable in your shell- A persistent user config at
~/.citracer/config.json, set once via:
Other config commands: citracer config show, citracer config get-s2-key (masked), citracer config clear-s2-key, citracer config path. The file is created with mode 600 on POSIX so other local users can't read it.
A
.envfile at the project root (copy.env.exampleand fill it in):S2_API_KEY=your_key_hereThe
.envfile is git-ignored.
If none of these are set, the unauthenticated public endpoint is used as fallback (much slower, frequent 429 backoffs).
Without an API key, deep traces (depth 4+) can take 10-20 minutes due to rate limiting. With a key, the same trace typically completes 3-5x faster.
Or pass it via `--email` or the `OPENALEX_EMAIL` environment variable.
## ๐ Usage
After `pip install citracer` the `citracer` command is on your `PATH`. The examples below use it directly. If you cloned the repo instead, use `python -m citracer` in place of `citracer`.
```bash
# From a local PDF
citracer --pdf test_data/crossad.pdf --keyword "channel-independent" --depth 5
# From an arXiv id (auto-downloads the root PDF)
citracer --arxiv 2211.14730 --keyword "self-attention"
# From a DOI or URL (arXiv, OpenReview, bioRxiv, medRxiv, SSRN)
citracer --doi 10.48550/arxiv.2211.14730 --keyword "patching"
citracer --doi 10.1016/j.isci.2018.09.017 --keyword "attention"
citracer --url https://openreview.net/forum?id=cGDAkQo1C0p --keyword "instance normalization"
citracer --url https://www.biorxiv.org/content/10.1101/2024.01.01.123456v1 --keyword "CRISPR"
# Multi-keyword tracing (union by default, --match-mode all for intersection)
citracer --pdf paper.pdf --keyword "channel-independent" --keyword "patching"
# Reverse trace: find papers that cite the source while mentioning the keyword
# in their citation context. No PDF downloads, pure S2 metadata. Limit is optional.
citracer --arxiv 2211.14730 --keyword "channel-independent" --reverse --reverse-limit 500
# Enrich unavailable nodes with metadata (abstract, citation count) via OpenAlex
citracer --pdf paper.pdf --keyword "attention" --enrich --email [email protected]
# Supply a local PDF for a node that citracer couldn't download
# Supply a local PDF or a URL for a node that citracer couldn't download
citracer --pdf paper.pdf --keyword "attention" --supply-pdf "doi:10.1234/foo=~/papers/foo.pdf"
citracer --pdf paper.pdf --keyword "attention" --supply-pdf "title:abc123=https://example.com/paper.pdf"
# Export the graph for downstream analysis
citracer --pdf paper.pdf --keyword "..." --export out/graph.json --export out/graph.graphml
# Diff against a previous trace to highlight new papers (orange nodes)
citracer --pdf paper.pdf --keyword "attention" --diff output/old_graph.json
citracer --pdf paper.pdf --keyword "attention" --diff output/old_graph.json --since 2025
citracer --pdf paper.pdf --keyword "attention" --since 2025-06
# Re-run a trace without re-resolving papers already cached locally
citracer --pdf paper.pdf --keyword "attention" --depth 7 --no-refetch
# Semantic matching: catch conceptual matches the regex misses
# (requires: pip install citracer[semantic])
citracer --pdf paper.pdf --keyword "channel-independent" --semantic
citracer --pdf paper.pdf --keyword "attention" --semantic --semantic-threshold 0.55
Source (exactly one required)
Flag | Description |
|---|---|
| Path to a local source PDF |
| DOI of the source paper (e.g. |
| arXiv id of the source paper (e.g. |
| URL of the source paper (arxiv.org, doi.org, openreview.net, biorxiv.org, medrxiv.org, ssrn.com) |
Trace options
Flag | Default | Description |
|---|---|---|
| required | Term (or concept) to trace through citations. By default, matches morphological variants via regex (e.g. "independent" also matches "independence", "independently"). With |
|
| In multi-keyword mode, |
|
| Maximum recursion depth (default |
| sentence-based | If set, fall back to a ยฑN character window for ref association instead of sentence-based |
| off | Ask GROBID to consolidate each bibliographic reference against CrossRef (more accurate titles/DOIs but ~2-5s extra per PDF) |
|
| Number of concurrent GROBID parse requests per BFS level |
|
| GROBID service URL |
| none | Semantic Scholar API key (see Installation for priority order) |
| off | Reverse trace: instead of walking down the source paper's bibliography, walk UP to papers that cite it. Filters citations by matching the keyword against Semantic Scholar citation contexts (the 1-2 sentences around each citation), so no PDFs are downloaded. Default |
|
| Max number of citing papers to fetch per level in reverse mode. Protects against runaway expansion on papers with thousands of citations |
| off | Enable metadata enrichment via OpenAlex for nodes missing abstract, citation count, or year. Anonymous mode (1 req/s); combine with |
| none | Email for OpenAlex polite pool (10 req/s). Implies |
| none | Supply a PDF for a specific node, as a local path or URL. Format: |
| none | Compare against a previous citracer JSON export and highlight new nodes (papers not in the baseline) in orange. Useful for monitoring how a citation graph evolves over time |
| none | Highlight nodes published on or after this date ( |
| off | Skip network resolution for papers already resolved in a previous run (metadata + PDF cached locally). Dramatically speeds up re-runs and avoids API rate limits |
| off | Enable semantic matching: after the regex pass, scan remaining sentences with a sentence-transformer embedding model to catch conceptual matches the regex missed (e.g. "univariate processing" for the keyword "channel-independent"). Requires |
|
| Sentence-transformer model name for |
|
| Cosine similarity threshold for semantic matching (0.0-1.0). Lower = more recall, higher = more precision. Implies |
Output
Flag | Default | Description |
|---|---|---|
|
| Output HTML file |
| none | Export the graph to a file. Format is derived from the extension: |
| off | Show passages directly in node tooltips |
|
| Local cache for PDFs and metadata (SQLite) |
| off | Do not open the result in a browser |
| off | Verbose logging |
๐จ Output
Nodes are colored by status:
Color | Status | Meaning |
|---|---|---|
blue |
| The source PDF |
green |
| PDF retrieved and the keyword (or concept, with |
gray |
| PDF retrieved and parsed, but the keyword was not found (neither by regex nor semantic matching) |
red |
| PDF could not be retrieved |
orange |
| Paper not present in the |
Edges come in two flavors:
Style | Type | Meaning |
|---|---|---|
solid dark | keyword-associated | Paper A cites paper B in the same sentence (or the next) as a keyword match (regex or semantic) |
dashed blue | bibliographic link | Paper A's bibliography also references paper B, independently of any keyword match. Hidden by default, toggle via the legend |
Interactive controls
A control panel in the top-left corner of the graph lets you tune the view on the fly:
Control | Options | Effect |
|---|---|---|
layout | Sugiyama (by year) (default) | Switches the layout algorithm. Sugiyama-by-year places the oldest papers at the top, making it easy to spot which paper first introduced the concept |
node size | in-graph citations (default) |
|
spread | slider (0.3ร to 3.0ร) | Rescales all node positions from the graph's centroid, stretching or compressing the layout without deforming it. Works with any layout mode |
curved edges | checkbox (on by default) | Toggle between curved (cubicBezier/curvedCW) and straight edge rendering |
Export PNG | button + scale selector (2x/3x/4x) | Export the current view as a high-resolution PNG. At 3x on a 1080p display, the output is 5760x3240 |
Export SVG | button | Export as a vector SVG file (lossless zoom, ideal for LaTeX figures) |
nodes (legend) | click rows to toggle | Hide/show nodes by status. When |
edges (legend) | click rows to toggle | Hide/show edges by type (keyword-associated vs. bibliographic link) |
Other interactive features:
- Hover any node โ side panel updates live with title, authors, year, citation count, status, centrality metrics (PageRank, betweenness, in/out degree), a PIVOT badge for pivot papers, keyword hits (regex matches are highlighted; semantic matches show a purple SEM badge with the note "conceptual match") and a collapsible abstract section when available
- Search box in the control panel โ fuzzy match by title or author, click a result to focus-and-pin the matching node
- Click a node โ pins the panel; a blue border is drawn around the node to show the pinned state. The pin survives clicks on the empty canvas, hover on other nodes, and pan/zoom. It's only released by clicking the same node again, pressing the ร close button on the info panel, or picking Unpin from the right-click menu
- Right-click any node โ context menu with Hide, Pin/Unpin, Open link, and Supply PDF (on unavailable nodes only, prompts for a local path or URL and generates the
--supply-pdfcommand to copy for the next run) - Drag any node anywhere. After initial placement the layout is released, so nothing snaps back
- Undo / Redo with
Ctrl+Z/Ctrl+Y(orCtrl+Shift+Z). Reverts node positions, filters, layout, and all control settings. History of up to 50 snapshots - Resizable panel: drag the right edge of the control panel to widen or narrow it
show N morein a panel with many hits โ expands the full list- LaTeX math in passages is rendered with KaTeX (
$...$,$$...$$,\(...\),\[...\]) - Automatic state persistence. Node positions, filters, pin state, dropdowns, spread slider and manually hidden nodes are all saved to
localStoragekeyed on a hash of the node-id set. A browser refresh restores the exact view you had. A new trace with a different paper set gets a fresh slate. The reset link at the bottom of the panel clears everything and reloads
Bibliometric analytics
Every trace automatically computes quantitative metrics on the citation graph:
Metric | Scope | Description |
|---|---|---|
PageRank | per-node | Importance of a paper relative to the citation structure |
Betweenness centrality | per-node | Identifies "bridge" papers that connect different clusters |
In/out degree | per-node | Number of incoming/outgoing edges in the graph |
Pivot detection | per-node | Flags the earliest keyword-matched paper (regex or semantic) in each connected component, plus high-betweenness papers with the keyword |
Graph density | global | Ratio of actual edges to maximum possible edges |
Avg degree | global | Mean number of connections per node |
Connected components | global | Number of weakly connected subgraphs |
Keyword density timeline | global | Per-year breakdown: total papers, papers with keyword, usage density |
The analytics collapsible section in the control panel shows global metrics, a clickable list of pivot papers (clicking focuses the node), and a keyword density timeline table with mini bar charts. Per-node metrics appear in the info panel when hovering or clicking a node.
All analytics are included in the JSON export ("analytics" key), the GraphML export (betweenness, pagerank, is_pivot as node attributes), and the reproducibility manifest.
Reproducibility
Every trace generates a manifest.json alongside the graph output, encoding everything needed to reproduce the exact same graph:
- citracer version, timestamp, full CLI command
- Source paper: type (pdf/doi/arxiv/url), raw input value, resolved title/DOI/arXiv ID
- Parameters: keywords, match mode, depth, context window, consolidate, reverse, enrich, GROBID URL
- Environment: Python version, platform, GROBID availability, API key/email status
- Results: node/edge counts, status breakdown, analytics summary (global metrics, timeline, pivot papers)
This allows anyone receiving a citracer graph to re-run the trace with identical settings. The manifest is also embedded in JSON exports under the "metadata" key.
๐ How it works
PDF parsing. GROBID processes the PDF and returns TEI XML. citracer walks the
<body>to reconstruct the plain text while recording the character offset of every inline<ref type="bibr">citation. The bibliography is extracted from<listBibl>. Figure-diagram paragraphs (detected by their density of mathematical Unicode characters) are skipped to avoid polluting the keyword matcher. Paragraphs that GROBID splits mid-sentence around narrative citations (a common pattern around"Since Smith et al. (2020) and Jones et al. (2021) have shown...") are glued back together with a length-preserving regex so sentence-based matching still sees the refs and the keyword together.Inline ref recovery. GROBID occasionally misses narrative citations like
DLinear Zeng et al. (2023), especially when the author name isn't preceded by a parenthesis. A supplementary pass scans the text for canonical author-year patterns (Surname et al. (Year),Surname & Other (Year),Surname (Year)) and adds them as inline refs whenever the(surname, year)signature matches a unique bibliography entry. In typical ML papers this recovers dozens of refs per document.Keyword matching (regex + optional semantic). The keyword is first compiled to a flexible regex that handles morphological variants (e.g.
channel-independentmatcheschannel-independence,channel independently,channelindependence). The body is segmented into sentences with pysbd, and each occurrence of the keyword is associated with the references cited in the same sentence or the immediately following one. When--semanticis enabled, a second pass scans every sentence the regex didn't match using a sentence-transformer embedding model: sentences whose embedding is close enough to the keyword (cosine similarity above the threshold) are added as additional hits. This catches conceptual matches where the idea is expressed with entirely different vocabulary (for example, tracing "channel-independent" also surfaces passages about "decoupled cross-channel correlations" or "per-variate processing"). Regex hits and semantic hits are unioned, so--semanticonly adds recall without losing any existing matches.Reference resolution. Each cited paper is resolved through the following cascade:
- If GROBID extracted a DOI or arXiv ID, use it directly.
- Otherwise, search arXiv by title (phrase first, then keyword fallback, with rapidfuzz validation). Search results are validated by fuzzy title match (threshold 85) and year cross-check (ยฑ3 years from the bibliography entry's year when known) to prevent false matches on similarly-titled papers from different eras.
- If arXiv has nothing, query Semantic Scholar with the same title + year validation, plus 429-aware backoff (also retrieves citation count and open-access PDF URL).
- As a last resort, search OpenReview (covers ICLR/TMLR papers not on arXiv).
- If
--enrichis set, query OpenAlex for missing metadata (abstract, citation count, OA URL).
PDF download cascade (in order): user-supplied PDF (
--supply-pdf) > arXiv > OpenReview > Sci-Hub (by DOI, tries multiple mirrors) > S2 open-access URL (covers PMC, publisher OA, bioRxiv, medRxiv, etc.) > preprint-specific download (bioRxiv, medRxiv, ChemRxiv, SSRN, PsyArXiv, AgriXiv, engrXiv).All resolved PDFs and metadata are cached in
./cache/.Recursion. The tracer is a BFS that processes papers in queue order. Each level's PDFs are parsed in parallel via a thread pool (
--grobid-workers, default 4), and the reference resolves inside a single paper are also parallelized. Deduplication uses a canonical ID (DOI > arXiv > OpenReview > title hash). When the same PDF is reached via a second path, the new edge is added without re-parsing. Years from bibliography entries can backfill a node's year when older (e.g. a preprint v1 2022 takes precedence over a publication year 2023), but only within a ยฑ2 year window of the first year we ever saw for that node. This prevents cascading from parser mistakes.Cross-graph bibliographic links. After the recursive trace is complete, a post-processing pass scans every parsed paper's bibliography against every other node in the graph and adds dashed "bibliographic link" edges for pairs that cite each other but not in the keyword's neighborhood. Matching is exact on DOI/arXiv IDs and fuzzy (rapidfuzz, threshold 88) on titles. No external API calls are needed: everything runs on the already-in-memory graph, so the cost is negligible.
Bibliometric analytics. After the trace completes, citracer computes per-node centrality metrics (PageRank, betweenness) and graph-wide statistics (density, connected components, keyword density timeline) using networkx. Pivot papers (the earliest keyword-matched paper in each connected component, plus high-betweenness nodes with the keyword) are automatically flagged. A reproducibility manifest (
manifest.json) is written alongside the graph, encoding the full trace parameters, environment, and results.Rendering. The graph is serialized to an interactive HTML page using pyvis, with a custom overlay providing the layout/size/spread controls, the legend filters, the side info panel, keyword highlighting, and KaTeX math.
Reverse trace mode (--reverse)
The forward algorithm walks DOWN from a root paper into its bibliography. --reverse walks UP: "who cites this paper, and which of them mention the keyword in their citation context?".
The key trick is that Semantic Scholar's /paper/{id}/citations endpoint returns a contexts field for each citing paper: an array of 1-2 sentence snippets around every place that paper cites the source. We apply the same morphological keyword regex to those snippets locally. A paper whose citation contexts don't contain the keyword is rejected without downloading anything. A paper with a matching context is added to the graph with the snippet as its keyword_hits, plus its title/authors/year/arxiv-id from S2 metadata. No GROBID call, no arXiv download. (--semantic is not available in reverse mode because the snippets are too short for reliable embedding-based matching.)
For a paper with 2000+ citations, this runs in ~10-30 seconds and typically surfaces 20-100 relevant papers, depending on how specific the keyword is.
Deep recursion (--depth > 2) in reverse mode can expand combinatorially. Each level multiplies the number of S2 API calls. Use --reverse-limit to cap growth.
Semantic matching (--semantic)
The default regex handles morphological variants (e.g. channel-independent matches channel-independence, channel independently) but misses papers that express the same concept with different vocabulary: "univariate processing", "per-channel modeling", "decoupled channel correlations".
--semantic adds a second pass after the regex: every sentence the regex didn't already match is embedded with a sentence-transformer model (default: all-mpnet-base-v2, ~420MB) and compared to the keyword by cosine similarity. Sentences above the threshold (default 0.40) are added as additional hits. The result is a union: all regex matches plus any conceptual matches the embedding caught.
Semantic hits appear in the info panel with a purple SEM badge labeled "conceptual match", so the user can distinguish them from regex hits at a glance. The header also shows a breakdown (e.g. "7 keyword hit(s) (5 regex + 2 semantic)").
--semantic-model NAME switches to a different model (e.g. all-MiniLM-L6-v2 for a lighter 80MB alternative). --semantic-threshold T tunes the similarity cutoff. Both flags imply --semantic.
--semantic adds ~500MB of dependencies (sentence-transformers + PyTorch). The first model load takes 5-10 seconds, then stays cached in memory. Not available in reverse trace mode.
Citracer can compare a new trace against a previous one to highlight what changed. This turns a one-shot snapshot into a monitoring workflow:
# Initial trace - export a baseline
# Months later - re-run the same trace and diff
Nodes that weren't in baseline.json are colored orange and labeled NEW in the info panel. Their original status (analyzed, no match, unavailable) is preserved; the orange overlay is purely visual. The legend gains a clickable new (since last run) row to toggle them on/off.
--since YYYY or --since YYYY-MM highlights nodes published on or after a date. When combined with --diff, both conditions must be met (intersection): the paper must be absent from the baseline and published after the given date.
Month-level precision uses the publicationDate field from Semantic Scholar (YYYY-MM-DD) when available. When only the publication year is known, --since 2024-06 falls back to a year-only comparison (year >= 2024). Nodes with no known date are skipped and a warning is logged.
Both is_new flags (on nodes and edges) are included in JSON and GraphML exports, so downstream scripts can consume the diff without re-running citracer.
paper_id is not fully stable across runs. If a paper was resolved by title hash in one run and by DOI in another, it may falsely appear as "new". Re-running both traces from the same cache directory minimizes this.
argparse entry point + GROBID health check + .env loader
โโโ pdf_parser.py
GROBID + TEI walking + figure-noise filter + paragraph merge + narrative ref supplementation + pymupdf fallback
โโโ keyword_matcher.py
morphological regex + sentence-based ref association (pysbd)
โโโ reference_resolver.py
arXiv-first cascade resolver (arxiv โ S2 โ OpenReview โ Sci-Hub โ OA โ preprints) with SQLite cache
โโโ source_resolver.py
routes --pdf / --doi / --arxiv / --url inputs to a local PDF path
โโโ preprint_resolver.py
maps DOIs to preprint server PDF URLs (bioRxiv, medRxiv, ChemRxiv, SSRN, PsyArXiv, AgriXiv, engrXiv)
โโโ metadata_enrichment.py
OpenAlex API client for enriching nodes with abstract, citation count, and OA URLs
โโโ metadata_cache.py
SQLite-backed key/value store for resolver metadata, thread-safe
โโโ analytics.py
bibliometric metrics: PageRank, betweenness, pivot detection, timeline
โโโ cross_citation.py
post-trace pass that adds dashed bibliographic-only edges between graph nodes
โโโ diff.py
--diff / --since: compare against a previous trace, mark new nodes/edges
โโโ tracer.py
BFS recursion with parallel parsing, deduplication, year anchoring
โโโ visualizer.py
pyvis rendering pipeline
โโโ exporter.py
GraphML / JSON export (includes analytics and manifest)
โโโ manifest.py
reproducibility manifest generation
โโโ models.py
dataclasses
โโโ api_types.py
TypedDicts for arxiv / Semantic Scholar / OpenReview / OpenAlex payloads
โโโ constants.py
every tunable threshold and timeout, in one place
โโโ user_config.py
persistent user-level config (~/.citracer/config.json)
โโโ utils.py
ID normalization, hashing, tqdm-safe logging setup
โโโ templates/ โโโ overlay.html.tmpl
the interactive control panel (HTML/CSS/JS) injected into the pyvis output
## ๐งฉ Dependencies
<div class="table-wrapper">
<table>
<thead>
<tr><th><p>Package</p></th><th><p>Used for</p></th></tr>
</thead>
<tbody>
<tr><td><p><a href="https://github.com/kermitt2/grobid">GROBID</a></p></td><td><p>PDF structural parsing (external service)</p></td></tr>
<tr><td><p><a href="https://lxml.de/">lxml</a></p></td><td><p>TEI XML processing</p></td></tr>
<tr><td><p><a href="https://pymupdf.readthedocs.io/">pymupdf</a></p></td><td><p>PDF text extraction (parser fallback)</p></td></tr>
<tr><td><p><a href="https://github.com/lukasschwab/arxiv.py">arxiv</a></p></td><td><p>arXiv search and download</p></td></tr>
<tr><td><p><a href="https://github.com/nipunsadvilkar/pySBD">pysbd</a></p></td><td><p>Sentence boundary detection</p></td></tr>
<tr><td><p><a href="https://pyvis.readthedocs.io/">pyvis</a></p></td><td><p>Interactive HTML graph rendering</p></td></tr>
<tr><td><p><a href="https://github.com/rapidfuzz/RapidFuzz">rapidfuzz</a></p></td><td><p>Fuzzy title matching</p></td></tr>
<tr><td><p><a href="https://networkx.org/">networkx</a></p></td><td><p>Graph analytics (centrality, components, PageRank)</p></td></tr>
<tr><td><p><a href="https://requests.readthedocs.io/">requests</a></p></td><td><p>HTTP client</p></td></tr>
<tr><td><p><a href="https://github.com/tqdm/tqdm">tqdm</a></p></td><td><p>Progress bar</p></td></tr>
<tr><td><p><a href="https://github.com/theskumar/python-dotenv">python-dotenv</a></p></td><td><p>Loading the Semantic Scholar key from a <code>.env</code> file</p></td></tr>
<tr><td><p><a href="https://www.sbert.net/">sentence-transformers</a></p></td><td><p>Semantic matching via sentence embeddings (optional: <code>pip install citracer[semantic]</code>)</p></td></tr>
<tr><td><p><a href="https://katex.org">KaTeX</a></p></td><td><p>LaTeX math rendering in the HTML output (CDN)</p></td></tr>
<tr><td><p><a href="https://visjs.github.io/vis-network/docs/network/">vis-network</a></p></td><td><p>Interactive network rendering (via pyvis, CDN)</p></td></tr>
</tbody>
</table>
</div>
External APIs:
- [arXiv API](https://info.arxiv.org/help/api/index.html)
- [Semantic Scholar Graph API](https://api.semanticscholar.org/api-docs/graph)
- [OpenReview API](https://docs.openreview.net/reference/api-v2)
- [OpenAlex API](https://docs.openalex.org/) (metadata enrichment, opt-in via `--enrich`)
- [Sci-Hub](https://sci-hub.in/) (paywall bypass for PDF download)
- Preprint servers: [bioRxiv](https://www.biorxiv.org/), [medRxiv](https://www.medrxiv.org/), [ChemRxiv](https://chemrxiv.org/), [SSRN](https://www.ssrn.com/), [PsyArXiv](https://psyarxiv.com/), [AgriXiv](https://agrixiv.org/), [engrXiv](https://engrxiv.org/) (PDF download via DOI detection)
## โ ๏ธ Limitations
- GROBID misclassifies a small fraction of references, in particular sub-citations with letter suffixes like `Liu et al., 2024b`, which the supplementation pass can't disambiguate. These are silently dropped.
- The narrative-citation supplementation pass skips ambiguous `(surname, year)` signatures (e.g. two different Zhou 2022 papers in the bibliography). These missed cases are rare but do happen in survey-heavy papers.
- pysbd handles most academic abbreviations but can occasionally split mid-sentence; falling back to `--context-window 300` is sometimes useful.
- arXiv enforces ~3 seconds between requests, so the first run on a deep trace can take several minutes. The local cache makes subsequent runs fast.
- Papers that cannot be resolved through any source in the download cascade (arXiv, OpenReview, Sci-Hub, S2 open-access, preprint servers) appear as `unavailable` red nodes. Books and some workshop proceedings are typically not retrievable. Use `--supply-pdf` to provide PDFs manually (local path or URL) for these nodes.
- The "Fruchterman-Reingold" layout option is implemented via vis.js's `forceAtlas2Based` solver, which is the closest approximation available natively. A proper Kamada-Kawai implementation isn't offered because vis.js doesn't ship one.
- `--semantic` matching depends on the quality of the sentence-transformer model. The default `all-mpnet-base-v2` was benchmarked at F1=0.93 on academic citation text (vs 0.86 for `all-MiniLM-L6-v2`). Domain-specific keywords may benefit from threshold tuning. Semantic matching is not available in reverse trace mode.
## ๐งช Development
```bash
pip install -r requirements-dev.txt
pytest tests/ -v
The test suite is hermetic, with no GROBID and no network. GROBID output is exercised via a pre-baked TEI fixture in tests/fixtures/sample.tei.xml, and every external API (arXiv, Semantic Scholar, OpenReview, PDF downloads) is mocked. Runs in under 2 seconds.
CI runs the suite on Python 3.10 / 3.11 / 3.12 via GitHub Actions on every push to main, every pull request, and on manual dispatch from the Actions tab. See .github/workflows/tests.yml.
๐ Citation
If you use citracer in your research, please cite it as:
โ๏ธ Authors
- Marc Pinet - Initial work - marcpinet