Private Symfony Recipes (or: how I stopped repeating myself)
I got tired of setting up the same things in every Symfony project.
Same configs.
Same services.
Same little pieces of glue code that aren’t hard—but add up.
Symfony recipes solve a lot of this.
Until you need your own.
At some point I decided:
“Surely there’s a way to do this privately.”
So I sat down one evening and started digging.
Docs open.
Composer configs half-understood.
A couple of tabs that looked promising but didn’t quite connect.
And for a while… nothing clicked.
I remember getting to the point where I was about ready to reach out to someone a lot smarter than me and just ask:
“Alright, what am I missing here?”
And then it finally clicked.
🧠 The idea
Symfony recipes aren’t magic.
They’re just:
- A repository
- With a specific structure
- That Flex knows how to read
So instead of relying only on the public recipes repo, you can point your project at your own.
A private recipes repository.
🪓 The basic setup (no fluff)
At a high level, you need:
1. A recipes repository
This is just a git repo with a structure like:
recipes/
vendor/
package/
version/
manifest.json
config/
src/
The important piece is the manifest.json.
That’s what tells Symfony Flex what to do.
2. A manifest
Example:
{
"copy-from-recipe": {
"config/": "%CONFIG_DIR%/"
}
}
This is where you define:
- files to copy
- configs to apply
- environment setup
3. Tell Composer where to find it
In your project’s composer.json:
{
"extra": {
"symfony": {
"endpoint": [
"https://your-private-recipes-repo",
"flex://defaults"
]
}
}
}
👉 Order matters. Your repo goes first.
4. Install your package
Now when you require your package:
composer require your/package
Symfony Flex will:
- check your private repo first
- apply your recipe
- then fall back to defaults if needed
💡 What I actually use this for
This is where it gets useful.
My private recipes handle things like:
- Shared configuration across projects
- Common service wiring
- Internal conventions I don’t want to rethink every time
Instead of:
“Alright, how did I set this up last time?”
It becomes:
“composer require → done”
⚠️ The gotchas (aka the part that took way too long)
A few things that tripped me up:
- The structure has to be exactly right
- The endpoint configuration is not intuitive the first time
- Debugging why a recipe didn’t apply is… not fun
If it’s not working, check:
- Is your repo reachable?
- Is the path correct?
- Is the version directory matching what Flex expects?
Most of my time wasn’t spent building this.
It was spent figuring out why it wasn’t working.
⚡ The payoff
Once it clicked, everything got easier.
New project?
- Require package
- Config shows up
- Services wired
- Done
No more:
- digging through old repos
- copying configs manually
- forgetting one small but important detail
Closing
This isn’t a flashy feature.
It’s not something you demo.
But it’s one of those things that quietly saves time every single time you start something new.
And if you’ve ever found yourself thinking:
“I’ve done this before… why am I doing it again?”
This might be worth the evening it takes to figure out.
Even if that evening is a little frustrating.

























