2019-05-21 08:24:01 +00:00
|
|
|
# ~*~ coding: utf-8 ~*~
|
|
|
|
#
|
|
|
|
|
|
|
|
import unicodecsv
|
2019-05-31 10:20:24 +00:00
|
|
|
import codecs
|
2019-05-27 07:50:44 +00:00
|
|
|
from datetime import datetime
|
2019-05-21 08:24:01 +00:00
|
|
|
|
|
|
|
from six import BytesIO
|
|
|
|
from rest_framework.renderers import BaseRenderer
|
|
|
|
from rest_framework.utils import encoders, json
|
|
|
|
|
|
|
|
from ..utils import get_logger
|
|
|
|
|
|
|
|
logger = get_logger(__file__)
|
|
|
|
|
|
|
|
|
|
|
|
class JMSCSVRender(BaseRenderer):
|
|
|
|
|
|
|
|
media_type = 'text/csv'
|
|
|
|
format = 'csv'
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _get_header(fields, template):
|
|
|
|
if template == 'import':
|
|
|
|
header = [
|
|
|
|
k for k, v in fields.items()
|
|
|
|
if not v.read_only and k != 'org_id'
|
|
|
|
]
|
|
|
|
elif template == 'update':
|
|
|
|
header = [k for k, v in fields.items() if not v.read_only]
|
|
|
|
else:
|
|
|
|
# template in ['export']
|
|
|
|
header = [k for k, v in fields.items() if not v.write_only]
|
|
|
|
return header
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _gen_table(data, header, labels=None):
|
|
|
|
labels = labels or {}
|
|
|
|
yield [labels.get(k, k) for k in header]
|
|
|
|
|
|
|
|
for item in data:
|
|
|
|
row = [item.get(key) for key in header]
|
|
|
|
yield row
|
|
|
|
|
2019-05-27 07:50:44 +00:00
|
|
|
def set_response_disposition(self, serializer, context):
|
|
|
|
response = context.get('response')
|
|
|
|
if response and hasattr(serializer, 'Meta') and \
|
|
|
|
hasattr(serializer.Meta, "model"):
|
|
|
|
model_name = serializer.Meta.model.__name__.lower()
|
|
|
|
now = datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
|
|
|
|
filename = "{}_{}.csv".format(model_name, now)
|
|
|
|
disposition = 'attachment; filename="{}"'.format(filename)
|
|
|
|
response['Content-Disposition'] = disposition
|
|
|
|
|
2019-05-21 08:24:01 +00:00
|
|
|
def render(self, data, media_type=None, renderer_context=None):
|
|
|
|
renderer_context = renderer_context or {}
|
|
|
|
request = renderer_context['request']
|
|
|
|
template = request.query_params.get('template', 'export')
|
|
|
|
view = renderer_context['view']
|
|
|
|
data = json.loads(json.dumps(data, cls=encoders.JSONEncoder))
|
|
|
|
if template == 'import':
|
|
|
|
data = [data[0]] if data else data
|
|
|
|
|
|
|
|
try:
|
|
|
|
serializer = view.get_serializer()
|
2019-05-27 07:50:44 +00:00
|
|
|
self.set_response_disposition(serializer, renderer_context)
|
2019-05-21 08:24:01 +00:00
|
|
|
except Exception as e:
|
|
|
|
logger.debug(e, exc_info=True)
|
|
|
|
value = 'The resource not support export!'.encode('utf-8')
|
|
|
|
else:
|
|
|
|
fields = serializer.get_fields()
|
|
|
|
header = self._get_header(fields, template)
|
|
|
|
labels = {k: v.label for k, v in fields.items() if v.label}
|
|
|
|
table = self._gen_table(data, header, labels)
|
|
|
|
|
|
|
|
csv_buffer = BytesIO()
|
2019-05-31 10:20:24 +00:00
|
|
|
csv_buffer.write(codecs.BOM_UTF8)
|
|
|
|
csv_writer = unicodecsv.writer(csv_buffer, encoding='utf-8')
|
2019-05-21 08:24:01 +00:00
|
|
|
for row in table:
|
|
|
|
csv_writer.writerow(row)
|
|
|
|
|
|
|
|
value = csv_buffer.getvalue()
|
|
|
|
|
|
|
|
return value
|