Source code for econuy.retrieval.prices

import datetime as dt
import warnings
from urllib.error import URLError, HTTPError
from io import BytesIO
from pathlib import Path
from zipfile import BadZipFile
from typing import Optional

import numpy as np
import pandas as pd
import requests
from opnieuw import retry
from pandas.tseries.offsets import MonthEnd
from scipy.stats import stats
from scipy.stats.mstats_basic import winsorize

from econuy import transform
from econuy.core import Pipeline
from econuy.utils import metadata, get_project_root
from econuy.utils.sources import urls
from econuy.utils.extras import cpi_details


[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=4, retry_window_after_first_call_in_seconds=60, ) def cpi() -> pd.DataFrame: """Get CPI data. Returns ------- Monthly CPI : pd.DataFrame """ name = "cpi" try: cpi = pd.read_excel( urls[name]["dl"]["main"], skiprows=7, usecols="A:B", index_col=0 ).dropna() except URLError as err: if "SSL: CERTIFICATE_VERIFY_FAILED" in str(err): certificate = Path(get_project_root(), "utils", "files", "ine_certs.pem") r = requests.get(urls[name]["dl"]["main"], verify=certificate) cpi = pd.read_excel( BytesIO(r.content), skiprows=7, usecols="A:B", index_col=0 ).dropna() else: raise err cpi.columns = ["Índice de precios al consumo"] cpi.rename_axis(None, inplace=True) cpi.index = cpi.index + MonthEnd(1) cpi = cpi.apply(pd.to_numeric, errors="coerce") cpi.rename_axis(None, inplace=True) metadata._set( cpi, area="Precios", currency="-", inf_adj="No", unit="2010-10=100", seas_adj="NSA", ts_type="-", cumperiods=1, ) return cpi
[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=4, retry_window_after_first_call_in_seconds=60, ) def cpi_divisions() -> pd.DataFrame: """Get CPI data by division. Returns ------- Monthly CPI by division : pd.DataFrame """ name = "cpi_divisions" raw = [] with pd.ExcelFile(urls[name]["dl"]["main"]) as excel: for sheet in excel.sheet_names: data = ( pd.read_excel(excel, sheet_name=sheet, skiprows=8, nrows=18) .dropna(how="all") .dropna(how="all", axis=1) ) data = data.loc[:, data.columns.str.contains("Índice")].dropna(how="all").T data.columns = [ "Índice General", "Alimentos y Bebidas No Alcohólicas", "Bebidas Alcohólicas, Tabaco y Estupefacientes", "Prendas de Vestir y Calzado", "Vivienda", "Muebles, Artículos para el Hogar y para la Conservación Ordinaria del Hogar", "Salud", "Transporte", "Comunicaciones", "Recreación y Cultura", "Educación", "Restaurantes y Hoteles", "Bienes y Servicios Diversos", ] raw.append(data) output = pd.concat(raw) output.index = pd.date_range(start="2011-01-31", freq="M", periods=len(output)) output.rename_axis(None, inplace=True) prev = pd.read_csv(urls[name]["dl"]["1997"], index_col=0, parse_dates=True) output = pd.concat([prev, output], axis=0) output = output.loc[~output.index.duplicated(keep="last"), :] output = output.apply(pd.to_numeric, errors="coerce") metadata._set( output, area="Precios", currency="-", inf_adj="No", unit="2010-12=100", seas_adj="NSA", ts_type="-", cumperiods=1, ) return output
[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=4, retry_window_after_first_call_in_seconds=60, ) def cpi_classes() -> pd.DataFrame: """Get CPI data by division, group and class. Returns ------- Monthly CPI by division, group and class : pd.DataFrame """ name = "cpi_classes" raw = [] nmonths = 0 with pd.ExcelFile(urls[name]["dl"]["main"]) as excel: for sheet in excel.sheet_names: data = ( pd.read_excel(excel, sheet_name=sheet, skiprows=8, nrows=147) .dropna(how="all") .dropna(how="all", axis=1) ) data["Unnamed: 1"] = np.where( data["Unnamed: 0"].str.len() == 2, "Div_" + data["Unnamed: 1"], np.where( data["Unnamed: 0"].str.len() == 3, "Gru_" + data["Unnamed: 1"], "Cls_" + data["Unnamed: 1"], ), ) data = data.iloc[:, 1:].T data.columns = data.iloc[0] data = data.iloc[1:] data = data.loc[data.index.str.contains("Índice")].reset_index(drop=True).iloc[:, 1:] data.rename(columns={"Cls_Índice General": "Índice General"}, inplace=True) data.index = range(nmonths, nmonths + len(data)) data.rename( columns={ "Div_Bebidas Alcoholicas, Tabaco y Estupefacientes": "Div_Bebidas Alcohólicas, Tabaco y Estupefacientes", "Div_Muebles, Artículos Para el Hogar y Para la Coservación Oridnaria del Hogar": "Div_Muebles, Artículos Para el Hogar y Para la Conservación Ordinaria del Hogar", "Gru_Enseñanza no atribuíble a ningún nivel": "Gru_Enseñanza no atribuible a ningún nivel", "Cls_Enseñanza no atribuíble a ningún nivel": "Cls_Enseñanza no atribuible a ningún nivel", "Cls_Otros aparatos, articulos y productos para la atención personal": "Cls_Otros aparatos, artículos y productos para la atención personal", }, inplace=True, ) raw.append(data) nmonths = nmonths + len(data) output = pd.concat(raw) output.index = pd.date_range("2011-01-31", freq="M", periods=len(output)) output.rename_axis(None, inplace=True) output = output.apply(pd.to_numeric, errors="coerce") metadata._set( output, area="Precios", currency="-", inf_adj="No", unit="2010-12=100", seas_adj="NSA", ts_type="-", cumperiods=1, ) return output
[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=4, retry_window_after_first_call_in_seconds=60, ) def inflation_expectations() -> pd.DataFrame: """Get data for the BCU inflation expectations survey. Returns ------- Monthly inflation expectations : pd.DataFrame """ name = "inflation_expectations" raw = ( pd.read_excel(urls[name]["dl"]["main"], skiprows=9) .dropna(how="all", axis=1) .dropna(thresh=4) ) mask = raw.iloc[-12:].isna().all() output = raw.loc[:, ~mask] output.columns = ( ["Fecha"] + ["Inflación mensual del mes corriente"] * 5 + ["Inflación para los próximos 6 meses"] * 5 + ["Inflación anual del año calendario corriente"] * 5 + ["Inflación anual de los próximos 12 meses"] * 5 + ["Inflación anual del año calendario siguiente"] * 5 + ["Inflación anual de los próximos 24 meses"] * 5 + ["Inflación anual a dos años calendario"] * 5 ) output.set_index("Fecha", inplace=True, drop=True) output.columns = output.columns + " - " + output.iloc[0] output = output.iloc[1:] output.rename_axis(None, inplace=True) output.index = pd.date_range(start="2004-01-31", freq="M", periods=len(output)) output = output.apply(pd.to_numeric, errors="coerce") metadata._set( output, area="Precios", currency="-", inf_adj="No", unit="%", seas_adj="NSA", ts_type="-", cumperiods=1, ) return output
[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=4, retry_window_after_first_call_in_seconds=60, ) def utilities() -> pd.DataFrame: """Get prices for government-owned utilities. Returns ------- Monthly utilities prices : pd.DataFrame """ name = "utilities" cpi_1997 = pd.read_excel(urls[name]["dl"]["1997"], skiprows=5, nrows=529).dropna() products = [ " Electricidad", " Supergas", " Agua corriente", " Combustibles Liquidos", " Servicio Telefonico", ] cpi_1997 = cpi_1997.loc[ cpi_1997["Rubros, Agrupaciones, Subrubros, Familias y Artículos"].isin(products) ] cpi_1997 = cpi_1997.T.iloc[1:] cpi_1997.columns = [ "Electricidad", "Supergás", "Combustibles líquidos", "Agua corriente", "Servicio telefónico", ] cpi_1997.index = pd.date_range(start="1997-03-31", freq="M", periods=len(cpi_1997)) cpi_1997.rename_axis(None, inplace=True) with pd.ExcelFile(urls[name]["dl"]["2019"]) as excel: cpi_2010 = [] for sheet in excel.sheet_names: data = pd.read_excel(excel, sheet_name=sheet, skiprows=8, nrows=640).dropna(how="all") products = [ "04510010", "04520020", "07221", "04410010", "08300010", "08300020", "08300030", ] data = data.loc[ data["Unnamed: 0"].isin(products), data.columns.str.contains("Indice|Índice", regex=True), ].T data.columns = [ "Agua corriente", "Electricidad", "Supergás", "Combustibles líquidos", "Servicio de telefonía fija", "Servicio de telefonía móvil", "Servicio de Internet", ] cpi_2010.append(data) cpi_2010 = pd.concat(cpi_2010) cpi_2010.index = pd.date_range(start="2010-12-31", freq="M", periods=len(cpi_2010)) output = pd.concat([cpi_1997 / cpi_1997.iloc[-1] * 100, cpi_2010]) output = output.loc[~output.index.duplicated(keep="last")] output.at[pd.Timestamp(2010, 12, 31), "Servicio telefónico"] = 100 metadata._set( output, area="Precios", currency="-", inf_adj="No", unit="2010-12=100", seas_adj="NSA", ts_type="-", cumperiods=1, ) return output
[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=4, retry_window_after_first_call_in_seconds=30, ) def ppi() -> pd.DataFrame: """Get PPI data. Returns ------- Monthly PPI : pd.DataFrame """ name = "ppi" raw = pd.read_excel(urls[name]["dl"]["main"], skiprows=7, index_col=0).dropna() raw.columns = [ "Índice general", "Ganadería, agricultura y silvicultura", "Pesca", "Explotación de minas y canteras", "Industrias manufactureras", ] raw.rename_axis(None, inplace=True) raw.index = raw.index + MonthEnd(1) output = raw.apply(pd.to_numeric, errors="coerce") metadata._set( output, area="Precios", currency="-", inf_adj="No", unit="2010-03=100", seas_adj="NSA", ts_type="-", cumperiods=1, ) return output
[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=4, retry_window_after_first_call_in_seconds=30, ) def nxr_monthly() -> pd.DataFrame: """Get monthly nominal exchange rate data. Returns ------- Monthly nominal exchange rates : pd.DataFrame Sell rate, monthly average and end of period. """ name = "nxr_monthly" try: nxr_raw = pd.read_excel(urls[name]["dl"]["main"], skiprows=4, index_col=0, usecols="A,C,F") except URLError as err: if "SSL: CERTIFICATE_VERIFY_FAILED" in str(err): certificate = Path(get_project_root(), "utils", "files", "ine_certs.pem") r = requests.get(urls[name]["dl"]["main"], verify=certificate) nxr_raw = pd.read_excel(BytesIO(r.content), skiprows=4, index_col=0, usecols="A,C,F") else: raise err nxr = nxr_raw.dropna(how="any", axis=0) nxr.columns = ["Tipo de cambio venta, fin de período", "Tipo de cambio venta, promedio"] nxr.index = nxr.index + MonthEnd(1) nxr = nxr.apply(pd.to_numeric, errors="coerce") nxr.rename_axis(None, inplace=True) metadata._set( nxr, area="Precios", currency="UYU/USD", inf_adj="No", unit="-", seas_adj="NSA", ts_type="-", cumperiods=1, ) return nxr
[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=10, retry_window_after_first_call_in_seconds=60, ) def nxr_daily( pipeline: Optional[Pipeline] = None, previous_data: pd.DataFrame = pd.DataFrame() ) -> pd.DataFrame: """Get daily nominal exchange rate data. Parameters ---------- pipeline : econuy.core.Pipeline or None, default None An instance of the econuy Pipeline class. previous_data : pd.DataFrame A DataFrame representing this dataset used to extract last available dates. Returns ------- Monthly nominal exchange rates : pd.DataFrame Sell rate, monthly average and end of period. """ if pipeline is not None: pipeline = Pipeline() if previous_data.empty: start_date = dt.datetime(1999, 12, 31) else: start_date = previous_data.index[-1] today = dt.datetime.now() - dt.timedelta(days=1) runs = (today - start_date).days // 360 data = [] base_url = urls["nxr_daily"]["dl"]["main"] if runs > 0: for i in range(1, runs + 1): from_ = (start_date + dt.timedelta(days=1)).strftime("%d/%m/%Y") to_ = (start_date + dt.timedelta(days=360)).strftime("%d/%m/%Y") dates = f"%22FechaDesde%22:%22{from_}%22,%22FechaHasta%22:%22{to_}" url = f"{base_url}{dates}%22,%22Grupo%22:%222%22}}" + "}" try: data.append(pd.read_excel(url)) start_date = dt.datetime.strptime(to_, "%d/%m/%Y") except (TypeError, BadZipFile): pass from_ = (start_date + dt.timedelta(days=1)).strftime("%d/%m/%Y") to_ = (dt.datetime.now() - dt.timedelta(days=1)).strftime("%d/%m/%Y") dates = f"%22FechaDesde%22:%22{from_}%22,%22FechaHasta%22:%22{to_}" url = f"{base_url}{dates}%22,%22Grupo%22:%222%22}}" + "}" try: data.append(pd.read_excel(url)) except ValueError as e: if "File is not a recognized excel file" in str(e): pass try: output = pd.concat(data, axis=0) output = output.pivot(index="Fecha", columns="Moneda", values="Venta").rename_axis(None) output.index = pd.to_datetime(output.index, format="%d/%m/%Y", errors="coerce") output.sort_index(inplace=True) output.replace(",", ".", regex=True, inplace=True) output.columns = ["Tipo de cambio US$, Cable"] output = output.apply(pd.to_numeric, errors="coerce") metadata._set( output, area="Precios", currency="UYU/USD", inf_adj="No", unit="-", seas_adj="NSA", ts_type="-", cumperiods=1, ) output.columns = output.columns.set_levels(["-"], level=2) output.rename_axis(None, inplace=True) except ValueError as e: if str(e) == "No objects to concatenate": return previous_data return output
# The `_contains_nan` function needs to be monkey-patched to avoid an error # when checking whether a Series is True def _new_contains_nan(a, nan_policy="propagate"): policies = ["propagate", "raise", "omit"] if nan_policy not in policies: raise ValueError( "nan_policy must be one of {%s}" % ", ".join("'%s'" % s for s in policies) ) try: with np.errstate(invalid="ignore"): # This [0] gets the value instead of the array, fixing the error contains_nan = np.isnan(np.sum(a))[0] except TypeError: try: contains_nan = np.nan in set(a.ravel()) except TypeError: contains_nan = False nan_policy = "omit" warnings.warn( "The input array could not be properly checked for " "nan values. nan values will be ignored.", RuntimeWarning, ) if contains_nan and nan_policy == "raise": raise ValueError("The input contains nan values") return contains_nan, nan_policy stats._contains_nan = _new_contains_nan
[docs]@retry( retry_on_exceptions=(HTTPError, URLError), max_calls_total=4, retry_window_after_first_call_in_seconds=60, ) def cpi_measures(pipeline: Optional[Pipeline] = None) -> pd.DataFrame: """ Get core CPI, Winsorized CPI, tradabe CPI, non-tradable CPI and residual CPI. Parameters ---------- pipeline : econuy.core.Pipeline or None, default None An instance of the econuy Pipeline class. Returns ------- Monthly CPI measures : pd.DataFrame """ if pipeline is None: pipeline = Pipeline() name = "cpi_measures" xls_10 = pd.ExcelFile(urls[name]["dl"]["2010-"]) prod_97 = ( pd.read_excel(urls[name]["dl"]["1997"], skiprows=5) .dropna(how="any") .set_index("Rubros, Agrupaciones, Subrubros, Familias y Artículos") .T ) weights_97 = pd.read_excel(urls[name]["dl"]["1997_weights"], index_col=0).drop_duplicates( subset="Descripción", keep="first" ) weights = pd.read_excel(xls_10, usecols="A:C", skiprows=13, index_col=0).dropna(how="any") weights.columns = ["Item", "Weight"] weights_8 = weights.loc[weights.index.str.len() == 8] sheets = [] for sheet in xls_10.sheet_names: raw = pd.read_excel(xls_10, sheet_name=sheet, usecols="D:IN", skiprows=8).dropna(how="all") proc = raw.loc[:, raw.columns.str.contains("Indice|Índice")].dropna(how="all") sheets.append(proc.T) complete_10 = pd.concat(sheets) complete_10 = complete_10.iloc[:, 1:] complete_10.columns = [weights["Item"], weights.index] complete_10.index = pd.date_range(start="2010-12-31", periods=len(complete_10), freq="M") diff_8 = complete_10.loc[ :, complete_10.columns.get_level_values(level=1).str.len() == 8 ].pct_change() win = pd.DataFrame(winsorize(diff_8, limits=(0.05, 0.05), axis=1)) win.index = diff_8.index win.columns = diff_8.columns.get_level_values(level=1) cpi_win = win.mul(weights_8.loc[:, "Weight"].T) cpi_win = cpi_win.sum(axis=1).add(1).cumprod().mul(100) weights_97["Weight"] = ( weights_97["Rubro"] .fillna(weights_97["Agrupación, subrubro, familia"]) .fillna(weights_97["Artículo"]) .drop(columns=["Rubro", "Agrupación, subrubro, familia", "Artículo"]) ) prod_97 = prod_97.loc[:, list(cpi_details["1997_base"].keys())] prod_97.index = pd.date_range(start="1997-03-31", periods=len(prod_97), freq="M") weights_97 = ( weights_97[weights_97["Descripción"].isin(cpi_details["1997_weights"])] .set_index("Descripción") .drop(columns=["Rubro", "Agrupación, subrubro, " "familia", "Artículo"]) ).div(100) weights_97.index = prod_97.columns prod_10 = complete_10.loc[:, list(cpi_details["2010_base"].keys())] prod_10 = prod_10.loc[:, ~prod_10.columns.get_level_values(level=0).duplicated()] prod_10.columns = prod_10.columns.get_level_values(level=0) weights_10 = ( weights.loc[weights["Item"].isin(list(cpi_details["2010_base"].keys()))].drop_duplicates( subset="Item", keep="first" ) ).set_index("Item") items = [] weights = [] for item, weight, details in zip( [prod_10, prod_97], [weights_10, weights_97], ["2010_base", "1997_base"] ): for tradable in [True, False]: items.append( item.loc[ :, [k for k, v in cpi_details[details].items() if v["Tradable"] is tradable] ] ) aux = weight.loc[ [k for k, v in cpi_details[details].items() if v["Tradable"] is tradable] ] weights.append(aux.div(aux.sum())) for core in [True, False]: items.append( item.loc[:, [k for k, v in cpi_details[details].items() if v["Core"] is core]] ) aux = weight.loc[[k for k, v in cpi_details[details].items() if v["Core"] is core]] weights.append(aux.div(aux.sum())) intermediate = [] for item, weight in zip(items, weights): intermediate.append(item.mul(weight.squeeze()).sum(1)) output = [] for x, y in zip(intermediate[:4], intermediate[4:]): aux = pd.concat( [ y.pct_change().loc[y.index < "2011-01-01"], x.pct_change().loc[x.index > "2011-01-01"], ] ) output.append(aux.fillna(0).add(1).cumprod().mul(100)) pipeline.get("cpi") cpi_re = pipeline.dataset cpi_re = cpi_re.loc[cpi_re.index >= "1997-03-31"] output = pd.concat([cpi_re] + output + [cpi_win], axis=1) output.columns = [ "Índice de precios al consumo: total", "Índice de precios al consumo: transables", "Índice de precios al consumo: no transables", "Índice de precios al consumo: subyacente", "Índice de precios al consumo: residual", "Índice de precios al consumo: Winsorized 0.05", ] output = output.apply(pd.to_numeric, errors="coerce") metadata._set( output, area="Precios", currency="-", inf_adj="No", unit="2010-12=100", seas_adj="NSA", ts_type="-", cumperiods=1, ) output = transform.rebase(output, start_date="2010-12-01", end_date="2010-12-31") output.rename_axis(None, inplace=True) return output