自動化無しに生活無し

WEB開発関係を中心に備忘録をまとめています

【Django】アップロードしたPythonファイルを動かす【subprocessと動的import】

  • 作成日時:
  • 最終更新日時:
  • Categories: サーバーサイド
  • Tags: django
thumbnail

アップロードしたPythonファイルをDjango側で動作させる。

仮想環境を動かし、事前にインストールしておいたライブラリも動くようにする。

views.py

from django.shortcuts import render,redirect
from django.views import View

from .models import Document
from .forms import DocumentForm

from django.conf import settings
import subprocess

import importlib.util
import os



class IndexView(View):
    def get(self, request, *args, **kwargs):

        context                 = {}
        context["documents"]    = Document.objects.all()


        # アップロードされたPythonを実行する。
        document        = Document.objects.filter(id=1).first()

        # アップロードされたPythonプログラムのファイルパスを作成
        program_path    = str(settings.BASE_DIR) + document.file.url

        # 仮想環境のファイルパスを作成(pythonファイルのある場所)
        venv_path       = str(settings.BASE_DIR) + "/venv/bin/"


        #== subprocess モジュールで実行する==============

    
        # 実行結果を出力する。(仮想環境を有効にする)
        result  = subprocess.run(["python", program_path], capture_output=True,text=True , cwd=venv_path)
        print(result.stdout)


        #===動的にimportして実行する==============

        # ファイル名からモジュール名を生成
        module_name = os.path.splitext(os.path.basename(program_path))[0]

        # モジュール仕様を作成し、動的にインポート
        spec        = importlib.util.spec_from_file_location(module_name, program_path)
        module      = importlib.util.module_from_spec(spec)
        spec.loader.exec_module(module)
    

        # 前提: アップロードするファイルに、hello関数があること
        module.hello()


        return render(request,"upload/index.html",context)

    def post(self, request, *args, **kwargs):

        form        = DocumentForm(request.POST,request.FILES)

        if not form.is_valid():
            print("バリデーションNG")
            print(form.errors)
            return redirect("upload:document")

        print("バリデーションOK")
        form.save()

        return redirect("upload:index")

index   = IndexView.as_view()

subprocess モジュールを使う

まず、BASE_DIRとアップロードされたPythonのファイルパスを連結する。

次に、BASE_DIRと仮想環境のファイルパスを連結する。

subprocess.run()の第一引数にコマンドを指定。cwdに仮想環境のファイルパスを指定。capture_output=True,text=Trueで標準出力を手に入れる。

動的にimportして動かす

動的にimportして動かす場合、アップロードするファイルに決まった関数やクラスがあることを前提とする。

ソースコード

https://github.com/seiya0723/django-upload-python/

スポンサーリンク