eu_vat_rates_data · Ruby
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-EU -
vat_name— official name of the VAT tax in the country's primary official language -
vat_abbr— short abbreviation used locally (e.g. "ALV", "MwSt", "TVA") -
format— human-readable VAT number format (e.g."ATU + 8 digits") — unique to this package -
pattern— regex for VAT number validation + built-invalid_format?()— free, no API key needed — unique to this package - No dependencies — pure Ruby 3.0+
- Data bundled in the gem — 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) · PHP (Packagist) · Go
Installation
gem install eu_vat_rates_data
# or in Gemfile:
gem 'eu_vat_rates_data'Usage
require "eu_vat_rates_data"
# Full rate hash for a country
fi = EuVatRatesData.get_rate("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" => nil,
# "parking" => nil
# }
# Just the standard rate
EuVatRatesData.get_standard_rate("DE") # => 19.0
# EU membership check — false for non-EU countries (GB, NO, CH, ...)
if EuVatRatesData.eu_member?(user_input)
rate = EuVatRatesData.get_rate(user_input)
end
# Dataset membership check (all 44 countries)
if EuVatRatesData.has_rate?(user_input)
rate = EuVatRatesData.get_rate(user_input)
end
# All 44 countries at once
EuVatRatesData.all_rates.each do |code, rate|
puts "#{code}: #{rate['standard']}%"
end
# When were EU rates last fetched?
puts EuVatRatesData.data_version # e.g. "2026-03-27"
# VAT number format validation — no API key, no network call
EuVatRatesData.valid_format?("ATU12345678") # => true
EuVatRatesData.valid_format?("DE123456789") # => true
EuVatRatesData.valid_format?("INVALID") # => false
# Access format metadata directly
at = EuVatRatesData.get_rate("AT")
puts at["format"] # "ATU + 8 digits"
puts at["pattern"] # "^ATU\\d{8}$"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 RubyGems 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.