mirror of
https://github.com/mayswind/ezbookkeeping.git
synced 2026-05-13 22:47:33 +08:00
add the National Bank of Kazakhstan exchange rates data source (#564)
* add the National Bank of Kazakhstan exchange rates data source * fix import order, sort exchange rate data by country name. * fix National Bank of Kazakhstan exchange rate reference url * add integration test for the National Bank of Kazakhstan exchange rate provider
This commit is contained in:
@@ -536,6 +536,7 @@ custom_map_tile_server_default_zoom_level = 14
|
||||
# "national_bank_of_georgia": https://nbg.gov.ge/en/monetary-policy/currency
|
||||
# "central_bank_of_hungary": https://www.mnb.hu/en/arfolyamok
|
||||
# "bank_of_israel": https://www.boi.org.il/en/economic-roles/financial-markets/exchange-rates/
|
||||
# "national_bank_of_kazakhstan": https://nationalbank.kz/en/exchangerates/ezhednevnye-oficialnye-rynochnye-kursy-valyut
|
||||
# "central_bank_of_myanmar": https://forex.cbm.gov.mm/index.php/fxrate
|
||||
# "norges_bank": https://www.norges-bank.no/en/topics/Statistics/exchange_rates/
|
||||
# "national_bank_of_poland": https://nbp.pl/en/statistic-and-financial-reporting/rates/
|
||||
|
||||
@@ -135,6 +135,22 @@ func TestExchangeRatesApiLatestExchangeRateHandler_BankOfIsraelDataSource(t *tes
|
||||
checkExchangeRatesHaveSpecifiedCurrencies(t, exchangeRateResponse.BaseCurrency, supportedCurrencyCodes, exchangeRateResponse.ExchangeRates)
|
||||
}
|
||||
|
||||
func TestExchangeRatesApiLatestExchangeRateHandler_NationalBankOfKazakhstan(t *testing.T) {
|
||||
exchangeRateResponse := executeLatestExchangeRateHandler(t, settings.NationalBankOfKazakhstanDataSource)
|
||||
|
||||
if exchangeRateResponse == nil {
|
||||
return
|
||||
}
|
||||
|
||||
assert.Equal(t, "KZT", exchangeRateResponse.BaseCurrency)
|
||||
|
||||
supportedCurrencyCodes := []string{"AED", "AMD", "AUD", "AZN", "BRL", "BYN", "CAD", "CHF", "CNY", "CZK",
|
||||
"DKK", "EUR", "GBP", "GEL", "HKD", "HUF", "INR", "IRR", "JPY", "KGS", "KRW", "KWD", "MDL", "MXN",
|
||||
"MYR", "NOK", "PLN", "RUB", "SAR", "SEK", "SGD", "THB", "TJS", "TRY", "UAH", "USD", "UZS", "ZAR"}
|
||||
|
||||
checkExchangeRatesHaveSpecifiedCurrencies(t, exchangeRateResponse.BaseCurrency, supportedCurrencyCodes, exchangeRateResponse.ExchangeRates)
|
||||
}
|
||||
|
||||
func TestExchangeRatesApiLatestExchangeRateHandler_CentralBankOfMyanmarDataSource(t *testing.T) {
|
||||
exchangeRateResponse := executeLatestExchangeRateHandler(t, settings.CentralBankOfMyanmarDataSource)
|
||||
|
||||
|
||||
@@ -40,6 +40,9 @@ func InitializeExchangeRatesDataSource(config *settings.Config) error {
|
||||
} else if config.ExchangeRatesDataSource == settings.BankOfIsraelDataSource {
|
||||
Container.current = newCommonHttpExchangeRatesDataProvider(config, &BankOfIsraelDataSource{})
|
||||
return nil
|
||||
} else if config.ExchangeRatesDataSource == settings.NationalBankOfKazakhstanDataSource {
|
||||
Container.current = newCommonHttpExchangeRatesDataProvider(config, &NationalBankOfKazakhstanDataSource{})
|
||||
return nil
|
||||
} else if config.ExchangeRatesDataSource == settings.CentralBankOfMyanmarDataSource {
|
||||
Container.current = newCommonHttpExchangeRatesDataProvider(config, &CentralBankOfMyanmarDataSource{})
|
||||
return nil
|
||||
|
||||
@@ -0,0 +1,160 @@
|
||||
package exchangerates
|
||||
|
||||
import (
|
||||
"encoding/xml"
|
||||
"math"
|
||||
"net/http"
|
||||
"time"
|
||||
|
||||
"github.com/mayswind/ezbookkeeping/pkg/core"
|
||||
"github.com/mayswind/ezbookkeeping/pkg/errs"
|
||||
"github.com/mayswind/ezbookkeeping/pkg/log"
|
||||
"github.com/mayswind/ezbookkeeping/pkg/models"
|
||||
"github.com/mayswind/ezbookkeeping/pkg/utils"
|
||||
"github.com/mayswind/ezbookkeeping/pkg/validators"
|
||||
)
|
||||
|
||||
const nationalBankOfKazakhstanExchangeRateUrl = "https://www.nationalbank.kz/rss/rates_all.xml"
|
||||
const nationalBankOfKazakhstanExchangeRateReferenceUrl = "https://nationalbank.kz/en/exchangerates/ezhednevnye-oficialnye-rynochnye-kursy-valyut"
|
||||
const nationalBankOfKazakhstanDataSource = "Қазақстан Республикасының Ұлттық Банкі"
|
||||
const nationalBankOfKazakhstanBaseCurrency = "KZT"
|
||||
|
||||
const nationalBankOfKazakhstanUpdateDateFormat = "02.01.2006"
|
||||
const nationalBankOfKazakhstanUpdateDateTimezone = "Asia/Almaty"
|
||||
|
||||
// NationalBankOfKazakhstanDataSource defines the structure of exchange rates data source of the national bank of Kazakhstan
|
||||
type NationalBankOfKazakhstanDataSource struct {
|
||||
HttpExchangeRatesDataSource
|
||||
}
|
||||
|
||||
// NationalBankOfKazakhstanExchangeRates represents the exchange rates data from the national bank of Kazakhstan
|
||||
type NationalBankOfKazakhstanExchangeRates struct {
|
||||
Channel struct {
|
||||
Items []*NationalBankOfKazakhstanExchangeRate `xml:"item"`
|
||||
} `xml:"channel"`
|
||||
}
|
||||
|
||||
// NationalBankOfKazakhstanExchangeRate represents the exchange rate data from the national bank of Kazakhstan
|
||||
type NationalBankOfKazakhstanExchangeRate struct {
|
||||
Currency string `xml:"title"`
|
||||
Rate string `xml:"description"`
|
||||
Unit string `xml:"quant"`
|
||||
Date string `xml:"pubDate"`
|
||||
}
|
||||
|
||||
// ToLatestExchangeRateResponse returns a view-object according to original data from the national bank of Kazakhstan
|
||||
func (e *NationalBankOfKazakhstanExchangeRates) ToLatestExchangeRateResponse(c core.Context) *models.LatestExchangeRateResponse {
|
||||
if e == nil || len(e.Channel.Items) < 1 {
|
||||
log.Errorf(c, "[national_bank_of_kazakhstan_datasource.ToLatestExchangeRateResponse] exchange rates is empty")
|
||||
return nil
|
||||
}
|
||||
|
||||
timezone, err := time.LoadLocation(nationalBankOfKazakhstanUpdateDateTimezone)
|
||||
if err != nil {
|
||||
log.Errorf(c, "[national_bank_of_kazakhstan_datasource.ToLatestExchangeRateResponse] failed to load timezone, timezone name is %s", nationalBankOfKazakhstanUpdateDateTimezone)
|
||||
return nil
|
||||
}
|
||||
|
||||
exchangeRates := make(models.LatestExchangeRateSlice, 0, len(e.Channel.Items))
|
||||
latestUpdateTime := int64(0)
|
||||
|
||||
for i := 0; i < len(e.Channel.Items); i++ {
|
||||
exchangeRate := e.Channel.Items[i]
|
||||
|
||||
if _, exists := validators.AllCurrencyNames[exchangeRate.Currency]; !exists {
|
||||
continue
|
||||
}
|
||||
|
||||
updateTime, err := time.ParseInLocation(nationalBankOfKazakhstanUpdateDateFormat, exchangeRate.Date, timezone)
|
||||
|
||||
if err != nil {
|
||||
log.Errorf(c, "[central_bank_of_kazakhstan_datasource.ToLatestExchangeRateResponse] failed to parse update date, datetime is %s", exchangeRate.Date)
|
||||
return nil
|
||||
}
|
||||
|
||||
if updateTime.Unix() > latestUpdateTime {
|
||||
latestUpdateTime = updateTime.Unix()
|
||||
}
|
||||
|
||||
finalRate := exchangeRate.ToLatestExchangeRate(c)
|
||||
if finalRate == nil {
|
||||
continue
|
||||
}
|
||||
|
||||
exchangeRates = append(exchangeRates, finalRate)
|
||||
}
|
||||
|
||||
return &models.LatestExchangeRateResponse{
|
||||
DataSource: nationalBankOfKazakhstanDataSource,
|
||||
ReferenceUrl: nationalBankOfKazakhstanExchangeRateReferenceUrl,
|
||||
UpdateTime: latestUpdateTime,
|
||||
BaseCurrency: nationalBankOfKazakhstanBaseCurrency,
|
||||
ExchangeRates: exchangeRates,
|
||||
}
|
||||
}
|
||||
|
||||
// ToLatestExchangeRate returns a data pair according to original data from the national bank of Kazakhstan
|
||||
func (e *NationalBankOfKazakhstanExchangeRate) ToLatestExchangeRate(c core.Context) *models.LatestExchangeRate {
|
||||
rate, err := utils.StringToFloat64(e.Rate)
|
||||
if err != nil {
|
||||
log.Warnf(c, "[national_bank_of_kazakhstan_datasource.ToLatestExchangeRate] failed to parse rate, currency is %s, rate is %s", e.Currency, e.Rate)
|
||||
return nil
|
||||
}
|
||||
|
||||
if rate <= 0 {
|
||||
log.Warnf(c, "[national_bank_of_kazakhstan_datasource.ToLatestExchangeRate] rate is invalid, currency is %s, rate is %s", e.Currency, e.Rate)
|
||||
return nil
|
||||
}
|
||||
|
||||
unit, err := utils.StringToFloat64(e.Unit)
|
||||
if err != nil {
|
||||
log.Warnf(c, "[national_bank_of_kazakhstan_datasource.ToLatestExchangeRate] failed to parse unit, currency=%s, unit=%s", e.Currency, e.Unit)
|
||||
}
|
||||
|
||||
if unit <= 0 {
|
||||
log.Warnf(c, "[national_bank_of_kazakhstan_datasource.ToLatestExchangeRate] unit is less or equal zero, currency is %s, unit is %s", e.Currency, e.Unit)
|
||||
return nil
|
||||
}
|
||||
|
||||
finalRate := rate / unit
|
||||
if math.IsInf(finalRate, 0) {
|
||||
log.Warnf(c, "[national_bank_of_kazakhstan_datasource.ToLatestExchangeRate] final exchange rate calculation failed, currency is %s, unit is %s, rate is %s", e.Currency, e.Unit, e.Rate)
|
||||
return nil
|
||||
}
|
||||
|
||||
return &models.LatestExchangeRate{
|
||||
Currency: e.Currency,
|
||||
Rate: utils.Float64ToString(finalRate),
|
||||
}
|
||||
}
|
||||
|
||||
// BuildRequests returns the national bank of Kazakhstan exchange rates http requests
|
||||
func (e *NationalBankOfKazakhstanDataSource) BuildRequests() ([]*http.Request, error) {
|
||||
req, err := http.NewRequest("GET", nationalBankOfKazakhstanExchangeRateUrl, nil)
|
||||
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return []*http.Request{req}, nil
|
||||
}
|
||||
|
||||
// Parse returns the common response entity according to the national bank of Kazakhstan data source raw response
|
||||
func (e *NationalBankOfKazakhstanDataSource) Parse(c core.Context, content []byte) (*models.LatestExchangeRateResponse, error) {
|
||||
nationalBankOfKazakhstanData := &NationalBankOfKazakhstanExchangeRates{}
|
||||
err := xml.Unmarshal(content, nationalBankOfKazakhstanData)
|
||||
|
||||
if err != nil {
|
||||
log.Errorf(c, "[national_bank_of_kazakhstan_datasource.Parse] failed to parse xml data, content is %s, because %s", string(content), err.Error())
|
||||
return nil, errs.ErrFailedToRequestRemoteApi
|
||||
}
|
||||
|
||||
latestExchangeRateResponse := nationalBankOfKazakhstanData.ToLatestExchangeRateResponse(c)
|
||||
|
||||
if latestExchangeRateResponse == nil {
|
||||
log.Errorf(c, "[national_bank_of_kazakhstan_datasource.Parse] failed to parse latest exchange rate data, content is %s", string(content))
|
||||
return nil, errs.ErrFailedToRequestRemoteApi
|
||||
}
|
||||
|
||||
return latestExchangeRateResponse, nil
|
||||
}
|
||||
@@ -0,0 +1,182 @@
|
||||
package exchangerates
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
|
||||
"github.com/mayswind/ezbookkeeping/pkg/core"
|
||||
"github.com/mayswind/ezbookkeeping/pkg/models"
|
||||
)
|
||||
|
||||
const nationalBankOfKazakhstanMinimumRequiredContent = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n" +
|
||||
" <rss version=\"2.0\">\n" +
|
||||
" <channel>\n" +
|
||||
" <item>\n" +
|
||||
" <title>USD</title>\n" +
|
||||
" <pubDate>28.04.2026</pubDate>\n" +
|
||||
" <description>450.50</description>\n" +
|
||||
" <quant>1</quant>\n" +
|
||||
" </item>\n" +
|
||||
" <item>\n" +
|
||||
" <title>VND</title>\n" +
|
||||
" <pubDate>28.04.2026</pubDate>\n" +
|
||||
" <description>0.018</description>\n" +
|
||||
" <quant>10</quant>\n" +
|
||||
" </item>\n" +
|
||||
" </channel>\n" +
|
||||
"</rss>"
|
||||
|
||||
func TestNationalBankOfKazakhstanDataSource_StandardDataExtractBaseCurrency(t *testing.T) {
|
||||
dataSource := &NationalBankOfKazakhstanDataSource{}
|
||||
context := core.NewNullContext()
|
||||
|
||||
resp, err := dataSource.Parse(context, []byte(nationalBankOfKazakhstanMinimumRequiredContent))
|
||||
assert.Equal(t, nil, err)
|
||||
assert.Equal(t, "KZT", resp.BaseCurrency)
|
||||
}
|
||||
|
||||
func TestNationalBankOfKazakhstanDataSource_StandardDataExtractUpdateTime(t *testing.T) {
|
||||
dataSource := &NationalBankOfKazakhstanDataSource{}
|
||||
context := core.NewNullContext()
|
||||
|
||||
resp, err := dataSource.Parse(context, []byte(nationalBankOfKazakhstanMinimumRequiredContent))
|
||||
assert.Equal(t, nil, err)
|
||||
|
||||
assert.Equal(t, int64(1777316400), resp.UpdateTime)
|
||||
}
|
||||
|
||||
func TestNationalBankOfKazakhstanDataSource_StandardDataExtractExchangeRates(t *testing.T) {
|
||||
dataSource := &NationalBankOfKazakhstanDataSource{}
|
||||
context := core.NewNullContext()
|
||||
|
||||
resp, err := dataSource.Parse(context, []byte(nationalBankOfKazakhstanMinimumRequiredContent))
|
||||
assert.Equal(t, nil, err)
|
||||
|
||||
assert.Contains(t, resp.ExchangeRates, &models.LatestExchangeRate{
|
||||
Currency: "USD",
|
||||
Rate: "450.5",
|
||||
})
|
||||
|
||||
assert.Contains(t, resp.ExchangeRates, &models.LatestExchangeRate{
|
||||
Currency: "VND",
|
||||
Rate: "0.0018",
|
||||
})
|
||||
}
|
||||
|
||||
func TestNationalBankOfKazakhstanDataSource_BlankContent(t *testing.T) {
|
||||
dataSource := &NationalBankOfKazakhstanDataSource{}
|
||||
context := core.NewNullContext()
|
||||
|
||||
_, err := dataSource.Parse(context, []byte(""))
|
||||
assert.NotEqual(t, nil, err)
|
||||
}
|
||||
|
||||
func TestNationalBankOfKazakhstanDataSource_EmptyData(t *testing.T) {
|
||||
dataSource := &NationalBankOfKazakhstanDataSource{}
|
||||
context := core.NewNullContext()
|
||||
|
||||
content := "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n" +
|
||||
"<rss version=\"2.0\">\n" +
|
||||
"<channel>\n" +
|
||||
"</channel>\n" +
|
||||
"</rss>"
|
||||
|
||||
_, err := dataSource.Parse(context, []byte(content))
|
||||
assert.NotEqual(t, nil, err)
|
||||
}
|
||||
|
||||
func TestNationalBankOfKazakhstanDataSource_InvalidCurrency(t *testing.T) {
|
||||
dataSource := &NationalBankOfKazakhstanDataSource{}
|
||||
context := core.NewNullContext()
|
||||
|
||||
content := "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n" +
|
||||
" <rss version=\"2.0\">\n" +
|
||||
" <channel>\n" +
|
||||
" <item>\n" +
|
||||
" <title>XXX</title>\n" +
|
||||
" <pubDate>28.04.2026</pubDate>\n" +
|
||||
" <description>450.50</description>\n" +
|
||||
" <quant>1</quant>\n" +
|
||||
" </item>\n" +
|
||||
" </channel>\n" +
|
||||
"</rss>"
|
||||
|
||||
resp, err := dataSource.Parse(context, []byte(content))
|
||||
assert.Equal(t, nil, err)
|
||||
assert.Len(t, resp.ExchangeRates, 0)
|
||||
}
|
||||
|
||||
func TestNationalBankOfKazakhstanDataSource_InvalidUnit(t *testing.T) {
|
||||
dataSource := &NationalBankOfKazakhstanDataSource{}
|
||||
context := core.NewNullContext()
|
||||
|
||||
content := "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n" +
|
||||
" <rss version=\"2.0\">\n" +
|
||||
" <channel>\n" +
|
||||
" <item>\n" +
|
||||
" <title>USD</title>\n" +
|
||||
" <pubDate>28.04.2026</pubDate>\n" +
|
||||
" <description>450.50</description>\n" +
|
||||
" <quant>null</quant>\n" +
|
||||
" </item>\n" +
|
||||
" </channel>\n" +
|
||||
"</rss>"
|
||||
|
||||
resp, err := dataSource.Parse(context, []byte(content))
|
||||
assert.Equal(t, nil, err)
|
||||
assert.Len(t, resp.ExchangeRates, 0)
|
||||
|
||||
content = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n" +
|
||||
" <rss version=\"2.0\">\n" +
|
||||
" <channel>\n" +
|
||||
" <item>\n" +
|
||||
" <title>USD</title>\n" +
|
||||
" <pubDate>28.04.2026</pubDate>\n" +
|
||||
" <description>450.50</description>\n" +
|
||||
" <quant>0</quant>\n" +
|
||||
" </item>\n" +
|
||||
" </channel>\n" +
|
||||
"</rss>"
|
||||
|
||||
resp, err = dataSource.Parse(context, []byte(content))
|
||||
assert.Equal(t, nil, err)
|
||||
assert.Len(t, resp.ExchangeRates, 0)
|
||||
}
|
||||
|
||||
func TestNationalBankOfKazakhstanDataSource_InvalidRate(t *testing.T) {
|
||||
dataSource := &NationalBankOfKazakhstanDataSource{}
|
||||
context := core.NewNullContext()
|
||||
|
||||
content := "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n" +
|
||||
" <rss version=\"2.0\">\n" +
|
||||
" <channel>\n" +
|
||||
" <item>\n" +
|
||||
" <title>USD</title>\n" +
|
||||
" <pubDate>28.04.2026</pubDate>\n" +
|
||||
" <description>null</description>\n" +
|
||||
" <quant>1</quant>\n" +
|
||||
" </item>\n" +
|
||||
" </channel>\n" +
|
||||
"</rss>"
|
||||
|
||||
resp, err := dataSource.Parse(context, []byte(content))
|
||||
assert.Equal(t, nil, err)
|
||||
assert.Len(t, resp.ExchangeRates, 0)
|
||||
|
||||
content = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n" +
|
||||
" <rss version=\"2.0\">\n" +
|
||||
" <channel>\n" +
|
||||
" <item>\n" +
|
||||
" <title>USD</title>\n" +
|
||||
" <pubDate>28.04.2026</pubDate>\n" +
|
||||
" <description>0</description>\n" +
|
||||
" <quant>1</quant>\n" +
|
||||
" </item>\n" +
|
||||
" </channel>\n" +
|
||||
"</rss>"
|
||||
|
||||
resp, err = dataSource.Parse(context, []byte(content))
|
||||
assert.Equal(t, nil, err)
|
||||
assert.Len(t, resp.ExchangeRates, 0)
|
||||
}
|
||||
+18
-16
@@ -128,22 +128,23 @@ const (
|
||||
|
||||
// Exchange rates data source types
|
||||
const (
|
||||
BankOfCanadaDataSource string = "bank_of_canada"
|
||||
CzechNationalBankDataSource string = "czech_national_bank"
|
||||
DanmarksNationalbankDataSource string = "danmarks_national_bank"
|
||||
EuroCentralBankDataSource string = "euro_central_bank"
|
||||
NationalBankOfGeorgiaDataSource string = "national_bank_of_georgia"
|
||||
CentralBankOfHungaryDataSource string = "central_bank_of_hungary"
|
||||
BankOfIsraelDataSource string = "bank_of_israel"
|
||||
CentralBankOfMyanmarDataSource string = "central_bank_of_myanmar"
|
||||
NorgesBankDataSource string = "norges_bank"
|
||||
NationalBankOfPolandDataSource string = "national_bank_of_poland"
|
||||
NationalBankOfRomaniaDataSource string = "national_bank_of_romania"
|
||||
BankOfRussiaDataSource string = "bank_of_russia"
|
||||
SwissNationalBankDataSource string = "swiss_national_bank"
|
||||
NationalBankOfUkraineDataSource string = "national_bank_of_ukraine"
|
||||
CentralBankOfUzbekistanDataSource string = "central_bank_of_uzbekistan"
|
||||
UserCustomExchangeRatesDataSource string = "user_custom"
|
||||
BankOfCanadaDataSource string = "bank_of_canada"
|
||||
CzechNationalBankDataSource string = "czech_national_bank"
|
||||
DanmarksNationalbankDataSource string = "danmarks_national_bank"
|
||||
EuroCentralBankDataSource string = "euro_central_bank"
|
||||
NationalBankOfGeorgiaDataSource string = "national_bank_of_georgia"
|
||||
CentralBankOfHungaryDataSource string = "central_bank_of_hungary"
|
||||
BankOfIsraelDataSource string = "bank_of_israel"
|
||||
NationalBankOfKazakhstanDataSource string = "national_bank_of_kazakhstan"
|
||||
CentralBankOfMyanmarDataSource string = "central_bank_of_myanmar"
|
||||
NorgesBankDataSource string = "norges_bank"
|
||||
NationalBankOfPolandDataSource string = "national_bank_of_poland"
|
||||
NationalBankOfRomaniaDataSource string = "national_bank_of_romania"
|
||||
BankOfRussiaDataSource string = "bank_of_russia"
|
||||
SwissNationalBankDataSource string = "swiss_national_bank"
|
||||
NationalBankOfUkraineDataSource string = "national_bank_of_ukraine"
|
||||
CentralBankOfUzbekistanDataSource string = "central_bank_of_uzbekistan"
|
||||
UserCustomExchangeRatesDataSource string = "user_custom"
|
||||
)
|
||||
|
||||
const (
|
||||
@@ -1196,6 +1197,7 @@ func loadExchangeRatesConfiguration(config *Config, configFile *ini.File, sectio
|
||||
dataSource == NationalBankOfGeorgiaDataSource ||
|
||||
dataSource == CentralBankOfHungaryDataSource ||
|
||||
dataSource == BankOfIsraelDataSource ||
|
||||
dataSource == NationalBankOfKazakhstanDataSource ||
|
||||
dataSource == CentralBankOfMyanmarDataSource ||
|
||||
dataSource == NorgesBankDataSource ||
|
||||
dataSource == NationalBankOfPolandDataSource ||
|
||||
|
||||
Reference in New Issue
Block a user