kopia lustrzana https://github.com/OpenDroneMap/WebODM
127 wiersze
6.1 KiB
Python
127 wiersze
6.1 KiB
Python
import mimetypes
|
|
import os
|
|
|
|
from django.http import FileResponse
|
|
from django.http import HttpResponse
|
|
from wsgiref.util import FileWrapper
|
|
from rest_framework import status
|
|
from rest_framework.response import Response
|
|
from app.plugins.views import TaskView
|
|
from worker.tasks import execute_grass_script
|
|
from app.plugins.grass_engine import grass, GrassEngineException, cleanup_grass_context
|
|
from worker.celery import app as celery
|
|
from app.plugins import get_current_plugin
|
|
|
|
class TaskChangeMapGenerate(TaskView):
|
|
def post(self, request, pk=None):
|
|
|
|
role = request.data.get('role', 'reference')
|
|
if role == 'reference':
|
|
reference_pk = pk
|
|
compare_task_pk = request.data.get('other_task', None)
|
|
else:
|
|
reference_pk = request.data.get('other_task', None)
|
|
compare_task_pk = pk
|
|
|
|
reference_task = self.get_and_check_task(request, reference_pk)
|
|
if compare_task_pk is None:
|
|
return Response({'error': 'You must select a task to compare to.'}, status=status.HTTP_400_BAD_REQUEST)
|
|
compare_task = self.get_and_check_task(request, compare_task_pk)
|
|
|
|
reference_pc = os.path.abspath(reference_task.get_asset_download_path("georeferenced_model.laz"))
|
|
reference_dsm = os.path.abspath(reference_task.get_asset_download_path("dsm.tif"))
|
|
reference_dtm = os.path.abspath(reference_task.get_asset_download_path("dtm.tif"))
|
|
|
|
compare_pc = os.path.abspath(compare_task.get_asset_download_path("georeferenced_model.laz"))
|
|
compare_dsm = os.path.abspath(compare_task.get_asset_download_path("dsm.tif"))
|
|
compare_dtm = os.path.abspath(compare_task.get_asset_download_path("dtm.tif"))
|
|
|
|
plugin = get_current_plugin()
|
|
|
|
# We store the aligned DEMs on the persistent folder, to avoid recalculating them in the future
|
|
aligned_dsm = plugin.get_persistent_path("{}_{}_dsm.tif".format(pk, compare_task_pk))
|
|
aligned_dtm = plugin.get_persistent_path("{}_{}_dtm.tif".format(pk, compare_task_pk))
|
|
|
|
try:
|
|
context = grass.create_context({'auto_cleanup' : False, 'location': 'epsg:3857', 'python_path': plugin.get_python_packages_path()})
|
|
format = request.data.get('format', 'GPKG')
|
|
epsg = int(request.data.get('epsg', '3857'))
|
|
supported_formats = ['GPKG', 'ESRI Shapefile', 'DXF', 'GeoJSON']
|
|
if not format in supported_formats:
|
|
raise GrassEngineException("Invalid format {} (must be one of: {})".format(format, ",".join(supported_formats)))
|
|
min_area = float(request.data.get('min_area', 40))
|
|
min_height = float(request.data.get('min_height', 5))
|
|
resolution = float(request.data.get('resolution', 0.5))
|
|
display_type = request.data.get('display_type', 'contour')
|
|
can_align_and_rasterize = request.data.get('align', 'false')
|
|
|
|
current_dir = os.path.dirname(os.path.abspath(__file__))
|
|
context.add_param('reference_pc', reference_pc)
|
|
context.add_param('compare_pc', compare_pc)
|
|
context.add_param('reference_dsm', reference_dsm)
|
|
context.add_param('reference_dtm', reference_dtm)
|
|
context.add_param('compare_dsm', compare_dsm)
|
|
context.add_param('compare_dtm', compare_dtm)
|
|
context.add_param('aligned_dsm', aligned_dsm)
|
|
context.add_param('aligned_dtm', aligned_dtm)
|
|
context.add_param('format', format)
|
|
context.add_param('epsg', epsg)
|
|
context.add_param('display_type', display_type)
|
|
context.add_param('resolution', resolution)
|
|
context.add_param('min_area', min_area)
|
|
context.add_param('min_height', min_height)
|
|
context.add_param('can_align_and_rasterize', can_align_and_rasterize)
|
|
|
|
celery_task_id = execute_grass_script.delay(os.path.join(current_dir, "changedetection.py"), context.serialize()).task_id
|
|
|
|
return Response({'celery_task_id': celery_task_id}, status=status.HTTP_200_OK)
|
|
except GrassEngineException as e:
|
|
return Response({'error': str(e)}, status=status.HTTP_200_OK)
|
|
|
|
class TaskChangeMapCheck(TaskView):
|
|
def get(self, request, pk=None, celery_task_id=None):
|
|
res = celery.AsyncResult(celery_task_id)
|
|
if not res.ready():
|
|
return Response({'ready': False}, status=status.HTTP_200_OK)
|
|
else:
|
|
result = res.get()
|
|
if result.get('error', None) is not None:
|
|
cleanup_grass_context(result['context'])
|
|
return Response({'ready': True, 'error': result['error']})
|
|
|
|
output = result.get('output')
|
|
if not output or not os.path.exists(output):
|
|
cleanup_grass_context(result['context'])
|
|
return Response({'ready': True, 'error': output})
|
|
|
|
request.session['change_detection_' + celery_task_id] = output
|
|
return Response({'ready': True})
|
|
|
|
|
|
class TaskChangeMapDownload(TaskView):
|
|
def get(self, request, pk=None, celery_task_id=None):
|
|
change_detection_file = request.session.get('change_detection_' + celery_task_id, None)
|
|
|
|
if change_detection_file is not None:
|
|
filename = os.path.basename(change_detection_file)
|
|
filesize = os.stat(change_detection_file).st_size
|
|
|
|
f = open(change_detection_file, "rb")
|
|
|
|
# More than 100mb, normal http response, otherwise stream
|
|
# Django docs say to avoid streaming when possible
|
|
stream = filesize > 1e8
|
|
if stream:
|
|
response = FileResponse(f)
|
|
else:
|
|
response = HttpResponse(FileWrapper(f),
|
|
content_type=(mimetypes.guess_type(filename)[0] or "application/zip"))
|
|
|
|
response['Content-Type'] = mimetypes.guess_type(filename)[0] or "application/zip"
|
|
response['Content-Disposition'] = "attachment; filename={}".format(filename)
|
|
response['Content-Length'] = filesize
|
|
|
|
return response
|
|
else:
|
|
return Response({'error': 'Invalid change_detecton download id'})
|