web-dev-qa-db-ja.com

ModuleNotFoundError-サービスを開始しようとしたときに「main」という名前のモジュールがありません

環境:

  • Django 3.7を使用してPythonアプリケーションを作成しました。
  • 第2世代のGoogle App Engineスタンダード環境を使用しようとしています。

私のアプリケーションは、python manage.py runserver経由で実行すると問題なく動作します。それでも、Google App Engineにデプロイしようとすると、突然停止します。

Traceback (most recent call last):
  File "/tmp/tmphgUsp3/lib/python3.7/site-packages/gunicorn/arbiter.py", line 583, in spawn_worker
    worker.init_process()
  File "/tmp/tmphgUsp3/lib/python3.7/site-packages/gunicorn/workers/base.py", line 129, in init_process
    self.load_wsgi()
  File "/tmp/tmphgUsp3/lib/python3.7/site-packages/gunicorn/workers/base.py", line 138, in load_wsgi
    self.wsgi = self.app.wsgi()
  File "/tmp/tmphgUsp3/lib/python3.7/site-packages/gunicorn/app/base.py", line 67, in wsgi
    self.callable = self.load()
  File "/tmp/tmphgUsp3/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 52, in load
    return self.load_wsgiapp()
  File "/tmp/tmphgUsp3/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 41, in load_wsgiapp
    return util.import_app(self.app_uri)
  File "/tmp/tmphgUsp3/lib/python3.7/site-packages/gunicorn/util.py", line 350, in import_app
    __import__(module)
ModuleNotFoundError: No module named 'main'

多数のスレッドを調べましたが、問題を見つけることができません。 (参考として、dev_appserver.pyエミュレータでも同じ問題が発生しますが、これは良いことです)。

以下は私のapp.yamlです

runtime: python37
env: standard

handlers:
- url: /static
  static_dir: static/
- url: .*
  script: demosite.wsgi.main

私のwsgi.pyファイルは次のパスにあります:demosite/wsgi.pyそしてその内容は次のようになります:

import os

from Django.core.wsgi import get_wsgi_application

os.environ.setdefault('Django_SETTINGS_MODULE', 'demosite.settings')

main = get_wsgi_application()

私のsettings.pyファイル:

import os

class AppSettings(object):
    GoogleCloudProject = os.getenv('GOOGLE_CLOUD_PROJECT')

BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))

SECRET_KEY = 'say what?'

DEBUG = True

ALLOWED_HOSTS = [
    '*'
]


INSTALLED_APPS = [
    'anchor.apps.AnchorConfig',
    'crispy_forms',
    'Django.contrib.admin',
    'Django.contrib.auth',
    'Django.contrib.contenttypes',
    'Django.contrib.sessions',
    'Django.contrib.messages',
    'Django.contrib.staticfiles',
]

MIDDLEWARE = [
    'Django.middleware.security.SecurityMiddleware',
    'Django.contrib.sessions.middleware.SessionMiddleware',
    'Django.middleware.common.CommonMiddleware',
    'Django.middleware.csrf.CsrfViewMiddleware',
    'Django.contrib.auth.middleware.AuthenticationMiddleware',
    'Django.contrib.messages.middleware.MessageMiddleware',
    'Django.middleware.clickjacking.XFrameOptionsMiddleware',
]

ROOT_URLCONF = 'demosite.urls'

TEMPLATES = [
    {
        'BACKEND': 'Django.template.backends.Django.DjangoTemplates',
        'DIRS': [],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'Django.template.context_processors.debug',
                'Django.template.context_processors.request',
                'Django.contrib.auth.context_processors.auth',
                'Django.contrib.messages.context_processors.messages',
            ],
        },
    },
]

WSGI_APPLICATION = 'demosite.wsgi.main'

try:
    import MySQLdb
except ImportError:
    import pymysql
    pymysql.install_as_MySQLdb()


if os.getenv('SERVER_SOFTWARE', '').startswith('Google App Engine'):
    DATABASES = {
        'default': {
            'ENGINE': 'Django.db.backends.mysql',
            'NAME': 'webapp',
            'USER': 'aasdeytst',
            'PASSWORD': 'asdasygetasfasdfasd.',
            'Host': 'asdgiuasfivaasd',
            'PORT': '3306'
        }
    }
else:
    DATABASES = {
        'default': {
            'ENGINE': 'Django.db.backends.mysql',
            'NAME': 'webapp',
            'USER': 'awthdsfhfdhdf',
            'PASSWORD': 'asdasdasdagwdatwt',
            'Host': 'localhost',
            'PORT': '3306'
        }
    }

AUTH_PASSWORD_VALIDATORS = [
    {
        'NAME': 'Django.contrib.auth.password_validation.UserAttributeSimilarityValidator',
    },
    {
        'NAME': 'Django.contrib.auth.password_validation.MinimumLengthValidator',
    },
    {
        'NAME': 'Django.contrib.auth.password_validation.CommonPasswordValidator',
    },
    {
        'NAME': 'Django.contrib.auth.password_validation.NumericPasswordValidator',
    },
]

ANGUAGE_CODE = 'en-us'

TIME_ZONE = 'UTC'

USE_I18N = True

USE_L10N = True

USE_TZ = True

STATIC_ROOT = 'static'
STATIC_URL = '/static/'

CRISPY_TEMPLATE_PACK = 'bootstrap4'

LOGIN_REDIRECT_URL = 'index'
LOGIN_URL = 'login'

SESSION_EXPIRE_AT_BROWSER_CLOSE = False
SESSION_COOKIE_AGE = 1800

何が欠けていますか、何が間違っていますか?私はこの問題を無意味に理解しようと4時間近く費やしてきました。

11
Marley

デフォルトでは、App Engineはmain.pyというファイルでapp変数を探します。 2つのオプションがあります。AppEngineが期待する場所にWSGIアプリを配置するか、カスタムエントリポイントを定義します。

App Engineが期待する場所にWSGIアプリを配置します。

app変数を含むmain.pyというファイルを作成できます。この変数は、正しい場所から単純にインポートされ、エイリアスが付けられます。

from demosite.wsgi import main as app

カスタムエントリポイントを追加する:

https://cloud.google.com/appengine/docs/standard/python3/config/appref から:

entrypoint:オプション。アプリの起動時に実行されるコマンド。アプリでHTTPリクエストを受信するには、entrypointに、PORT環境変数で指定されたポートでリッスンするWebサーバーを起動するコマンドを含める必要があります。 entrypointを指定しない場合、App EngineはGunicornウェブサーバーを構成して起動します。

デフォルトではこれです:

entrypoint: gunicorn -b :$PORT main:app

あなたは次のようなものが必要でしょう:

entrypoint: gunicorn -b :$PORT demosite.wsgi:main

アプリケーションの起動の詳細については、こちらをご覧ください: https://cloud.google.com/appengine/docs/standard/python3/runtime#application_startup

4
Dustin Ingram

追加:

Main.pyは、app.yamlがあるアプリケーションのルートにある必要があります。

そして、内容もそうであることができます:

   from mysite.wsgi import application

   # App Engine by default looks for a main.py file at the root of the app
   # directory with a WSGI-compatible object called app.
   # This file imports the WSGI-compatible object of your Django app,
   # application from mysite/wsgi.py and renames it app so it is discoverable by
   # App Engine without additional configuration.
   # Alternatively, you can add a custom entrypoint field in your app.yaml:
   # entrypoint: gunicorn -b :$PORT mysite.wsgi
   app = application