Skip to content

A Python library for dealing with VAT MOSS and Norway VAT on digital services. Includes VAT ID validation, rate calculation based on place of supply, exchange rate and currency tools for invoices.

License

Notifications You must be signed in to change notification settings

wbond/vat_moss-python

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

32 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

vat_moss

A Python library for VAT MOSS tasks required of non-EU companies selling software to customers in the EU and Norway. Functionality includes:

  • Determining the VAT rate for a customer based on any of the following:
    • Billing address
    • Declared country of residence
    • IP address geolocation via a GeoLite2 database
    • Telephone number
  • Validation of EU and Norwegian VAT IDs against government web services
  • Tools for generating VAT-compliant invoices:
    • Fetching European Central Bank exchange rate information
    • Configuring exchange rate information for the money package
    • Formatting foreign currencies when displaying VAT tax due in national currency

This library has codified all of the standard rate VAT tax rules as of January 2015, and includes code to handle the various VAT exemptions that occur in the EU. This was primarily built to support companies selling software licenses or SaaS. Ebook tax rates may be different - this library does not currently differentiate for those.

Resources

In the process of writing this library, I performed quite a bit of research about VAT, VAT MOSS and how to deal with it as a small software company. Hopefully the information below will prove useful to others:

  • VAT MOSS Overview - a document discussing the non-code aspects of VAT MOSS, such as general concepts and terms, proof of supply details, exchange rates, invoices, registration and returns
  • VAT MOSS Implementation Approach - a document discussing how I am using this library and vat-moss.js to deal with VAT MOSS
  • vat-moss.js - a port of this library to Javascript. Some features are different primarily due to the fact that the VAT validation webservices do not support cross-domain AJAX requests.

Dependencies

Python 2.7, 3.3 or 3.4. No third-party packages required.

Version

0.11.0 - changelog

Installation

pip install vat_moss

API

Code examples are written in Python 3. The only change that should be needed to convert them to Python 2 is to replace urllib.error with urllib2.

Determine VAT Rate from Billing Address

The user's VAT Rate can be determined by processing a payment and using the billing address from the payment provider.

The method signature is vat_moss.billing_address.calculate_rate(country_code, postal_code, city). This will return a tuple of (Decimal rate, country code, exception name or None). Examples:

  • (Decimal('0.0'), 'US', None)
  • (Decimal('0.19'), 'DE', None)
  • (Decimal('0.0'), 'DE', 'Heligoland')

The exception name will be one of the exemptions to the normal VAT rates. See the end of http://ec.europa.eu/taxation_customs/resources/documents/taxation/vat/how_vat_works/rates/vat_rates_en.pdf for a full list.

import vat_moss.billing_address

try:
    # Values from payment provider
    country_code = 'US'
    postal_code = '01950'
    city = 'Newburyport'

    result = vat_moss.billing_address.calculate_rate(country_code, postal_code, city)
    rate, country_code, exception_name = result

    # Save place of supply proof

except (ValueError):
    # One of the user input values is empty or not a string

For place of supply proof, you should save the country code, postal code, city name, detected rate and any exception name.

Determine VAT Rate from Declared Residence

The user's VAT Rate can be determined by prompting the user with a list of valid countries obtained from vat_moss.declared_residence.options(). If the user chooses a country with one or more exceptions, the user should be presented with another list of "None" and each exception name. This should be labeled something like: "Special VAT Rate".

The method signature to get the appropriate rate is vat_moss.declared_residence.calculate_rate(country_code, exception_name). This will return a tuple of (Decimal rate, country code, exception name or None). Examples:

  • (Decimal('0.0'), 'US', None)
  • (Decimal('0.19'), 'DE', None)
  • (Decimal('0.0'), 'DE', 'Heligoland')

The exception name will be one of the exemptions to the normal VAT rates. See the end of http://ec.europa.eu/taxation_customs/resources/documents/taxation/vat/how_vat_works/rates/vat_rates_en.pdf for a full list.

import vat_moss.declared_residence

try:
    # Loop through this list of dicts and build a <select> using the 'name' key
    # as the text and 'code' key as the value. The 'exceptions' key is a list of
    # valid VAT exception names for that country. You will probably need to
    # write some JS to show a checkbox if the selected country has exceptions,
    # and then present the user with another <select> allowing then to pick
    # "None" or one of the exception names.
    residence_options = vat_moss.declared_residence.options()

    # Values from user input
    country_code = 'DE'
    exception_name = 'Heligoland'

    result = vat_moss.declared_residence.calculate_rate(country_code, exception_name)
    rate, country_code, exception_name = result

    # Save place of supply proof

