This repository has been archived by the owner on Dec 9, 2022. It is now read-only.
/
views.py
227 lines (195 loc) · 9.58 KB
/
views.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
from django.shortcuts import render
from django.http import HttpResponse
from django.views.generic import View
from django.views.decorators.csrf import csrf_exempt
from django.utils.decorators import method_decorator
from django.core.urlresolvers import reverse
from django.contrib.sites.shortcuts import get_current_site
from lxml import etree
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.types import Integer
from sqlalchemy.schema import Column
from sqlalchemy import inspection,log
from sqlalchemy import util as sqlalchemy_util
from sqlalchemy.sql import util as sql_util
from sqlalchemy.orm.mapper import Mapper as BaseMapper
from pycsw.server import Csw as PyCsw
from itertools import chain
from .models import Application
from .pycswsettings import build_pycsw_settings
import logging
import traceback
from pycsw import util
from lxml import etree
from pycsw.server import write_boundingbox
logger = logging.getLogger(__name__)
class Csw(PyCsw):
def _write_record(self, recobj, queryables):
''' replicate from original method.
Only changes is the column separator in links is "\t" instead of ","
'''
if self.kvp['elementsetname'] == 'brief':
elname = 'BriefRecord'
elif self.kvp['elementsetname'] == 'summary':
elname = 'SummaryRecord'
else:
elname = 'Record'
record = etree.Element(util.nspath_eval('csw:%s' % elname,
self.context.namespaces))
if ('elementname' in self.kvp and
len(self.kvp['elementname']) > 0):
for elemname in self.kvp['elementname']:
if (elemname.find('BoundingBox') != -1 or
elemname.find('Envelope') != -1):
bboxel = write_boundingbox(util.getqattr(recobj,
self.context.md_core_model['mappings']['pycsw:BoundingBox']),
self.context.namespaces)
if bboxel is not None:
record.append(bboxel)
else:
value = util.getqattr(recobj, queryables[elemname]['dbcol'])
if value:
etree.SubElement(record,
util.nspath_eval(elemname,
self.context.namespaces)).text = value
elif 'elementsetname' in self.kvp:
if (self.kvp['elementsetname'] == 'full' and
util.getqattr(recobj, self.context.md_core_model['mappings']\
['pycsw:Typename']) == 'csw:Record' and
util.getqattr(recobj, self.context.md_core_model['mappings']\
['pycsw:Schema']) == 'http://www.opengis.net/cat/csw/2.0.2' and
util.getqattr(recobj, self.context.md_core_model['mappings']\
['pycsw:Type']) != 'service'):
# dump record as is and exit
return etree.fromstring(util.getqattr(recobj,
self.context.md_core_model['mappings']['pycsw:XML']), self.context.parser)
etree.SubElement(record,
util.nspath_eval('dc:identifier', self.context.namespaces)).text = \
util.getqattr(recobj,
self.context.md_core_model['mappings']['pycsw:Identifier'])
for i in ['dc:title', 'dc:type']:
val = util.getqattr(recobj, queryables[i]['dbcol'])
if not val:
val = ''
etree.SubElement(record, util.nspath_eval(i,
self.context.namespaces)).text = val
if self.kvp['elementsetname'] in ['summary', 'full']:
# add summary elements
keywords = util.getqattr(recobj, queryables['dc:subject']['dbcol'])
if keywords is not None:
for keyword in keywords.split(','):
etree.SubElement(record,
util.nspath_eval('dc:subject',
self.context.namespaces)).text = keyword
val = util.getqattr(recobj, queryables['dc:format']['dbcol'])
if val:
etree.SubElement(record,
util.nspath_eval('dc:format',
self.context.namespaces)).text = val
# links
rlinks = util.getqattr(recobj,
self.context.md_core_model['mappings']['pycsw:Links'])
if rlinks:
links = rlinks.split('^')
for link in links:
linkset = link.split('\t')
etree.SubElement(record,
util.nspath_eval('dct:references',
self.context.namespaces),
scheme=linkset[2].replace("\"",""")).text = linkset[-1]
for i in ['dc:relation', 'dct:modified', 'dct:abstract']:
val = util.getqattr(recobj, queryables[i]['dbcol'])
if val is not None:
etree.SubElement(record,
util.nspath_eval(i, self.context.namespaces)).text = val
if self.kvp['elementsetname'] == 'full': # add full elements
for i in ['dc:date', 'dc:creator', \
'dc:publisher', 'dc:contributor', 'dc:source', \
'dc:language', 'dc:rights']:
val = util.getqattr(recobj, queryables[i]['dbcol'])
if val:
etree.SubElement(record,
util.nspath_eval(i, self.context.namespaces)).text = val
# always write out ows:BoundingBox
bboxel = write_boundingbox(getattr(recobj,
self.context.md_core_model['mappings']['pycsw:BoundingBox']),
self.context.namespaces)
if bboxel is not None:
record.append(bboxel)
return record
@inspection._self_inspects
@log.class_logger
class Mapper(BaseMapper):
def _configure_pks(self):
self.tables = sql_util.find_tables(self.mapped_table)
self._pks_by_table = {}
self._cols_by_table = {}
all_cols = sqlalchemy_util.column_set(chain(*[col.proxy_set for col in self._columntoproperty]))
pk_cols = sqlalchemy_util.column_set(c for c in all_cols if c.primary_key)
# identify primary key columns which are also mapped by this mapper.
tables = set(self.tables + [self.mapped_table])
self._all_tables.update(tables)
self._cols_by_table[self.mapped_table] = all_cols
primary_key = [c for c in all_cols if c.name in self._primary_key_argument]
self._pks_by_table[self.mapped_table] = primary_key
self.primary_key = tuple(primary_key)
self._log("Identified primary key columns: %s", primary_key)
# determine cols that aren't expressed within our tables; mark these
# as "read only" properties which are refreshed upon INSERT/UPDATE
self._readonly_props = set(
self._columntoproperty[col]
for col in self._columntoproperty
if self._columntoproperty[col] not in self._identity_key_props and
(not hasattr(col, 'table') or
col.table not in self._cols_by_table))
class CswEndpoint(View):
application_records = {}
def get(self, request,app=None):
pycsw_settings = build_pycsw_settings(app)
server = Csw(rtconfig=pycsw_settings, env=request.META.copy())
if not app:
app = "all"
#request by named app, use app related view
record_table = Application.get_view_name(app)
try:
if not self.application_records.get(app,None):
base = declarative_base(bind=server.repository.engine,mapper=Mapper)
self.application_records[app] = type('dataset', (base,),
dict(__tablename__=record_table,__table_args__={'autoload': True,'schema': None},__mapper_args__={"primary_key":["id"]}))
server.repository.dataset = self.application_records[app]
except:
pass
server.request = "http://{}{}".format(get_current_site(request),
reverse("csw_endpoint"))
server.requesttype = request.method
server.kvp = self._normalize_params(request.GET)
response = server.dispatch()
return HttpResponse(response, content_type="application/xml")
@method_decorator(csrf_exempt)
def dispatch(self, request, *args, **kwargs):
return super(CswEndpoint, self).dispatch(request, *args, **kwargs)
def post(self, request,app=None):
pycsw_settings = build_pycsw_settings()
server = Csw(rtconfig=pycsw_settings, env=request.META.copy(),
version=self._get_post_version(request.body))
logger.info(request.body)
server.request = request.body
server.requesttype = request.method
status_code, response = server.dispatch()
return HttpResponse(response, status=status_code,
content_type="application/xml")
# TODO - Remove this method once pycsw mainlines the pending pull request
def _normalize_params(self, query_dict):
"""
A hack to overcome PyCSW's early normalizing of KVP args.
Since PyCSW normalizes KVP args in the server.dispatch_cgi() and
server.dispatch_wsgi() methods, we need to explicitly do the same
here, as we are bypassing these methods and calling server.dispatch()
"""
kvp = dict()
for k, v in query_dict.iteritems():
kvp[k.lower()] = v
return kvp
def _get_post_version(self, raw_request):
exml = etree.fromstring(raw_request)
return exml.get("version")