/ecbrates

Simple HTTP server for serving currency rates taken from European Central Bank (ECB). Compatible with subset of fixer.io API.

Primary LanguageC++

ECB rates

This repository hosts a few utility programs for handling historical currencies rates obtained from the European Central Bank.

One (ecb_fetcher) can fetch either historical or daily data, show it on the screen and optionally save as XML to a file.

Another one is a simple http server which serves data as JSON. It can load initial data from a given file or from give URL (there are reasonable defaults). On receiving SIGUSR1 it can fetch more data from given URL (again, there is a default address for it). This can be used in a cron script to do it periodically.

It was written just for fun back in 2018 to test some C++ 17 features and libraries. And refactored in 2024.

Quick start

Prerequisite: gcc or clang supporting C++ 17, ninja, cmake, make and ssl. For example on Debian or Ubuntu system this should be enough:

sudo apt install build-essential g++ libssl-dev ninja-build cmake`

Build and run it

make release
./build_release_cpu_native/bin/ecb_server --port 8080

It will fetch historical data from the ECB (default URLs are provided, call ecb_server without parameters to see all the options) and start listening on a given port.

Options

There are two types of requests are accepted:

/api/latest - to get latest data /api/YYYY-MM-DD - to get data from given day

The optional parameters:

  • base - is the base currency (lower- or uppercase) - the values of other currencies are in relation to base. Default is EUR. Example: usd.
  • symbols - coma-separated list of currencies to return. Example: usd,nok,sek. The result is simply filtered, so if the currency symbols are not existing then simply nothing will be returned.

Getting data

On the other console (or via browser):

curl localhost:8080/api/latest

Result:

{
 "base" : "EUR",
 "date" : "2024-03-27",
 "rates" :
 {
    "AUD" : 1.66,
    "BGN" : 1.9558,
    "BRL" : 5.3969,
    "CAD" : 1.4712,
    "CHF" : 0.9811,
    "CNY" : 7.8182,
    "CZK" : 25.319,
    "DKK" : 7.4582,
    "GBP" : 0.85768,
    "HKD" : 8.4615,
    "HUF" : 395.4,
    "IDR" : 17194.2,
    "ILS" : 3.9582,
    "INR" : 90.134,
    "ISK" : 149.9,
    "JPY" : 163.52,
    "KRW" : 1461.31,
    "MXN" : 17.9367,
    "MYR" : 5.1192,
    "NOK" : 11.6825,
    "NZD" : 1.8043,
    "PHP" : 60.849,
    "PLN" : 4.3148,
    "RON" : 4.9721,
    "SEK" : 11.506,
    "SGD" : 1.4586,
    "THB" : 39.381,
    "TRY" : 34.9005,
    "USD" : 1.0816,
    "ZAR" : 20.4716
  },
 "success" : true
}

Or with parameters:

curl 'localhost:8080/api/latest?base=usd&symbols=eur,nok,dkk'

Result:

{
  "base" : "USD",
  "date" : "2024-03-27",
  "rates" :
  {
    "DKK" : 6.89553,
    "EUR" : 0.924556,
    "NOK" : 10.8011
  },
  "success" : true
}

Longer start

To install it using nginx (setting it up with other web server should be similar):

  • compile the ecb_server (make release) and run the binary (e.g. ecb_server --port 8080). Of course ideally the binary should be started with a starting script every time the system starts.

  • tell nginx to proxy some requests to it:

location /ecbrates/ {
      rewrite ^/ecbrates/(.*) /api/$1 break;
      proxy_pass http://localhost:8080/;
      proxy_set_header Host $host:$server_port;
      proxy_pass_request_body off; # body is not used, so don't forward it
}

Now it should be possible to access the service via the web server at /ecbrates/latest and /ecbrates/yyyy-mm-dd URLs.

One more thing is to periodically send SIGUSR1 signal to the running ecb_server to load more data. ECB publishes new data on every working day, so once a data (at night) should be enough.

Assuming the ecb_server runs as a current user (it does not need any special privileges) one could create a simple crontab task:

  • crontab -e to open an editor
  • add a line to send the signal to all instances of ecb_server: 0 3 * * * killall -SIGUSR1 ecb_server

The line executes the command at 03:00 AM every day.