ESCatastroLib 0.0.1rc0__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- escatastrolib-0.0.1rc0/ESCatastroLib/__init__.py +3 -0
- escatastrolib-0.0.1rc0/ESCatastroLib/models/Calle.py +49 -0
- escatastrolib-0.0.1rc0/ESCatastroLib/models/InfoCatastral.py +303 -0
- escatastrolib-0.0.1rc0/ESCatastroLib/models/Municipio.py +29 -0
- escatastrolib-0.0.1rc0/ESCatastroLib/models/__init__.py +4 -0
- escatastrolib-0.0.1rc0/ESCatastroLib/utils/__init__.py +4 -0
- escatastrolib-0.0.1rc0/ESCatastroLib/utils/exceptions.py +21 -0
- escatastrolib-0.0.1rc0/ESCatastroLib/utils/statics.py +122 -0
- escatastrolib-0.0.1rc0/ESCatastroLib/utils/utils.py +102 -0
- escatastrolib-0.0.1rc0/ESCatastroLib.egg-info/PKG-INFO +288 -0
- escatastrolib-0.0.1rc0/ESCatastroLib.egg-info/SOURCES.txt +22 -0
- escatastrolib-0.0.1rc0/ESCatastroLib.egg-info/dependency_links.txt +1 -0
- escatastrolib-0.0.1rc0/ESCatastroLib.egg-info/requires.txt +2 -0
- escatastrolib-0.0.1rc0/ESCatastroLib.egg-info/top_level.txt +1 -0
- escatastrolib-0.0.1rc0/LICENSE +201 -0
- escatastrolib-0.0.1rc0/PKG-INFO +288 -0
- escatastrolib-0.0.1rc0/README.md +262 -0
- escatastrolib-0.0.1rc0/setup.cfg +4 -0
- escatastrolib-0.0.1rc0/setup.py +36 -0
- escatastrolib-0.0.1rc0/tests/test_calle.py +18 -0
- escatastrolib-0.0.1rc0/tests/test_import.py +8 -0
- escatastrolib-0.0.1rc0/tests/test_info_catastral.py +36 -0
- escatastrolib-0.0.1rc0/tests/test_municipio.py +15 -0
- escatastrolib-0.0.1rc0/tests/test_utils.py +20 -0
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
from .Municipio import Municipio
|
|
2
|
+
from ..utils.statics import URL_BASE_CALLEJERO
|
|
3
|
+
from ..utils.utils import comprobar_errores
|
|
4
|
+
|
|
5
|
+
import requests
|
|
6
|
+
import json
|
|
7
|
+
|
|
8
|
+
class Calle:
|
|
9
|
+
"""
|
|
10
|
+
Clase que representa una calle.
|
|
11
|
+
Args:
|
|
12
|
+
municipio (Municipio): Objeto que representa el municipio al que pertenece la calle.
|
|
13
|
+
nombre_calle (str, optional): Nombre de la calle. Default es None.
|
|
14
|
+
tipo_via (str, optional): Tipo de vía de la calle. Default es None.
|
|
15
|
+
Raises:
|
|
16
|
+
Exception: Se produce cuando ocurre un error al asignar la calle.
|
|
17
|
+
Exception: Se produce cuando no se especifica el nombre de la calle.
|
|
18
|
+
Exception: Se produce cuando no se encuentran calles disponibles.
|
|
19
|
+
Attributes:
|
|
20
|
+
calle (str): Nombre de la calle asignada.
|
|
21
|
+
tipo_via (str): Tipo de vía de la calle asignada.
|
|
22
|
+
municipio (Municipio): Objeto que representa el municipio al que pertenece la calle.
|
|
23
|
+
"""
|
|
24
|
+
|
|
25
|
+
def __init__(self, municipio: Municipio, nombre_calle: str|None = None, tipo_via: str|None = None):
|
|
26
|
+
uri = f'{URL_BASE_CALLEJERO}/ObtenerCallejero'
|
|
27
|
+
|
|
28
|
+
request = requests.get(uri, params={
|
|
29
|
+
'Provincia': municipio.provincia,
|
|
30
|
+
'Municipio': municipio.municipio,
|
|
31
|
+
'NomVia': nombre_calle,
|
|
32
|
+
'TipoVia': tipo_via
|
|
33
|
+
})
|
|
34
|
+
|
|
35
|
+
if request.status_code == 200:
|
|
36
|
+
calle_dict = json.loads(request.text)
|
|
37
|
+
|
|
38
|
+
if comprobar_errores(calle_dict) and nombre_calle:
|
|
39
|
+
if len(calle_dict.get('consulta_callejeroResult').get('callejero').get('calle')) == 1:
|
|
40
|
+
self.calle = calle_dict.get('consulta_callejeroResult').get('callejero').get('calle')[0].get('dir').get('nv')
|
|
41
|
+
self.tipo_via = calle_dict.get('consulta_callejeroResult').get('callejero').get('calle')[0].get('dir').get('tv')
|
|
42
|
+
self.municipio = municipio
|
|
43
|
+
else:
|
|
44
|
+
raise Exception(f"""Error al asignar calle. No has especificado la calle completamente, o ese nombre está en varios lados (indica el tipo de vía que aparezca al principio de la vía como PZ o CL). Estas son las calles disponibles con esa búsqueda: {','.join(f"{cl.get('dir').get('tv')} {cl.get('dir').get('nv')}" for cl in calle_dict.get('consulta_callejeroResult').get('callejero').get('calle'))}""")
|
|
45
|
+
else:
|
|
46
|
+
raise Exception(f"""Error al asignar calle. No has especificado la calle. Estas son las calles disponibles: {','.join(f"{cl.get('dir').get('tv')} {cl.get('dir').get('nv')}" for cl in calle_dict.get('consulta_callejeroResult').get('callejero').get('calle'))}""")
|
|
47
|
+
|
|
48
|
+
else:
|
|
49
|
+
raise Exception(f'Error al listar calles: Puede que no exista ese municipio.')
|
|
@@ -0,0 +1,303 @@
|
|
|
1
|
+
import requests
|
|
2
|
+
import json
|
|
3
|
+
import xmltodict
|
|
4
|
+
|
|
5
|
+
from ..utils.statics import URL_BASE_CALLEJERO, URL_BASE_GEOGRAFIA, URL_BASE_CROQUIS_DATOS
|
|
6
|
+
from ..utils.utils import comprobar_errores
|
|
7
|
+
from ..utils.exceptions import ErrorServidorCatastro
|
|
8
|
+
from .Calle import Calle, Municipio
|
|
9
|
+
|
|
10
|
+
class ParcelaCatastral:
|
|
11
|
+
"""
|
|
12
|
+
Clase que representa una parcela catastral.
|
|
13
|
+
Args:
|
|
14
|
+
rc (str, optional): La referencia catastral de la parcela. Defaults to None. Puede ir solo.
|
|
15
|
+
|
|
16
|
+
provincia (int|str, optional): El código o nombre de la provincia. Defaults to None. Se usa para buscar por dirección o parcela.
|
|
17
|
+
municipio (int|str, optional): El código o nombre del municipio. Defaults to None. Se usa para buscar por dirección o parcela.
|
|
18
|
+
poligono (int, optional): El número del polígono. Defaults to None. Se usa para buscar por parcela.
|
|
19
|
+
parcela (int, optional): El número de la parcela. Defaults to None. Se usa para buscar por parcela.
|
|
20
|
+
tipo_via (str, optional): El tipo de vía de la dirección. Defaults to None. Se usa para buscar por dirección.
|
|
21
|
+
calle (str, optional): El nombre de la calle de la dirección. Defaults to None. Se usa para buscar por dirección.
|
|
22
|
+
numero (str, optional): El número de la dirección. Defaults to None. Se usa para buscar por dirección.
|
|
23
|
+
Raises:
|
|
24
|
+
ValueError: Se lanza si no se proporciona suficiente información para realizar la búsqueda o si la RC corresponde a una MetaParcela.
|
|
25
|
+
ErrorServidorCatastro: Se lanza si hay un error en el servidor del Catastro.
|
|
26
|
+
Attributes:
|
|
27
|
+
rc (str): La referencia catastral de la parcela.
|
|
28
|
+
provincia (int|str): El código o nombre de la provincia.
|
|
29
|
+
municipio (int|str): El código o nombre del municipio.
|
|
30
|
+
poligono (int): El número del polígono. Sólo se da en terrenos Rústicos.
|
|
31
|
+
parcela (int): El número de la parcela. Sólo se da en terrenos Rústicos.
|
|
32
|
+
tipo_via (str): El tipo de vía de la dirección. Sólo se da en terrenos Urbanos.
|
|
33
|
+
calle (str): El nombre de la calle de la dirección. Sólo se da en terrenos Urbanos.
|
|
34
|
+
numero (str): El número de la dirección. Sólo se da en terrenos Urbanos.
|
|
35
|
+
url_croquis (str): La URL del croquis de la parcela.
|
|
36
|
+
tipo (str): El tipo de la parcela (Urbano o Rústico).
|
|
37
|
+
antiguedad (str): La antigüedad de la parcela (solo para parcelas urbanas).
|
|
38
|
+
uso (str): El uso de la parcela (solo para parcelas urbanas).
|
|
39
|
+
nombre_paraje (str): El nombre del paraje (solo para parcelas rústicas).
|
|
40
|
+
regiones (list): Una lista de regiones de la parcela, cada una con una descripción y superficie.
|
|
41
|
+
geocentro (dict): Las coordenadas del geocentro de la parcela.
|
|
42
|
+
geometria (list): Una lista de puntos que representan la geometría de la parcela.
|
|
43
|
+
"""
|
|
44
|
+
|
|
45
|
+
def __create_regions(self, info_cadastre: dict):
|
|
46
|
+
self.regiones = []
|
|
47
|
+
if self.tipo == 'Urbano':
|
|
48
|
+
iterator = list(info_cadastre.values())[0].get('bico').get('lcons')
|
|
49
|
+
elif self.tipo == 'Rústico':
|
|
50
|
+
iterator = list(info_cadastre.values())[0].get('bico').get('lspr')
|
|
51
|
+
for region in iterator:
|
|
52
|
+
if self.tipo == 'Rústico':
|
|
53
|
+
self.regiones.append({
|
|
54
|
+
'descripcion': region.get('dspr').get('dcc'),
|
|
55
|
+
'superficie': region.get('dspr').get('ssp')
|
|
56
|
+
})
|
|
57
|
+
elif self.tipo == 'Urbano':
|
|
58
|
+
self.regiones.append({
|
|
59
|
+
'descripcion': region.get('lcd'),
|
|
60
|
+
'superficie': region.get('dfcons').get('stl')
|
|
61
|
+
})
|
|
62
|
+
|
|
63
|
+
|
|
64
|
+
def __create_geometry(self):
|
|
65
|
+
geometry_request = requests.get(f'{URL_BASE_GEOGRAFIA}',
|
|
66
|
+
params={
|
|
67
|
+
'service':'wfs',
|
|
68
|
+
'version':'2',
|
|
69
|
+
'request':'getfeature',
|
|
70
|
+
'STOREDQUERIE_ID':'GetParcel',
|
|
71
|
+
'refcat': self.rc,
|
|
72
|
+
'srsname': 'EPSG:4326'
|
|
73
|
+
})
|
|
74
|
+
|
|
75
|
+
geometry = xmltodict.parse(geometry_request.content)
|
|
76
|
+
geoposition = geometry.get('FeatureCollection').get('member').get('cp:CadastralParcel').get('cp:referencePoint').get('gml:Point').get('gml:pos').split(' ')
|
|
77
|
+
self.geocentro = {
|
|
78
|
+
'x': geoposition[0],
|
|
79
|
+
'y': geoposition[1]
|
|
80
|
+
}
|
|
81
|
+
parcel_geometry = geometry.get('FeatureCollection').get('member').get('cp:CadastralParcel').get('cp:geometry').get('gml:MultiSurface').get('gml:surfaceMember').get('gml:Surface').get('gml:patches').get('gml:PolygonPatch').get('gml:exterior').get('gml:LinearRing').get('gml:posList').get('#text').split(' ')
|
|
82
|
+
self.geometria = [
|
|
83
|
+
{
|
|
84
|
+
'x': parcel_geometry[2*idx],
|
|
85
|
+
'y': parcel_geometry[2*idx+1]
|
|
86
|
+
} for idx in range(len(parcel_geometry)//2)
|
|
87
|
+
]
|
|
88
|
+
|
|
89
|
+
def __create_from_rc(self, rc: str):
|
|
90
|
+
"""Create an instance of InfoCatastral from a RC (Referencia Catastral) string."""
|
|
91
|
+
req1 = requests.get(f'{URL_BASE_CALLEJERO}/Consulta_DNPRC',
|
|
92
|
+
params={'RefCat': rc})
|
|
93
|
+
|
|
94
|
+
info_cadastre = json.loads(req1.content)
|
|
95
|
+
if comprobar_errores(info_cadastre):
|
|
96
|
+
cudnp = info_cadastre.get("consulta_dnprcResult", {}).get("control", {}).get("cudnp", 1)
|
|
97
|
+
|
|
98
|
+
if cudnp > 1:
|
|
99
|
+
raise ErrorServidorCatastro(mensaje="Esta parcela tiene varias referencias catastrales. Usa un objeto MetaParcela.")
|
|
100
|
+
else:
|
|
101
|
+
self.url_croquis = requests.get(URL_BASE_CROQUIS_DATOS, params={'refcat': self.rc}).url
|
|
102
|
+
self.municipio = info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('dt').get('nm')
|
|
103
|
+
self.provincia = info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('dt').get('np')
|
|
104
|
+
self.tipo = 'Rústico' if info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('idbi').get('cn') == 'RU' else 'Urbano'
|
|
105
|
+
if self.tipo == 'Urbano':
|
|
106
|
+
self.calle = f"{info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('dt').get('locs').get('lous').get('lourb').get('dir').get('tv')} {info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('dt').get('locs').get('lous').get('lourb').get('dir').get('nv')}"
|
|
107
|
+
self.numero = info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('dt').get('locs').get('lous').get('lourb').get('dir').get('pnp')
|
|
108
|
+
self.antiguedad = info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('debi').get('ant')
|
|
109
|
+
self.uso = info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('debi').get('luso')
|
|
110
|
+
elif self.tipo == 'Rústico':
|
|
111
|
+
self.parcela = info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('dt').get('locs').get('lors').get('lorus').get('cpp').get('cpa')
|
|
112
|
+
self.poligono = info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('dt').get('locs').get('lors').get('lorus').get('cpp').get('cpo')
|
|
113
|
+
self.nombre_paraje = info_cadastre.get('consulta_dnprcResult').get('bico').get('bi').get('dt').get('locs').get('lors').get('lorus').get('npa')
|
|
114
|
+
|
|
115
|
+
self.__create_regions(info_cadastre)
|
|
116
|
+
self.__create_geometry()
|
|
117
|
+
|
|
118
|
+
self.superficie = sum(float(region.get('superficie')) for region in self.regiones)
|
|
119
|
+
|
|
120
|
+
def __create_from_parcel(self, provincia: str|None, municipio: str|None, poligono: str|None, parcela: str|None):
|
|
121
|
+
"""Create an instance of InfoCatastral from a parcela string."""
|
|
122
|
+
req = requests.get(f'{URL_BASE_CALLEJERO}/Consulta_DNPPP',
|
|
123
|
+
params={
|
|
124
|
+
'Provincia': provincia,
|
|
125
|
+
'Municipio': municipio,
|
|
126
|
+
'Poligono': poligono,
|
|
127
|
+
'Parcela': parcela
|
|
128
|
+
})
|
|
129
|
+
|
|
130
|
+
info_cadastre = json.loads(req.content)
|
|
131
|
+
if comprobar_errores(info_cadastre):
|
|
132
|
+
cudnp = info_cadastre.get("consulta_dnpppResult", {}).get("control", {}).get("cudnp", 1)
|
|
133
|
+
|
|
134
|
+
if cudnp > 1:
|
|
135
|
+
raise ErrorServidorCatastro(mensaje="Esta parcela tiene varias referencias catastrales. Usa un objeto MetaParcela.")
|
|
136
|
+
else:
|
|
137
|
+
self.rc = ''.join(info_cadastre.get('consulta_dnpppResult').get('bico').get('bi').get('idbi').get('rc').values())
|
|
138
|
+
self.__create_from_rc(self.rc)
|
|
139
|
+
|
|
140
|
+
def __create_from_address(self, provincia: str|None, municipio: str|None, tipo_via: str|None, calle: str|None, numero: str|None):
|
|
141
|
+
"""Create an instance of InfoCatastral from an address string."""
|
|
142
|
+
info_calle = Calle(
|
|
143
|
+
municipio=Municipio(
|
|
144
|
+
provincia=provincia,
|
|
145
|
+
municipio=municipio
|
|
146
|
+
),
|
|
147
|
+
tipo_via=tipo_via,
|
|
148
|
+
nombre_calle=calle
|
|
149
|
+
)
|
|
150
|
+
|
|
151
|
+
if info_calle:
|
|
152
|
+
req = requests.get(f'{URL_BASE_CALLEJERO}/Consulta_DNPLOC',
|
|
153
|
+
params={
|
|
154
|
+
'Provincia': info_calle.municipio.provincia,
|
|
155
|
+
'Municipio': info_calle.municipio.municipio,
|
|
156
|
+
'Sigla': info_calle.tipo_via,
|
|
157
|
+
'Calle': info_calle.calle,
|
|
158
|
+
'Numero': numero
|
|
159
|
+
})
|
|
160
|
+
|
|
161
|
+
if req.status_code == 200 and comprobar_errores(req.json()):
|
|
162
|
+
info_cadastre = json.loads(req.content)
|
|
163
|
+
cudnp = info_cadastre.get("consulta_dnplocResult", {}).get("control", {}).get("cudnp", 1)
|
|
164
|
+
|
|
165
|
+
if cudnp > 1:
|
|
166
|
+
raise ErrorServidorCatastro(mensaje="Esta parcela tiene varias referencias catastrales. Usa un objeto MetaParcela.")
|
|
167
|
+
else:
|
|
168
|
+
if 'lrcdnp' in info_cadastre.get('consulta_dnplocResult'):
|
|
169
|
+
self.rc = ''.join(info_cadastre.get('consulta_dnplocResult').get('lrcdnp').get('rcdnp')[0].get('rc').values())
|
|
170
|
+
elif 'bico' in info_cadastre.get('consulta_dnplocResult'):
|
|
171
|
+
self.rc = ''.join(info_cadastre.get('consulta_dnplocResult').get('bico').get('bi').get('idbi').get('rc').values())
|
|
172
|
+
self.__create_from_rc(self.rc)
|
|
173
|
+
elif 'lerr' in json.loads(req.content).get('consulta_dnplocResult') and json.loads(req.content)['consulta_dnplocResult']['lerr'][0]['cod'] == '43':
|
|
174
|
+
info_cadastre = json.loads(req.content)
|
|
175
|
+
raise Exception(f"Ese número no existe. Prueba con alguno de estos: {[num.get('num').get('pnp') for num in info_cadastre.get('consulta_dnplocResult').get('numerero').get('nump')]}")
|
|
176
|
+
|
|
177
|
+
|
|
178
|
+
else:
|
|
179
|
+
raise Exception('La calle no existe.')
|
|
180
|
+
|
|
181
|
+
def __init__(self, rc: str|None = None, provincia: int|str|None = None, municipio: int|str|None = None, poligono: int|None = None, parcela: int|None = None, tipo_via: str|None = None, calle: str|None = None, numero: str|None = None):
|
|
182
|
+
if rc:
|
|
183
|
+
self.rc = rc
|
|
184
|
+
self.__create_from_rc(rc)
|
|
185
|
+
elif provincia and municipio and poligono and parcela:
|
|
186
|
+
self.provincia = provincia
|
|
187
|
+
self.municipio = municipio
|
|
188
|
+
self.poligono = poligono
|
|
189
|
+
self.parcela = parcela
|
|
190
|
+
self.__create_from_parcel(provincia, municipio, poligono, parcela)
|
|
191
|
+
elif provincia and municipio and tipo_via and calle and numero:
|
|
192
|
+
self.provincia = provincia
|
|
193
|
+
self.municipio = municipio
|
|
194
|
+
self.calle = calle
|
|
195
|
+
self.numero = numero
|
|
196
|
+
self.__create_from_address(provincia, municipio, tipo_via, calle, numero)
|
|
197
|
+
else:
|
|
198
|
+
raise ValueError("No se ha proporcionado suficiente información para realizar la búsqueda")
|
|
199
|
+
|
|
200
|
+
class MetaParcela:
|
|
201
|
+
"""
|
|
202
|
+
Clase que representa una MetaParcela, es decir, una gran parcela catastral con
|
|
203
|
+
varias referencias catastrales (Parcelas Catastrales más pequeñas).
|
|
204
|
+
|
|
205
|
+
Args:
|
|
206
|
+
rc (str|None): La referencia catastral de la MetaParcela.
|
|
207
|
+
|
|
208
|
+
provincia (int|str|None): El nombre de la provincia donde se encuentra la MetaParcela.
|
|
209
|
+
municipio (int|str|None): El nombre del municipio donde se encuentra la MetaParcela.
|
|
210
|
+
poligono (int|None): El número de polígono de la MetaParcela. Sólo se usa para buscar por parcela.
|
|
211
|
+
parcela (int|None): El número de parcela de la MetaParcela. Sólo se usa para buscar por parcela.
|
|
212
|
+
tipo_via (str|None): El tipo de vía de la dirección de la MetaParcela. Sólo se usa para buscar por dirección.
|
|
213
|
+
calle (str|None): El nombre de la calle de la dirección de la MetaParcela. Sólo se usa para buscar por dirección.
|
|
214
|
+
numero (str|None): El número de la dirección de la MetaParcela. Sólo se usa para buscar por dirección.
|
|
215
|
+
Attributes:
|
|
216
|
+
rc (str): La referencia catastral de la MetaParcela.
|
|
217
|
+
parcelas (list): Una lista de ParcelaCatastral que representan las parcelas que componen la MetaParcela.
|
|
218
|
+
|
|
219
|
+
"""
|
|
220
|
+
|
|
221
|
+
def __create_from_rc(self, rc: str):
|
|
222
|
+
"""Create an instance of InfoCatastral from a RC (Referencia Catastral) string."""
|
|
223
|
+
req1 = requests.get(f'{URL_BASE_CALLEJERO}/Consulta_DNPRC',
|
|
224
|
+
params={'RefCat': rc})
|
|
225
|
+
|
|
226
|
+
info_cadastre = json.loads(req1.content)
|
|
227
|
+
if comprobar_errores(info_cadastre):
|
|
228
|
+
self.parcelas = []
|
|
229
|
+
num_parcelas = info_cadastre.get("consulta_dnprcResult", {}).get("control", {}).get("cudnp", 1)
|
|
230
|
+
for idx in range(num_parcelas):
|
|
231
|
+
rc = ''.join(info_cadastre.get('consulta_dnprcResult').get('lrcdnp').get('rcdnp')[idx].get('rc').values())
|
|
232
|
+
self.parcelas.append(ParcelaCatastral(rc=rc))
|
|
233
|
+
|
|
234
|
+
|
|
235
|
+
def __create_from_parcel(self, provincia: str|None, municipio: str|None, poligono: str|None, parcela: str|None):
|
|
236
|
+
"""Create an instance of InfoCatastral from a parcela string."""
|
|
237
|
+
req = requests.get(f'{URL_BASE_CALLEJERO}/Consulta_DNPPP',
|
|
238
|
+
params={
|
|
239
|
+
'Provincia': provincia,
|
|
240
|
+
'Municipio': municipio,
|
|
241
|
+
'Poligono': poligono,
|
|
242
|
+
'Parcela': parcela
|
|
243
|
+
})
|
|
244
|
+
|
|
245
|
+
info_cadastre = json.loads(req.content)
|
|
246
|
+
if comprobar_errores(info_cadastre):
|
|
247
|
+
self.parcelas = []
|
|
248
|
+
num_parcelas = info_cadastre.get("consulta_dnpppResult", {}).get("control", {}).get("cudnp", 1)
|
|
249
|
+
for idx in range(num_parcelas):
|
|
250
|
+
rc = ''.join(info_cadastre.get('consulta_dnpppResult').get('lrcdnp').get('rcdnp')[idx].get('rc').values())
|
|
251
|
+
self.parcelas.append(ParcelaCatastral(rc=rc))
|
|
252
|
+
|
|
253
|
+
def __create_from_address(self, provincia: str|None, municipio: str|None, tipo_via: str|None, calle: str|None, numero: str|None):
|
|
254
|
+
"""Create an instance of InfoCatastral from an address string."""
|
|
255
|
+
info_calle = Calle(
|
|
256
|
+
municipio=Municipio(
|
|
257
|
+
provincia=provincia,
|
|
258
|
+
municipio=municipio
|
|
259
|
+
),
|
|
260
|
+
tipo_via=tipo_via,
|
|
261
|
+
nombre_calle=calle
|
|
262
|
+
)
|
|
263
|
+
|
|
264
|
+
if info_calle:
|
|
265
|
+
req = requests.get(f'{URL_BASE_CALLEJERO}/Consulta_DNPLOC',
|
|
266
|
+
params={
|
|
267
|
+
'Provincia': info_calle.municipio.provincia,
|
|
268
|
+
'Municipio': info_calle.municipio.municipio,
|
|
269
|
+
'Sigla': info_calle.tipo_via,
|
|
270
|
+
'Calle': info_calle.calle,
|
|
271
|
+
'Numero': numero
|
|
272
|
+
})
|
|
273
|
+
|
|
274
|
+
if req.status_code == 200 and comprobar_errores(req.json()):
|
|
275
|
+
info_cadastre = json.loads(req.content)
|
|
276
|
+
self.parcelas = []
|
|
277
|
+
num_parcelas = info_cadastre.get("consulta_dnplocResult", {}).get("control", {}).get("cudnp", 1)
|
|
278
|
+
for idx in range(num_parcelas):
|
|
279
|
+
rc = ''.join(info_cadastre.get('consulta_dnplocResult').get('lrcdnp').get('rcdnp')[idx].get('rc').values())
|
|
280
|
+
self.parcelas.append(ParcelaCatastral(rc=rc))
|
|
281
|
+
|
|
282
|
+
else:
|
|
283
|
+
raise Exception('La calle no existe.')
|
|
284
|
+
|
|
285
|
+
def __init__(self, rc: str|None = None, provincia: int|str|None = None, municipio: int|str|None = None, poligono: int|None = None, parcela: int|None = None, tipo_via: str|None = None, calle: str|None = None, numero: str|None = None):
|
|
286
|
+
if rc:
|
|
287
|
+
self.rc = rc
|
|
288
|
+
self.__create_from_rc(rc)
|
|
289
|
+
elif provincia and municipio and poligono and parcela:
|
|
290
|
+
self.provincia = provincia
|
|
291
|
+
self.municipio = municipio
|
|
292
|
+
self.poligono = poligono
|
|
293
|
+
self.parcela = parcela
|
|
294
|
+
self.__create_from_parcel(provincia, municipio, poligono, parcela)
|
|
295
|
+
elif provincia and municipio and tipo_via and calle and numero:
|
|
296
|
+
self.provincia = provincia
|
|
297
|
+
self.municipio = municipio
|
|
298
|
+
self.calle = calle
|
|
299
|
+
self.numero = numero
|
|
300
|
+
self.__create_from_address(provincia, municipio, tipo_via, calle, numero)
|
|
301
|
+
else:
|
|
302
|
+
raise ValueError("No se ha proporcionado suficiente información para realizar la búsqueda")
|
|
303
|
+
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
from ..utils.utils import listar_municipios
|
|
2
|
+
|
|
3
|
+
class Municipio:
|
|
4
|
+
"""
|
|
5
|
+
Clase que representa un municipio.
|
|
6
|
+
Args:
|
|
7
|
+
provincia (str): El nombre de la provincia a la que pertenece el municipio.
|
|
8
|
+
municipio (str, optional): El nombre del municipio. Si no se proporciona, se lanzará una excepción con el municipio.
|
|
9
|
+
Attributes:
|
|
10
|
+
provincia (str): El nombre de la provincia a la que pertenece el municipio.
|
|
11
|
+
municipio (str): El nombre del municipio.
|
|
12
|
+
Raises:
|
|
13
|
+
Exception: Si no se especifica el municipio o si el municipio especificado no está disponible o si no se ha especificado completamente (te dará una lista de coincidencias).
|
|
14
|
+
"""
|
|
15
|
+
|
|
16
|
+
def __init__(self, provincia: str, municipio: str|None = None):
|
|
17
|
+
municipios = listar_municipios(provincia=provincia,municipio=municipio)
|
|
18
|
+
|
|
19
|
+
if municipio:
|
|
20
|
+
if len(municipios) == 1:
|
|
21
|
+
self.provincia = provincia
|
|
22
|
+
self.municipio = municipios[0]
|
|
23
|
+
elif municipio.upper() in municipios:
|
|
24
|
+
self.provincia = provincia
|
|
25
|
+
self.municipio = municipio.upper()
|
|
26
|
+
else:
|
|
27
|
+
raise Exception(f"Error al asignar municipio. No has especificado el municipio completamente. Estos son los municipios disponibles con esa búsqueda: {','.join(municipios)}")
|
|
28
|
+
else:
|
|
29
|
+
raise Exception(f"Error al asignar municipio. No has especificado el municipio. Estos son los municipios disponibles: {','.join(municipios)}")
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
class ErrorServidorCatastro(Exception):
|
|
2
|
+
"""
|
|
3
|
+
Clase de excepción personalizada para errores relacionados con el servidor del Catastro.
|
|
4
|
+
Atributos:
|
|
5
|
+
- mensaje (str): Mensaje de error específico del servidor del Catastro.
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
def __init__(self, mensaje=''):
|
|
9
|
+
self.message = f'Error del Catastro. {mensaje}'
|
|
10
|
+
super().__init__(self.message)
|
|
11
|
+
|
|
12
|
+
def lanzar_excepcion(mensaje_error: str) -> Exception:
|
|
13
|
+
"""
|
|
14
|
+
Lanza una excepción personalizada con el mensaje de error proporcionado.
|
|
15
|
+
Args:
|
|
16
|
+
mensaje_error (str): El mensaje de error que se incluirá en la excepción.
|
|
17
|
+
Returns:
|
|
18
|
+
Exception: La excepción personalizada con el mensaje de error.
|
|
19
|
+
"""
|
|
20
|
+
|
|
21
|
+
return ErrorServidorCatastro(mensaje=mensaje_error)
|
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
URL_BASE_CALLEJERO = 'https://ovc.catastro.meh.es/OVCServWeb/OVCWcfCallejero/COVCCallejero.svc/json'
|
|
2
|
+
URL_BASE_GEOGRAFIA = 'http://ovc.catastro.meh.es/INSPIRE/wfsCP.aspx'
|
|
3
|
+
URL_BASE_CROQUIS_DATOS = f'https://www1.sedecatastro.gob.es/CYCBienInmueble/SECImprimirCroquisYDatos.aspx'
|
|
4
|
+
|
|
5
|
+
MAPEOS_PROVINCIAS = {
|
|
6
|
+
'La Coruña': 'A Coruña',
|
|
7
|
+
'Orense': 'Ourense',
|
|
8
|
+
'Santa Cruz De Tenerife': 'S.C. Tenerife',
|
|
9
|
+
'Las Palmas De Gran Canaria': 'Las Palmas',
|
|
10
|
+
'Alicante': 'Alacant',
|
|
11
|
+
'Almería': 'Almeria',
|
|
12
|
+
'Cádiz': 'Cadiz',
|
|
13
|
+
'Córdoba': 'Cordoba',
|
|
14
|
+
'Gerona': 'Girona',
|
|
15
|
+
'Castellón De La Plana': 'Castello',
|
|
16
|
+
'Castellón': 'Castello',
|
|
17
|
+
'Lérida': 'Lleida',
|
|
18
|
+
'Málaga': 'Malaga',
|
|
19
|
+
'León': 'Leon',
|
|
20
|
+
'Ávila': 'Avila',
|
|
21
|
+
'Cáceres': 'Caceres',
|
|
22
|
+
'Baleares': 'Illes Balears',
|
|
23
|
+
'Islas Baleares': 'Illes Balears',
|
|
24
|
+
'Jaén': 'Jaen',
|
|
25
|
+
'València': 'Valencia'
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
TIPOS_VIA = {
|
|
29
|
+
"AC": "ACCESO",
|
|
30
|
+
"AG": "AGREGADO",
|
|
31
|
+
"AL": "ALDEA, ALAMEDA",
|
|
32
|
+
"AN": "ANDADOR",
|
|
33
|
+
"AR": "AREA, ARRABAL",
|
|
34
|
+
"AU": "AUTOPISTA",
|
|
35
|
+
"AV": "AVENIDA",
|
|
36
|
+
"AY": "ARROYO",
|
|
37
|
+
"BJ": "BAJADA",
|
|
38
|
+
"BL": "BLOQUE",
|
|
39
|
+
"BO": "BARRIO",
|
|
40
|
+
"BQ": "BARRANQUIL",
|
|
41
|
+
"BR": "BARRANCO",
|
|
42
|
+
"CA": "CAÑADA",
|
|
43
|
+
"CG": "COLEGIO, CIGARRAL",
|
|
44
|
+
"CH": "CHALET",
|
|
45
|
+
"CI": "CINTURON",
|
|
46
|
+
"CJ": "CALLEJA, CALLEJON",
|
|
47
|
+
"CL": "CALLE",
|
|
48
|
+
"CM": "CAMINO, CARMEN",
|
|
49
|
+
"CN": "COLONIA",
|
|
50
|
+
"CO": "CONCEJO, COLEGIO",
|
|
51
|
+
"CP": "CAMPA, CAMPO",
|
|
52
|
+
"CR": "CARRETERA, CARRERA",
|
|
53
|
+
"CS": "CASERIO",
|
|
54
|
+
"CT": "CUESTA, COSTANILLA",
|
|
55
|
+
"CU": "CONJUNTO",
|
|
56
|
+
"CY": "CALEYA",
|
|
57
|
+
"CZ": "CALLIZO",
|
|
58
|
+
"DE": "DETRÁS",
|
|
59
|
+
"DP": "DIPUTACION",
|
|
60
|
+
"DS": "DISEMINADOS",
|
|
61
|
+
"ED": "EDIFICIOS",
|
|
62
|
+
"EM": "EXTRAMUROS",
|
|
63
|
+
"EN": "ENTRADA, ENSANCHE",
|
|
64
|
+
"EP": "ESPALDA",
|
|
65
|
+
"ER": "EXTRARRADIO",
|
|
66
|
+
"ES": "ESCALINATA",
|
|
67
|
+
"EX": "EXPLANADA",
|
|
68
|
+
"FC": "FERROCARRIL",
|
|
69
|
+
"FN": "FINCA",
|
|
70
|
+
"GL": "GLORIETA",
|
|
71
|
+
"GR": "GRUPO",
|
|
72
|
+
"GV": "GRAN VIA",
|
|
73
|
+
"HT": "HUERTA, HUERTO",
|
|
74
|
+
"JR": "JARDINES",
|
|
75
|
+
"LA": "LAGO",
|
|
76
|
+
"LD": "LADO, LADERA",
|
|
77
|
+
"LG": "LUGAR",
|
|
78
|
+
"MA": "MALECON",
|
|
79
|
+
"MC": "MERCADO",
|
|
80
|
+
"ML": "MUELLE",
|
|
81
|
+
"MN": "MUNICIPIO",
|
|
82
|
+
"MS": "MASIAS",
|
|
83
|
+
"MT": "MONTE",
|
|
84
|
+
"MZ": "MANZANA",
|
|
85
|
+
"PB": "POBLADO",
|
|
86
|
+
"PC": "PLACETA",
|
|
87
|
+
"PD": "PARTIDA",
|
|
88
|
+
"PI": "PARTICULAR",
|
|
89
|
+
"PJ": "PASAJE, PASADIZO",
|
|
90
|
+
"PL": "POLIGONO",
|
|
91
|
+
"PM": "PARAMO",
|
|
92
|
+
"PQ": "PARROQUIA, PARQUE",
|
|
93
|
+
"PR": "PROLONGACION, CONTINUAC.",
|
|
94
|
+
"PS": "PASEO",
|
|
95
|
+
"PT": "PUENTE",
|
|
96
|
+
"PU": "PASADIZO",
|
|
97
|
+
"PZ": "PLAZA",
|
|
98
|
+
"QT": "QUINTA",
|
|
99
|
+
"RA": "RACONADA",
|
|
100
|
+
"RB": "RAMBLA",
|
|
101
|
+
"RC": "RINCON, RINCONA",
|
|
102
|
+
"RD": "RONDA",
|
|
103
|
+
"RM": "RAMAL",
|
|
104
|
+
"RP": "RAMPA",
|
|
105
|
+
"RR": "RIERA",
|
|
106
|
+
"RU": "RUA",
|
|
107
|
+
"SA": "SALIDA",
|
|
108
|
+
"SC": "SECTOR",
|
|
109
|
+
"SD": "SENDA",
|
|
110
|
+
"SL": "SOLAR",
|
|
111
|
+
"SN": "SALON",
|
|
112
|
+
"SU": "SUBIDA",
|
|
113
|
+
"TN": "TERRENOS",
|
|
114
|
+
"TO": "TORRENTE",
|
|
115
|
+
"TR": "TRAVESIA",
|
|
116
|
+
"UR": "URBANIZACION",
|
|
117
|
+
"VA": "VALLE",
|
|
118
|
+
"VD": "VIADUCTO",
|
|
119
|
+
"VI": "VIA",
|
|
120
|
+
"VL": "VIAL",
|
|
121
|
+
"VR": "VEREDA"
|
|
122
|
+
}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import requests
|
|
2
|
+
import json
|
|
3
|
+
|
|
4
|
+
from .statics import URL_BASE_CALLEJERO, MAPEOS_PROVINCIAS, TIPOS_VIA
|
|
5
|
+
from .exceptions import lanzar_excepcion
|
|
6
|
+
|
|
7
|
+
def comprobar_errores(respuesta: dict):
|
|
8
|
+
"""
|
|
9
|
+
Comprueba si la respuesta contiene errores.
|
|
10
|
+
|
|
11
|
+
Args:
|
|
12
|
+
respuesta (dict): El diccionario de respuesta.
|
|
13
|
+
|
|
14
|
+
Raises:
|
|
15
|
+
lanzar_excepcion: Si se encuentra un error en la respuesta.
|
|
16
|
+
|
|
17
|
+
Returns:
|
|
18
|
+
bool: True si no se encuentran errores, False en caso contrario.
|
|
19
|
+
"""
|
|
20
|
+
# Check if the response contains the expected structure
|
|
21
|
+
if len(list(respuesta.values())) > 0:
|
|
22
|
+
if list(respuesta.values())[0] is not None and 'lerr' in list(respuesta.values())[0].keys():
|
|
23
|
+
if 'err' in list(respuesta.values())[0]['lerr']:
|
|
24
|
+
raise lanzar_excepcion(mensaje_error=list(respuesta.values())[0]['lerr']['err'][0]['des'])
|
|
25
|
+
else:
|
|
26
|
+
raise lanzar_excepcion(mensaje_error=list(respuesta.values())[0]['lerr'][0]['des'])
|
|
27
|
+
return True
|
|
28
|
+
|
|
29
|
+
def listar_provincias():
|
|
30
|
+
"""
|
|
31
|
+
Obtiene una lista de provincias.
|
|
32
|
+
Returns:
|
|
33
|
+
list: Una lista de nombres de provincias.
|
|
34
|
+
Raises:
|
|
35
|
+
None
|
|
36
|
+
"""
|
|
37
|
+
|
|
38
|
+
response = requests.get(f'{URL_BASE_CALLEJERO}/ObtenerProvincias')
|
|
39
|
+
return [provincia.get('np') for provincia in response.json().get('consulta_provincieroResult').get('provinciero').get('prov')] if comprobar_errores(response.json()) else []
|
|
40
|
+
|
|
41
|
+
def listar_municipios(provincia: str, municipio: str|None = None):
|
|
42
|
+
|
|
43
|
+
"""
|
|
44
|
+
Obtiene una lista de municipios de España.
|
|
45
|
+
Args:
|
|
46
|
+
provincia (str): El nombre de la provincia. Preferentemente en mayúsculas o capitalizado.
|
|
47
|
+
municipio (str, optional): El nombre del municipio. Por defecto es None.
|
|
48
|
+
Returns:
|
|
49
|
+
List[str]: Una lista de nombres de municipios.
|
|
50
|
+
Raises:
|
|
51
|
+
Exception: Si la provincia no existe. Muestra un mensaje con las provincias disponibles.
|
|
52
|
+
"""
|
|
53
|
+
|
|
54
|
+
if provincia and provincia.capitalize() not in MAPEOS_PROVINCIAS and provincia.upper() not in listar_provincias():
|
|
55
|
+
raise Exception(f'La provincia {provincia} no existe. Las provincias de España son: {listar_provincias()}')
|
|
56
|
+
|
|
57
|
+
response = requests.get(f'{URL_BASE_CALLEJERO}/ObtenerMunicipios',
|
|
58
|
+
params={
|
|
59
|
+
'provincia' : MAPEOS_PROVINCIAS.get(provincia.capitalize())
|
|
60
|
+
if MAPEOS_PROVINCIAS.get(provincia.capitalize(),None) != None
|
|
61
|
+
else provincia ,
|
|
62
|
+
'municipio': municipio
|
|
63
|
+
})
|
|
64
|
+
if response.status_code == 200 and comprobar_errores(response.json()):
|
|
65
|
+
mun_dict_raw = json.loads(response.content)
|
|
66
|
+
return [mun.get('nm') for mun in mun_dict_raw.get('consulta_municipieroResult').get('municipiero').get('muni')]
|
|
67
|
+
else:
|
|
68
|
+
return []
|
|
69
|
+
|
|
70
|
+
|
|
71
|
+
def listar_tipos_via():
|
|
72
|
+
"""
|
|
73
|
+
Retorna una lista de los tipos de vía disponibles.
|
|
74
|
+
Returns:
|
|
75
|
+
list: Una lista de los tipos de vía disponibles.
|
|
76
|
+
"""
|
|
77
|
+
|
|
78
|
+
return TIPOS_VIA
|
|
79
|
+
|
|
80
|
+
def listar_calles(provincia: str, municipio: str):
|
|
81
|
+
"""
|
|
82
|
+
Devuelve una lista de calles para una provincia y municipio dados.
|
|
83
|
+
Args:
|
|
84
|
+
provincia (str): El nombre de la provincia.
|
|
85
|
+
municipio (str): El nombre del municipio.
|
|
86
|
+
Returns:
|
|
87
|
+
list: Una lista de calles en formato "tipo de vía nombre de vía".
|
|
88
|
+
"""
|
|
89
|
+
|
|
90
|
+
provincia_final = MAPEOS_PROVINCIAS.get(provincia.capitalize()) if provincia.capitalize() in MAPEOS_PROVINCIAS.keys() else provincia
|
|
91
|
+
if provincia_final.upper() in listar_provincias() and municipio.upper() in listar_municipios(provincia=provincia_final):
|
|
92
|
+
response = requests.get(f'{URL_BASE_CALLEJERO}/ObtenerCallejero',
|
|
93
|
+
params={
|
|
94
|
+
'Provincia': provincia_final,
|
|
95
|
+
'Municipio': municipio
|
|
96
|
+
})
|
|
97
|
+
if response.status_code == 200 and comprobar_errores(response.json()):
|
|
98
|
+
calles_dict_raw = json.loads(response.content)
|
|
99
|
+
return [f"{calle.get('dir').get('tv')} {calle.get('dir').get('nv')}" for calle in calles_dict_raw.get('consulta_callejeroResult').get('callejero').get('calle')]
|
|
100
|
+
else:
|
|
101
|
+
return []
|
|
102
|
+
else: return []
|