vatnode / eu-vat-rates-data
VAT rates for 44 European countries — EU-27 plus Norway, Switzerland, UK, and more. Updated daily, published automatically when rates change.
Fund package maintenance!
Requires
- php: >=8.1
Requires (Dev)
- phpunit/phpunit: ^11.0
This package is auto-updated.
Last update: 2026-04-03 09:01:43 UTC
README
VAT rates for 44 European countries — EU-27 plus Norway, Switzerland, UK, and more. EU rates sourced from the European Commission TEDB and checked daily. Non-EU rates maintained manually.
- Standard, reduced, super-reduced, and parking rates
eu_memberflag on every country —truefor EU-27,falsefor non-EUvat_name— official name of the VAT tax in the country's primary official languagevat_abbr— short abbreviation used locally (e.g. "ALV", "MwSt", "TVA")format— human-readable VAT number format (e.g."ATU + 8 digits") — unique to this packagepattern— regex for VAT number validation + built-invalidateFormat()— free, no API key needed — unique to this package- No dependencies — pure PHP 8.1+
- Data bundled in the package — works offline, no network calls
- EU rates checked daily via GitHub Actions, new version published only when rates change
Also available in: JavaScript/TypeScript (npm) · Python (PyPI) · Go · Ruby (RubyGems)
Installation
composer require vatnode/eu-vat-rates-data
Usage
use VATNode\EuVatRates\EuVatRates; // Full rate array for a country $fi = EuVatRates::getRate('FI'); // [ // 'country' => 'Finland', // 'currency' => 'EUR', // 'eu_member' => true, // 'vat_name' => 'Arvonlisävero', // 'vat_abbr' => 'ALV', // 'standard' => 25.5, // 'reduced' => [10.0, 13.5], // 'super_reduced' => null, // 'parking' => null, // ] // Just the standard rate EuVatRates::getStandardRate('DE'); // → 19.0 // EU membership check — false for non-EU countries (GB, NO, CH, ...) if (EuVatRates::isEuMember($userInput)) { $rate = EuVatRates::getRate($userInput); } // Dataset membership check (all 44 countries) if (EuVatRates::hasRate($userInput)) { $rate = EuVatRates::getRate($userInput); } // All 44 countries at once foreach (EuVatRates::getAllRates() as $code => $rate) { echo "{$code}: {$rate['standard']}%\n"; } // When were EU rates last fetched? echo EuVatRates::dataVersion(); // e.g. "2026-03-27" // VAT number format validation — no API key, no network call EuVatRates::validateFormat('ATU12345678'); // → true EuVatRates::validateFormat('DE123456789'); // → true EuVatRates::validateFormat('INVALID'); // → false // Access format metadata directly $at = EuVatRates::getRate('AT'); echo $at['format']; // "ATU + 8 digits" echo $at['pattern']; // "^ATU\d{8}$" // Flag emoji from a 2-letter country code — no lookup table, computed from regional indicator symbols EuVatRates::getFlag('FI'); // → "🇫🇮" EuVatRates::getFlag('DE'); // → "🇩🇪" EuVatRates::getFlag('XX'); // → "" (empty string for unknown/invalid codes)
Data source & update frequency
- EU-27 rates: European Commission TEDB, refreshed daily at 07:00 UTC
- Non-EU rates: maintained manually, updated on official rate changes
- Published to Packagist only when actual rates change
Keeping rates current
Rates are bundled at install time. A new package version is published automatically whenever rates change — but your installed version will not update itself.
Recommended: add Renovate or Dependabot to your repo. They detect new versions and open a PR automatically whenever rates change — no manual update commands needed.
Need real-time accuracy? Fetch the always-current JSON directly:
https://cdn.jsdelivr.net/gh/vatnode/eu-vat-rates-data@main/data/eu-vat-rates-data.json
No package needed — parse it with a single fetch() / http.get() / file_get_contents() call and cache locally.
Covered countries
EU-27 (daily auto-updates via EC TEDB):
AT BE BG CY CZ DE DK EE ES FI FR GR HR HU IE IT LT LU LV MT NL PL PT RO SE SI SK
Non-EU Europe (manually maintained):
AD AL BA CH GB GE IS LI MC MD ME MK NO RS TR UA XK
Need to validate VAT numbers?
This package provides VAT rates only. If you also need to validate EU VAT numbers against the official VIES database — confirming a business is VAT-registered — check out vatnode.dev, a simple REST API with a free tier.
curl https://api.vatnode.dev/v1/vat/FI17156132 \ -H "Authorization: Bearer vat_live_..." # → { "valid": true, "companyName": "Suomen Pehmeä Ikkuna Oy" }
License
MIT
If you find this useful, a ⭐ on GitHub is appreciated.