Show HN: Timber – Ollama for classical ML models, 336x faster than Python
126 points
8 hours ago
| 10 comments
| github.com
| HN
tl2do
5 hours ago
[-]
Since generative AI exploded, it's all anyone talks about. But traditional ML still covers a vast space in real-world production systems. I don't need this tool right now, but glad to see work in this area.
reply
visarga
2 hours ago
[-]
A nice way to use traditional ML models today is to do feature extraction with a LLM and classification on top with trad ML model. Why? because this way you can tune your own decision boundary, and piggy back on features from a generic LLM to power the classifier.

For example CV triage, you use a LLM with a rubric to extract features, choosing the features you are going to rely on does a lot of work here. Then collect a few hundred examples, label them (accept/reject) and train your trad ML model on top, it will not have the LLM biases.

You can probably use any LLM for feature preparation, and retrain the small model in seconds as new data is added. A coding agent can write its own small-model-as-a-tool on the fly and use it in the same session.

reply
benrutter
17 minutes ago
[-]
What do you mean by "feature extraction with an LLM?". I can get this for text based data, but would you do that on numeric data? Seems like there are better tools you could use for auto-ML in that sphere?

Unless by LLM feature extraction you mean something like "have claude code write some preprocessing pipeline"?

reply
mehdibl
6 hours ago
[-]
Ollama is quite a bad example here. Despite popular, it's a simple wrapper and more and more pushed by the app it wraps llama.cpp.

Don't understand here the parallel.

reply
eleventyseven
51 minutes ago
[-]
This is the vLLM of classic ML, not Ollama.
reply
ekianjo
3 hours ago
[-]
I guess the parallel is "Ollama serve" which provides you with a direct REST API to interact with a LLM.
reply
sieve
51 minutes ago
[-]
llama-cpp provides an API server as well via llama-server (and a competent webgui too).
reply
theanonymousone
2 hours ago
[-]
Wouldn't it be much more useful if the request received raw input (i.e. before feature extraction), and not the feature vector?
reply
marcyb5st
38 minutes ago
[-]
You can do that with Onnx. You can graft the preprocessing layers to the actual model [1] and then serve that. Honestly, I already thought that ONNX (CPU at least) was already low level code and already very optimized.

@Author - if you see this is it possible to add comparisons (ie "vanilla" inference latencies vs timber)?

[1] https://gist.github.com/msteiner-google/5f03534b0df58d32abcc... <-- A gist I put together in the past that goes from PyTorch to ONNX and grafts the preprocessing layers to the model, so you can pass the raw input.

reply
kossisoroyce
50 seconds ago
[-]
I'll check this out as soon as I am at my desk.
reply
brokensegue
5 hours ago
[-]
"classical ML" models typically have a more narrow range of applicability. in my mind the value of ollama is that you can easily download and swap-out different models with the same API. many of the models will be roughly interchangeable with tradeoffs you can compute.

if you're working on a fraud problem an open-source fraud model will probably be useless (if it even could exist). and if you own the entire training to inference pipeline i'm not sure what this offers? i guess you can easily swap the backends? maybe for ensembling?

reply
rudhdb773b
5 hours ago
[-]
If the focus is performance, why use a separate process and have to deal with data serialization overhead?

Why not a typical shared library that can be loaded in python, R, Julia, etc., and run on large data sets without even a memory copy?

reply
bob001
3 hours ago
[-]
This lets you not even need Python, r, Julia, etc but directly connect to your backend systems that are presumably in a fast language. If Python is in your call stack then you already don’t care about absolute performance.
reply
kossisoroyce
3 minutes ago
[-]
I owe you a beer!
reply
sriram_malhar
5 hours ago
[-]
Perhaps because the performance is good enough and this approach is much simpler and portable than shared libraries across platforms.
reply
kossisoroyce
2 minutes ago
[-]
Exactly. The objective is to abstract away completely. Shared libraries just add too much overhead.
reply
palashkulsh
2 hours ago
[-]
Nice idea, i needed something like it
reply
Dansvidania
6 hours ago
[-]
Can’t check it out yet, but the concept alone sounds great. Thank you for sharing.
reply
kossisoroyce
4 minutes ago
[-]
You're welcome!
reply
o10449366
4 hours ago
[-]
Can you tell us more about the motivation for this project? I'm very curious if it was driven by a specific use case.

I know there are specialized trading firms that have implemented projects like this, but most industry workflows I know of still involve data pipelines with scientists doing intermediate data transformations before they feed them into these models. Even the c-backed libraries like numpy/pandas still explicitly depend on the cpython API and can't be compiled away, and this data feed step tends to be the bottleneck in my experience.

That isn't to say this isn't a worthy project - I've explored similar initiatives myself - but my conclusion was that unless your data source is pre-configured to feed directly into your specific model without any intermediate transformation steps, optimizing the inference time has marginal benefit in the overall pipeline. I lament this as an engineer that loves making things go fast but has to work with scientists that love the convenience of jupyter notebooks and the APIs of numpy/pandas.

reply
kossisoroyce
5 minutes ago
[-]
The motivation was edge and latency-critical use cases on a product I consulted on. Feature vectors arrived pre-formed and a Python runtime in the hot path wass a non-starter. You're right that for most pipelines the transformation step is the bottleneck, not inference, and Timber doesn't solve that (though the Pipeline Fusion pass compiles sklearn scalers away entirely if your preprocessing is that simple). Timber is explicitly a tool for deployments where you've already solved the data plumbing and the model call itself is what's left to optimize.
reply
OutOfHere
2 hours ago
[-]
It would be safer to use a Zig or Rust or Nim target. C risks memory-unsafe behavior. The risk profile is even bigger for vibe-coded implementations.
reply
kossisoroyce
12 minutes ago
[-]
Fair point in general, but the risk profile here is actually quite low. The generated C is purely computational, with no heap allocation, no pointer arithmetic, no user-controlled memory, no IO. It's essentially a deeply nested if/else tree over a fixed-size float array. The "unsafe" surface in C is largely a non-issue when the code is statically shaped at compile time from a deterministic compiler pass.

Rust/Zig/Nim would add toolchain complexity with minimal safety gain for this specific output shape. Those were my considerations.

reply
jnstrdm05
6 hours ago
[-]
I have been waiting for this! Nice
reply
kossisoroyce
17 minutes ago
[-]
Glad you got it just in time!
reply