except (ValueError):
    # One of the user input values is empty or not a string

For place of supply proof, you should save the country code, detected rate and any exception name.

Determine VAT Rate from GeoLite2 Database

The company MaxMind offers a [http://dev.maxmind.com/geoip/geoip2/geolite2/](free geo IP lookup database).

For this you'll need to install something like the nginx module, apache module or one of the various programming language packages.

Personally I like to do it at the web server level since it is fast and always available.

Once you have the data, you need to feed the country code, subdivision name and city name into the method vat_moss.geoip2.calculate_rate(country_code, subdivision, city, address_country_code, address_exception). The subdivision should be the first subdivision name from the GeoLite2 database. The address_country_code and address_exception should be from vat_moss.billing_address.calculate_rate() or vat_moss.declared_residence.calculate_rate(). This information is necessary since some exceptions are city-specific and can't solely be detected by the user's IP address. This will return a tuple of (Decimal rate, country code, exception name or None). Examples:

  • (Decimal('0.0'), 'US', None)
  • (Decimal('0.19'), 'DE', None)
  • (Decimal('0.0'), 'DE', 'Heligoland')

The exception name will be one of the exemptions to the normal VAT rates. See the end of http://ec.europa.eu/taxation_customs/resources/documents/taxation/vat/how_vat_works/rates/vat_rates_en.pdf for a full list.

import vat_moss.geoip2

try:
    # Values from web server or API
    ip = '8.8.4.4'
    country_code = 'US'
    subdivision_name = 'Massachusetts'
    city_name = 'Newburyport'

    # Values from the result of vat_moss.billing_address.calculate_rate() or
    # vat_moss.declared_residence.calculate_rate()
    address_country_code = 'US'
    address_exception = None

    result = vat_moss.geoip2.calculate_rate(country_code, subdivision_name, city_name, address_country_code, address_exception)
    rate, country_code, exception_name = result

    # Save place of supply proof

except (ValueError):
    # One of the user input values is empty or not a string

For place of supply proof, you should save the IP address; country code, subdivision name and city name from GeoLite2; the detected rate and any exception name.

Omitting address_country_code and address_exception

If the address_country_code and address_exception are not provided, in some situations this function will not be able to definitively determine the VAT rate for the user. This is because some exemptions are for individual cities, which are only tracked via GeoLite2 at the district level. This sounds confusing, but if you look at the GeoLite2 data, you'll see some of the city entries are actually district names. Lame, I know.

In those situations, a vat_moss.errors.UndefinitiveError() exception will be raised.

Determine VAT Rate from International Phone Number

Prompt the user for their international phone number (with leading +). Once you have the data, you need to feed the phone number to vat_moss.phone_number.calculate_rate(phone_number, address_country_code, address_exception). The address_country_code and address_exception should be from vat_moss.billing_address.calculate_rate() or vat_moss.declared_residence.calculate_rate(). This information is necessary since some exceptions are city-specific and can't solely be detected by the user's phone number. This will return a tuple of (Decimal rate, country code, exception name or None). Examples:

  • (Decimal('0.0'), 'US', None)
  • (Decimal('0.19'), 'DE', None)
  • (Decimal('0.0'), 'DE', 'Heligoland')

The exception name will be one of the exemptions to the normal VAT rates. See the end of http://ec.europa.eu/taxation_customs/resources/documents/taxation/vat/how_vat_works/rates/vat_rates_en.pdf for a full list.

import vat_moss.phone_number

try:
    # Values from user
    phone_number = '+19785720330'

    # Values from the result of vat_moss.billing_address.calculate_rate() or
    # vat_moss.declared_residence.calculate_rate()
    address_country_code = 'US'
    address_exception = None

    result = vat_moss.phone_number.calculate_rate(phone_number, address_country_code, address_exception)
    rate, country_code, exception_name = result

    # Save place of supply proof

except (ValueError):
    # One of the user input values is empty or not a string

For place of supply proof, you should save the phone number, detected rate and any exception name.

Omitting address_country_code and address_exception

If the address_country_code and address_exception are not provided, in some situations this function will not be able to definitively determine the VAT rate for the user. This is because some exemptions are for individual cities, which can not be definitely determined by the user's phone number area code.

In those situations, a vat_moss.errors.UndefinitiveError() exception will be raised.

Validate a VAT ID

EU businesses do not need to be charged VAT. Instead, under the VAT reverse charge mechanism, you provide them with an invoice listing the price of your digital services, and they are responsible for figuring out the VAT due and paying it, according to their normal accounting practices.

The way to determine if a customer in the EU is a business is to validate their VAT ID.

VAT IDs should contain the two-character country code. See http://en.wikipedia.org/wiki/VAT_identification_number for more info.

The VAT ID can have spaces, dashes or periods within it. Some basic formatting checks are done to prevent expensive HTTP calls to the web services that validate the numbers. However, extensive checksum are not validated. If the format looks fairly correct, it gets sent along to the web server.

import vat_moss.id
import vat_moss.errors
import urllib.error

try:
    result = vat_moss.id.validate('GB GD001')
    if result:
        country_code, normalized_id, company_name = result
        # Do your processing to not charge VAT

except (vat_moss.errors.InvalidError):
    # Make the user enter a new value

except (vat_moss.errors.WebServiceUnavailableError):
    # There was an error processing the request within the VIES service.
    #
    # Unfortunately this tends to happen a lot with EU countries because the
    # VIES service is a proxy for 28 separate member-state APIs.
    #
    # Tell your customer they have to pay VAT and can recover it
    # through appropriate accounting practices.

Fetch Exchange Rates for Invoices

When creating invoices, it is necessary to present the VAT tax amount in the national currency of the country the customer resides in. Since most merchants will be selling in a single currency, it will be often necessary to convert amount into one of the 11 currencies used throughout the EU and Norway.

The exchange rates used for these conversions should come from the European Central Bank. They provide an XML file that is updated on business days between 2:15 and 3:00pm CET.

The vat_moss.exchange_rates.fetch() method will download this XML file and return a tuple containing the date of rates, as a string in the format YYYY-MM-DD, and a dict object with the keys being three-character currency codes and the values being Decimal() objects of the current rates, with the Euro (EUR) being the base.

Since these rates are only updated once a day, and the fetching of the XML could be subject to latency, the rates should be fetched once a day and cached locally. To prevent introducing lag to visitors of your site, it may make the most sense to use a scheduled job to fetch the rates and cache then. Personally, I fetch the rates daily and store them in a database table for future reference.

import vat_moss.exchange_rates
import urllib.error

try:
    date, rates = vat_moss.exchange_rates.fetch()
    # Add rates to database table, or other local cache

except (urllib.error.URLError):
    # An error occured fetching the rates - requeue the job

Configure money Package Exchange Rates

The money package for Python is a reasonable choice for working with monetary values. The vat_moss.exchange_rates submodule includes a function that will use the exchange rates from the ECB to configure the exchange rates for money.

The first parameter is the base currency, which should always be EUR when working with the data from the ECB. The second parameter should be a dict with the keys being three-character currency codes and the values being Decimal() objects representing the rates.

from decimal import Decimal
from money import Money
import vat_moss.exchange_rates

# Grab the exchange rates from you local cache
rates = {'EUR': Decimal('1.0000'), 'GBP': Decimal('0.77990'),}
vat_moss.exchange_rates.setup_xrates('EUR', rates)

# Now work with your money
amount = Money('10.00', 'USD')
eur_amount = amount.to('EUR')

Format European Currencies for Invoices

With the laws concerning invoices, it is necessary to show at least the VAT tax due in the national currency of the country where your customer resides. To help in properly formatting the currency amount for the invoice, the vat_moss.exchange_rates.format(amount, currency=None) function exists.

This function accepts either a Money object, or a Decimal object plus a string three-character currency code. It returns the amount formatted using the local country rules for amounts. For currencies that share symbols, such as the Danish Krone, Swedish Krona and Norwegian Krone, the symbols are modified by adding the country initial before kr, as is typical in English writing.

from decimal import Decimal
from money import Money
import vat_moss.exchange_rates

# Using a Money object
amount = Money('4101.79', 'USD')
print(vat_moss.exchange_rates.format(amount))

# Using a decimal and currency code
amount = Decimal('4101.79')
currency = 'USD'
print(vat_moss.exchange_rates.format(amount, currency))

The various output formats that are returned by this function include:

Currency Output
BGN 4,101.79 Lev
CZK 4.101,79 Kč
DKK 4.101,79 Dkr
EUR €4.101,79
GBP £4,101.79
HRK 4.101,79 Kn
HUF 4.101,79 Ft
NOK 4.101,79 Nkr
PLN 4 101,79 Zł
RON 4.101,79 Lei
SEK 4 101,79 Skr
USD $4,101.79

Tests

Almost 500 unit and integrations tests are included with this library. They can be run by executing the following in a terminal:

python tests.py

License

MIT License - see the LICENSE file.

About

A Python library for dealing with VAT MOSS and Norway VAT on digital services. Includes VAT ID validation, rate calculation based on place of supply, exchange rate and currency tools for invoices.

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages