Source code for shoop.addons.admin_module.views.upload

# -*- coding: utf-8 -*-
# This file is part of Shoop.
#
# Copyright (c) 2012-2016, Shoop Ltd. All rights reserved.
#
# This source code is licensed under the AGPLv3 license found in the
# LICENSE file in the root directory of this source tree.
from __future__ import unicode_literals

import os
import shutil
import tempfile
import traceback
import uuid
import zipfile

from django import forms
from django.core.urlresolvers import reverse
from django.http.response import HttpResponseRedirect
from django.utils.translation import ugettext_lazy as _
from django.views.generic import FormView

from shoop.addons.installer import PackageInstaller
from shoop.admin.toolbar import PostActionButton, Toolbar
from shoop.admin.utils.urls import manipulate_query_string
from shoop.utils.excs import Problem
from shoop.utils.iterables import first


[docs]class AddonUploadForm(forms.Form): file = forms.FileField( label=_("Addon file (ZIP)"), help_text=_("Only upload addon files you trust.") )
[docs]class AddonUploadView(FormView): form_class = AddonUploadForm template_name = "shoop/admin/addons/upload.jinja" title = "Upload Addon"
[docs] def form_valid(self, form): file = form.cleaned_data["file"] if not file.name.endswith(".zip"): raise Problem(_("Only ZIP files are supported")) # TODO: Maybe verify the file before saving? filename = "shoop-addon-%s-%s" % (uuid.uuid4(), os.path.basename(file.name)) with open(os.path.join(tempfile.gettempdir(), filename), "wb") as outf: shutil.copyfileobj(file, outf) return HttpResponseRedirect( manipulate_query_string( reverse("shoop_admin:addon.upload_confirm"), file=filename ) )
[docs] def get_context_data(self, **kwargs): context = super(AddonUploadView, self).get_context_data(**kwargs) context["toolbar"] = Toolbar([ PostActionButton( icon="fa fa-upload", form_id="upload_form", text=_("Upload"), extra_css_class="btn-success", ) ]) return context
[docs]class AddonUploadConfirmView(FormView): form_class = forms.Form template_name = "shoop/admin/addons/upload_confirm.jinja" title = "Upload Addon"
[docs] def get_addon_path(self): filename = os.path.basename(self.request.REQUEST.get("file")) path = os.path.join(tempfile.gettempdir(), filename) if not os.path.isfile(path): raise ValueError("File not found") if hasattr(os, "geteuid") and os.stat(path).st_uid != os.geteuid(): raise ValueError("File not owned by current user") return path
[docs] def get_context_data(self, **kwargs): context = super(AddonUploadConfirmView, self).get_context_data(**kwargs) with zipfile.ZipFile(self.get_addon_path()) as zf: context["filenames"] = sorted(zf.namelist()) pkg_info_path = first(filename for filename in context["filenames"] if filename.endswith("PKG-INFO")) if pkg_info_path: context["pkg_info"] = zf.read(pkg_info_path).decode("UTF-8", "replace") context["toolbar"] = Toolbar([ PostActionButton( icon="fa fa-download", form_id="install_form", text=_("Install Addon"), extra_css_class="btn-success", ) ]) return context
[docs] def form_valid(self, form): installer = PackageInstaller() self.template_name = "shoop/admin/addons/upload_complete.jinja" context = {} try: installer.install_package(self.get_addon_path()) except Exception: context["error"] = traceback.format_exc() context["success"] = False else: context["success"] = True try: # Try cleaning up behind ourselves os.unlink(self.get_addon_path()) except Exception: pass context["log_content"] = installer.get_log() return self.render_to_response(context)