The faithful assistant for your FrankenPHP Workers.
igor-php is an ultra-fast static linter written in Go that prepares your Symfony application for the persistent memory model of FrankenPHP.
Like the legendary assistant, igor checks every connection and part of your application to ensure it won't "blow up" when the lightning strikes (Worker Mode).
- β‘ Lightning Fast: Scans hundreds of files in milliseconds using Go's native multi-threading.
- π Deep Audit: Automatically detects Symfony projects and audits every shared service defined in the container, including those in
vendor/and external bundles. - π― Surgical Precision: Detects complex state mutations (
$this->prop[],static::$prop, increments) without false positives. - π§ Intelligent: Verifies not just the presence of
ResetInterface, but ensures all mutated properties are correctly reset. Automatically ignoresreadonlyproperties and classes (PHP 8.1+) as they are immutable by design. - π‘οΈ Safety First: Catches dangerous
exit()ordie()calls, and warns about PHP Superglobals ($_GET,$_POST, etc.) or local static variables that could leak state between requests. - π Zero Noise: Automatically ignores
Symfony\andDoctrine\namespaces, and common data folders (Entity,Dto,ApiResource). - π¦ Project vs. Vendor: Clear separation between your code and third-party dependencies, with tailored recommendations for each.
- π― Selective Ignore: Skip specific lines using the
// @igor-ignorecomment, or target entire classes, methods, and properties using modern PHP 8 Attributes (#[WorkerSafe]).
- Go: Required to compile or install the binary.
- PHP 8.1+: Required for the Deep Audit mode. Igor uses PHP Reflection to precisely locate service files within your project and
vendor/directory. Without PHP, Igor will fall back to a standard directory scan.
composer require --dev igor-php/igor-phpTo make Igor even more reliable, you can enable the embedded PHP bundle. It generates a precise service map directly from your container, which Igor Go will use to audit your services.
Add the bundle to your config/bundles.php:
return [
// ...
IgorPhp\IgorBundle\IgorPhpBundle::class => ['dev' => true, 'test' => true],
];Or manually in your Kernel.php:
public function registerBundles(): iterable
{
// ...
if ($this->getEnvironment() === 'dev') {
yield new IgorPhp\IgorBundle\IgorPhpBundle();
}
}go install github.com/igor-php/igor-php@latestIgor can automatically detect your project type Symfony and generate a default configuration for you:
# Initialize igor.json
igor-php init
# Initialize with a custom name/path
igor-php init -c custom-igor.jsonOnce initialized (or using defaults), let Igor audit your project:
# Standard usage
igor-php .
# Generate a baseline to ignore existing errors
igor-php --generate-baseline
# Custom configuration file
igor-php --config custom-igor.json .
# or shorthand
igor-php -c custom-igor.json .
# Custom console path, environment and verbose mode
igor-php --console app/console --env stage --verbose .
# Non-Symfony project or skip Symfony discovery
igor-php --no-agent .Igor can also audit standard PHP projects that don't use the Symfony framework. In this case, use the --no-agent flag to disable automatic container discovery.
When using Igor without Symfony, you should manually define which directories or vendor packages to audit in your igor.json:
{
"scan_vendors": ["my-company/internal-library"],
"exclude": ["tests", "Data", "vendor/symfony"]
}π‘ Note: Without Symfony, Igor performs a recursive scan of your project directory (excluding folders in
exclude). Usingscan_vendorsallows you to force the audit of specific third-party libraries even without the Symfony service map.
Want to understand why Igor is vital for your Worker environment? Check these real-world scenarios from our Leak Lab:
Igor identifies all leaks (Static, Stateful, Incomplete Reset) and dangerous global function calls automatically.
We've built an interactive laboratory using Symfony and FrankenPHP. You can run it locally with Docker and see the memory leaks with your own eyes.
When a Symfony project is detected, Igor combines three layers of discovery to ensure maximum reliability:
- Level 1: Project Code (Recursive Scan): Igor scans all PHP files in your project directory (excluding
vendor,var,tests, etc.). This ensures that even if Symfony "inlines" or "hides" a service for optimization, Igor will still find and audit it. - Level 2: Smart Filtering (Composer): Igor automatically parses your
composer.jsonto identify packages inrequire-dev. It will automatically exclude any service originating from these packages to reduce noise and focus only on production-ready code. - Level 3: Igor Agent (Embedded Bundle): By enabling the optional PHP bundle, Igor becomes "infallible". The bundle hooks into the Symfony compilation process to export the exact map of all active shared services.
Igor reads the require-dev section of your composer.json. When it audits your Symfony container, it checks the physical path of each service. If a service is located inside a vendor/ directory belonging to a dev package (like phpunit/phpunit or symfony/maker-bundle), Igor will automatically skip it.
The IgorPhpBundle includes a CompilerPass that runs every time you clear your Symfony cache (php bin/console cache:clear).
β οΈ Important: You must runphp bin/console cache:clearwhenever you add or modify services in your Symfony project to ensure the Igor Agent map is up-to-date.
- What it does: It iterates through the
ContainerBuilder, identifies all Shared Services, and extracts their class names and IDs. - The Cache: It writes this information into a small JSON file:
var/cache/<env>/igor_service_map.json. - The Benefit: The Go binary reads this file instead of executing the heavy
debug:containercommand. This makes the audit launch near-instant and ensures 100% accuracy, even for services added by complex compiler passes or decorators.
{
"definitions": {
"app.mail_service": {
"class": "App\\Service\\MailService",
"public": true,
"shared": true
},
"logger": {
"class": "Monolog\\Logger",
"public": true,
"shared": true
}
},
"aliases": {
"Psr\\Log\\LoggerInterface": "logger"
}
}You can customize Igor's behavior by creating an igor.json file at the root of your project:
{
"exclude": ["vendor", "var", "src/Entity"],
"safe_namespaces": ["Symfony\\", "Doctrine\\", "Twig\\", "IgorPhp\\IgorBundle\\"],
"console_path": "bin/console",
"env": "dev",
"verbose": false
}Time taken: 1.2s
π‘ RECOMMENDATIONS: [PROJECT] Since this is your code, you should refactor these services to be stateless or implement ResetInterface to clear the state between requests. [VENDOR] This is third-party code. If you can't fix it, consider setting a 'max_requests' limit in your Worker configuration to mitigate memory leaks.
---
## βοΈ Configuration
You can customize Igor's behavior by creating an `igor.json` file at the root of your project:
```json
{
"exclude": ["vendor", "tests", "Entity"],
"safe_namespaces": ["Symfony\\", "Doctrine\\", "IgorPhp\\IgorBundle\\"],
"scan_vendors": ["my-company/internal-bundle"],
"baseline": "igor-baseline.json",
"console_path": "bin/console",
"env": "dev",
"verbose": false
}
- exclude: List of directories to skip during indexing.
- safe_namespaces: Igor will ignore state mutations in classes starting with these prefixes.
- scan_vendors: List of sub-directories within
vendor/to scan recursively. - baseline: Path to a baseline file containing findings to ignore.
- console_path: Custom path to the Symfony console binary. Defaults to
bin/console. - env: Symfony environment to use for container analysis. Defaults to
dev. - verbose: Enable verbose output to see skipped services and reasons. Defaults to
false.
Igor can export findings in a structured JSON format and help you triage them using an LLM. This is particularly useful for distinguishing between harmless state (e.g., caches) and dangerous data leaks.
Generate a ready-to-use prompt for your favorite LLM (ChatGPT, Claude, etc.):
# 1. Export the audit to JSON
igor-php --output llm . > audit.json
# 2. Generate the review prompt
igor-php review audit.jsonIgor will create igor-review-prompt.md. Simply copy its content into an LLM to get a detailed security analysis and remediation plan.
Configure Igor to call an LLM directly by updating your igor.json:
If you have gemini-cli installed and configured, Igor can use it directly:
{
"llm": {
"provider": "gemini",
"model": "gemini-1.5-pro"
}
}If you run Ollama locally, Igor can use its OpenAI-compatible endpoint. This is great for privacy, but please note that triage quality depends heavily on the model size. Smaller local models (like Llama 3 8B) are significantly less capable than large online models for complex security triage.
{
"llm": {
"provider": "ollama",
"model": "llama3"
}
}Note: Igor defaults the api_url to http://localhost:11434/v1 for Ollama.
{
"llm": {
"provider": "openai",
"api_url": "https://api.openai.com/v1",
"api_key_env": "OPENAI_API_KEY",
"model": "gpt-4o"
}
}Then run:
# For Option C, ensure the API key is set
export OPENAI_API_KEY=your_secret_key
igor-php review audit.jsonIgor will automatically send the audit to the LLM and save the report to igor-review.md.
If you have a specific line that you know is safe, you can use the // @igor-ignore annotation:
// @igor-ignore
$this->cache = $data; // This line will be ignored
$this->counter++; // @igor-ignore - This line tooInstead of line-by-line comments, you can use modern PHP 8 attributes to exclude entire classes, specific methods, or individual properties.
First, import the attribute (available via the embedded Symfony bundle):
use IgorPhp\IgorBundle\Attribute\WorkerSafe;Then decorate your code elements:
-
Class-level: Ignore all state leak and mutation findings within the entire class.
#[WorkerSafe(scope: 'boot-time', reason: 'Configuration is frozen after warmup')] class MyService { // All mutations and state checks inside this class are ignored }
-
Method-level: Ignore state mutations occurring inside a specific method.
class MyService { #[WorkerSafe] public function warmUp() { $this->cache = ['foo' => 'bar']; // This mutation is ignored } }
-
Property-level: Ignore all mutations on a specific property and exclude it from the
ResetInterfaceverification. Works flawlessly with both standard and constructor-promoted properties!class MyService { #[WorkerSafe] private $cache = []; // Mutations and missing reset checks are ignored public function __construct( #[WorkerSafe] private StatefulService $safeService, // Promoted property is safe! ) {} }
When using the Deep Audit mode (Symfony), Igor might analyze fewer services than your total container count. Use the --verbose flag to see exactly why a service was skipped. Common reasons include:
- π Duplicate File: Multiple Service IDs (aliases, locators, etc.) pointing to the same PHP file. Igor only audits each unique file once.
- β»οΈ Non-shared (Prototype): Services marked as
shared: falseare recreated on every request and don't persist state between workers. They are safe by design. - Ξ» Closures / Synthetic: Services that don't map to a physical PHP class (like Closures or synthetic services) cannot be statically analyzed.
- π‘οΈ Safe Namespace: The class belongs to a namespace defined in
safe_namespaces(likeSymfony\orDoctrine\).
π‘ Pro Tip: If you notice Entities, DTOs, or Data Models appearing in the Igor audit, it means they are registered as "Shared Services" in your Symfony container. This is usually a configuration error in your
services.yaml. You should exclude these directories from autowiring:# config/services.yaml services: App\: resource: '../src/' exclude: - '../src/Entity/' - '../src/Dto/' - '../src/Kernel.php'
Igor is designed to work out-of-the-box in your CI pipelines. It will exit with code 1 if any error is found, effectively stopping your build.
When running inside GitHub Actions, Igor automatically generates inline annotations. This means errors will appear directly in your Pull Request review, right next to the code causing the issue.
name: Static Analysis
on: [push, pull_request]
jobs:
igor:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Setup PHP
uses: shivammathur/setup-php@v2
with:
php-version: '8.3'
- name: Install Dependencies
run: composer install --no-progress --prefer-dist
- name: Warmup Symfony Cache (for Deep Audit)
run: php bin/console cache:warmup --env=dev
- name: Run Igor Audit
run: vendor/bin/igor-php .- Phanalist: Special thanks to
phanalistand its ruleE0012(Stateful Service) which inspired Igor's core mutation detection logic. - Gemini CLI: This project was built with the help of Gemini CLI.
- FrankenPHP: For the amazing server that makes these checks necessary!
MIT