Categorygithub.com/bojanz/currency
modulepackage
1.3.0
Repository: https://github.com/bojanz/currency.git
Documentation: pkg.go.dev

# README

currency Build Coverage Status Go Report Card PkgGoDev

Handles currency amounts, provides currency information and formatting.

Powered by CLDR v46, in just ~40kb of data.

Backstory: https://bojanz.github.io/price-currency-handling-go/

Features

  1. All currency codes, their numeric codes and fraction digits.
  2. Currency symbols and formats for all locales.
  3. Country mapping (country code => currency code).
  4. Amount struct, with value semantics (Fowler's Money pattern)
  5. Formatter, for formatting amounts and parsing formatted amounts.
    amount, _ := currency.NewAmount("275.98", "EUR")
    total, _ := amount.Mul("4")

    locale := currency.NewLocale("fr")
    formatter := currency.NewFormatter(locale)
    fmt.Println(formatter.Format(total)) // 1 103,92 €

    // Convert the amount to Iranian rial and show it in Farsi.
    total, _ = total.Convert("IRR", "45.538")
    total = total.Round()
    locale = currency.NewLocale("fa")
    formatter = currency.NewFormatter(locale)
    fmt.Println(formatter.Format(total)) // ‎ریال ۵۰٬۲۷۰

Design goals

Real decimal implementation under the hood.

Currency amounts can't be floats. Storing integer minor units (2.99 => 299) becomes problematic once there are multiple currencies (difficult to sort in the DB), or there is a need for sub-minor-unit precision (due to merchant or tax requirements, etc). A real arbitrary-precision decimal type is required. Since Go doesn't have one natively, a userspace implementation is used, provided by the cockroachdb/apd package. The Amount struct provides an easy to use abstraction on top of it, allowing the underlying implementation to be replaced in the future without a backwards compatibility break.

Smart filtering of CLDR data.

The "modern" subset of CLDR locales is used, reducing the list from ~560 to ~370 locales.

Once gathered, locales are filtered to remove all data not used by this package, and then deduplicated by parent (e.g. don't keep fr-CH if fr has the same data).

Currency symbols are grouped together to avoid repetition. For example:

"ARS": {
    {"ARS", []string{"en", "fr-CA"}},
    {"$", []string{"es-AR"}},
    {"$AR", []string{"fr"}},
}

Currency names are not included because they are rarely shown, but need significant space. Instead, they can be fetched on the frontend via Intl.DisplayNames.

Easy to compare.

Amount structs can be compared via google/go-cmp thanks to the built-in Equal() method.

Usable with a PostgreSQL composite type.

Thanks to the driver.Valuer and sql.Scanner interfaces, applications using the pgx driver can store amounts in a composite type.

Example schema:

CREATE TYPE price AS (
   number NUMERIC,
   currency_code TEXT
);

CREATE TABLE products (
   id CHAR(26) PRIMARY KEY,
   name TEXT NOT NULL,
   price price NOT NULL,
   created_at TIMESTAMPTZ NOT NULL,
   updated_at TIMESTAMPTZ
);

Note that the number and currency_code columns can have any name, only their ordering matters.

Example struct:

type Product struct {
	ID          string
	Name        string
	Price       currency.Amount
	CreatedAt   time.Time
	UpdatedAt   time.Time
}

Example scan:

p := Product{}
row := tx.QueryRow(ctx, `SELECT id, name, price, created_at, updated_at FROM products WHERE id = $1`, id)
err := row.Scan(&p.ID, &p.Name, &p.Price, &p.CreatedAt, &p.UpdatedAt)

See our database integration notes for other examples (MySQL/MariaDB, SQLite).

# Functions

ForCountryCode returns the currency code for a country code.
GetCurrencyCodes returns all known currency codes.
GetDigits returns the number of fraction digits for a currency code.
GetNumericCode returns the numeric code for a currency code.
GetSymbol returns the symbol for a currency code.
IsValid checks whether a currency code is valid.
NewAmount creates a new Amount from a numeric string and a currency code.
NewAmountFromBigInt creates a new Amount from a big.Int and a currency code.
NewAmountFromInt64 creates a new Amount from an int64 and a currency code.
NewFormatter creates a new formatter for the given locale.
NewLocale creates a new Locale from its string representation.

# Constants

CLDRVersion is the CLDR version from which the data is derived.
DefaultDigits is a placeholder for each currency's number of fraction digits.
DisplayCode shows the currency code.
DisplayNone shows nothing, hiding the currency.
DisplaySymbol shows the currency symbol.
RoundDown rounds towards 0, truncating extra digits.
RoundHalfDown rounds up if the next digit is > 5.
RoundHalfEven rounds up if the next digit is > 5.
RoundHalfUp rounds up if the next digit is >= 5.
RoundUp rounds away from 0.

# Structs

Amount stores a decimal number with its currency code.
Formatter formats and parses currency amounts.
InvalidCurrencyCodeError is returned when a currency code is invalid or unrecognized.
InvalidNumberError is returned when a numeric string can't be converted to a decimal.
Locale represents a Unicode locale identifier.
MismatchError is returned when two amounts have mismatched currency codes.

# Type aliases

Display represents the currency display type.
RoundingMode determines how the amount will be rounded.