1

method="POST" を持つフォームを送信すると、それを処理するビューがメソッド GET を取得し、ファイルのアップロードに必要な POST になるように見えません。私の既存のものを缶詰めするだけではなく(1.4にアップグレードする前に1週間前に働いていました)、https://docs.djangoproject.com/en/dev/topics/http/file-uploads/からサンプルをアップロードするフォームを使用しようとしました、ここで何が問題になっているかわかりますか? Chrome Incognito を使用して新しいセッションを試してみましたが、結果は同じでした。method="POST" を含めない場合、メソッド GET で送信され、フィールドはメールの request.GET 辞書に含まれます。ただし、POST/post に変更すると、メソッド GET のままになり、フィールドは request.POST または request.GET 辞書にありません。この動作を引き起こしている Django 1.4 のバグはありますか? 足りないものはありますか?無関係な潜在的な変数をすべて排除するために、簡単な例を作成しようとしました。

簡単なテスト シナリオ:

urls.py:

url(r'^upload_test/+$', 'views.presentation.UploadTest'),
url(r'^uploaded_test/+$', 'views.presentation.UploadedTest'),

ビュー/presentation.py:

def Notify(subject, body):
    """
        Send an email to <to_email>, from <from_email> with the
        subject and body supplied.
    """
    send_mail(subject, body, '<from_email>', ['<to_email>'], fail_silently=False)

def UploadTest(request):
    Notify("UploadTest", "METHOD: " + str(request.method) + """

    """ + str(request))
    return render_to_response('upload.htm', context_instance=RequestContext(request))

def UploadedTest(request):
    Notify("UploadedTest", "METHOD: " + str(request.method) + """

    """ + str(request) + """

    FILES:
    """ + str(request.FILES))
    return render_to_response('upload.htm', context_instance=RequestContext(request))

upload.htm (注意方法は POST です。念のため小文字も試しました):

<form
    id="upload_file_form"
    action="/uploaded_test"
    method="POST"
    enctype="multipart/form-data"
    target="_blank"
>
    {% csrf_token %}
    <input type="text" name="extra_field" value="extra post field" />
    <input type="file" name="upload_file" id="upload_file" />
    <input type="submit" name="upload" value="Submit" />
</form>

簡単なテスト シナリオの結果

私が受け取る電子メール (1 つ目はフォームの事前送信、2 つ目は投稿されたフォーム ページであることに注意してください):

最初のメール (件名「UploadTest」):

METHOD: GET

<WSGIRequest
    path:/upload_test/,
    GET:<QueryDict: {}>,
    POST:<QueryDict: {}>,
    COOKIES:{
        'csrftoken': '...',
        'sessionid': '...'},
    META:{'CSRF_COOKIE': '...',
    'DOCUMENT_ROOT': '...',
    'GATEWAY_INTERFACE': 'CGI/1.1',
    'HTTPS': '1',
    'HTTP_ACCEPT': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
    'HTTP_ACCEPT_ENCODING': 'gzip, deflate',
    'HTTP_ACCEPT_LANGUAGE': 'en-us,en;q=0.5',
    'HTTP_CONNECTION': 'keep-alive',
    'HTTP_COOKIE': '... csrftoken=...; CFID=16005; CFTOKEN=...; sessionid=...',
    'HTTP_HOST': '...',
    'HTTP_USER_AGENT': 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:11.0) Gecko/20100101 Firefox/11.0',
    'PATH_INFO': u'/upload_test/',
    'PATH_TRANSLATED': '...site.wsgi/upload_test/',
    'QUERY_STRING': '',
    'REMOTE_ADDR': '10.1.0.100',
    'REMOTE_PORT': '56037',
    'REQUEST_METHOD': 'GET',
    'REQUEST_URI': '/upload_test/',
    'SCRIPT_FILENAME': '...site.wsgi',
    'SCRIPT_NAME': u'',
    'SERVER_ADDR': '10.1.10.12',
    'SERVER_ADMIN': '...',
    'SERVER_NAME': '...',
    'SERVER_PORT': '443',
    'SERVER_PROTOCOL': 'HTTP/1.1',
    'SERVER_SIGNATURE': '',
    'SERVER_SOFTWARE': 'Apache',
    'SSL_TLS_SNI': '...',
    'mod_wsgi.application_group': '...|',
    'mod_wsgi.callable_object': 'application',
    'mod_wsgi.handler_script': '',
    'mod_wsgi.input_chunked': '0',
    'mod_wsgi.listener_host': '',
    'mod_wsgi.listener_port': '443',
    'mod_wsgi.process_group': '',
    'mod_wsgi.request_handler': 'wsgi-script',
    'mod_wsgi.script_reloading': '1',
    'mod_wsgi.version': (3, 3),
    'wsgi.errors': <mod_wsgi.Log object at 0x1962048>,
    'wsgi.file_wrapper': <built-in method file_wrapper of mod_wsgi.Adapter object at 0xb73d2770>,
    'wsgi.input': <mod_wsgi.Input object at 0x1962020>,
    'wsgi.multiprocess': True,
    'wsgi.multithread': True,
    'wsgi.run_once': False,
    'wsgi.url_scheme': 'https',
    'wsgi.version': (1, 1)}>

2 番目の電子メール (件名は「UploadedTest」、メソッドは「GET」のまま、HTTP_REFERER が正しく、request.FILES が存在するが空であることに注意してください。):

METHOD: GET

<WSGIRequest
    path:/uploaded_test/,
    GET:<QueryDict: {}>,
    POST:<QueryDict: {}>,
    COOKIES:{
        'csrftoken': '...',
        'sessionid': '...'
    },
    META:{'CSRF_COOKIE': '...',
    'DOCUMENT_ROOT': '...',
    'GATEWAY_INTERFACE': 'CGI/1.1',
    'HTTPS': '1',
    'HTTP_ACCEPT': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
    'HTTP_ACCEPT_ENCODING': 'gzip, deflate',
    'HTTP_ACCEPT_LANGUAGE': 'en-us,en;q=0.5',
    'HTTP_CONNECTION': 'keep-alive',
    'HTTP_COOKIE': 'csrftoken=...; sessionid=...',
    'HTTP_HOST': '...',
    'HTTP_REFERER': '.../upload_test/',
    'HTTP_USER_AGENT': 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:11.0) Gecko/20100101 Firefox/11.0',
    'PATH_INFO': u'/uploaded_test/',
    'PATH_TRANSLATED': '...site.wsgi/uploaded_test/',
    'QUERY_STRING': '',
    'REMOTE_ADDR': '10.1.0.100',
    'REMOTE_PORT': '56040',
    'REQUEST_METHOD': 'GET',
    'REQUEST_URI': '/uploaded_test/',
    'SCRIPT_FILENAME': '...site.wsgi',
    'SCRIPT_NAME': u'',
    'SERVER_ADDR': '10.1.10.12',
    'SERVER_ADMIN': '...',
    'SERVER_NAME': '...',
    'SERVER_PORT': '443',
    'SERVER_PROTOCOL': 'HTTP/1.1',
    'SERVER_SIGNATURE': '',
    'SERVER_SOFTWARE': 'Apache',
    'SSL_TLS_SNI': '...',
    'mod_wsgi.application_group': '...|',
    'mod_wsgi.callable_object': 'application',
    'mod_wsgi.handler_script': '',
    'mod_wsgi.input_chunked': '0',
    'mod_wsgi.listener_host': '',
    'mod_wsgi.listener_port': '443',
    'mod_wsgi.process_group': '',
    'mod_wsgi.request_handler': 'wsgi-script',
    'mod_wsgi.script_reloading': '1',
    'mod_wsgi.version': (3, 3),
    'wsgi.errors': <mod_wsgi.Log object at 0x1956b88>,
    'wsgi.file_wrapper': <built-in method file_wrapper of mod_wsgi.Adapter object at 0x21cf0f8>,
    'wsgi.input': <mod_wsgi.Input object at 0x1961048>,
    'wsgi.multiprocess': True,
    'wsgi.multithread': True,
    'wsgi.run_once': False,
    'wsgi.url_scheme': 'https',
    'wsgi.version': (1, 1)}>

    FILES:
    <MultiValueDict: {}>

疑わしいですが、おそらく私の wsgi ファイル、または私の Apache サイト ディレクティブに何か問題があります。念のため、それらを次に示します。

...wsgi:

import os
import sys

path = '/var'
if path not in sys.path:
   sys.path.append(path)

path = '/var/www'
if path not in sys.path:
   sys.path.append(path)

path = '/var/www/site'
if path not in sys.path:
   sys.path.append(path)

os.environ['DJANGO_SETTINGS_MODULE'] = 'site.settings'

import django.core.handlers.wsgi
application = django.core.handlers.wsgi.WSGIHandler()

Apache サイト対応ディレクティブ:

<VirtualHost 10.1.10.12:80>
    ServerName ...
    RewriteEngine on
    RewriteCond %{HTTPS} !=on
    RewriteRule ^/(.*) https://%{SERVER_NAME}/$1 [R,L]
</VirtualHost>


<Directory /var/www/site>
    RewriteEngine on
    RewriteCond 1 =1
    RewriteRule ^/(.*) https://%{SERVER_NAME}
</Directory>


<VirtualHost 10.1.10.12:443>
    ServerName ...
    DocumentRoot /var/www/site
    ServerAlias ...
    ErrorLog /var/log/apache2/site-error.log
    LogLevel warn
    CustomLog /var/log/apache2/site-access.log combined
    WSGIScriptAlias / /var/www/site/site.wsgi
    SSLEngine on
    SSLCertificateFile ...
    SSLCertificateKeyFile ...

    # Media serving, must specify which files to serve and not
    # hand their urls to the django framework.
    Alias /robots.txt .../robots.txt
    Alias /favicon.ico .../favicon.ico

    Alias /templates/ .../django_templates/
    Alias /static/ .../django_static/
    Alias /images/ .../django_static/images/

    <Directory /usr/local/lib/python2.6/dist-packages/django/contrib/admin/media/>
            Order deny,allow
            Allow from all
    </Directory>

    <Directory .../django_static/>
            Order deny,allow
            Allow from all
    </Directory>

    <Directory /usr/local/lib/python2.6/dist-packages/django/contrib/admin/media/js>
            Order allow,deny
            Allow from all
    </Directory>

    <Directory .../uploads/>
            Order allow,deny
            Allow from all
    </Directory>

    XSendFile on

4

2 に答える 2

2

おそらく、urls.py または正規表現についてもっと読む必要があります。誤ってフォームを開始スラッシュなしで投稿先に変更してしまいaction="uploaded_test"、request.method が "POST" である 404 エラー レポートを受け取りました (何 ?!@#$*&)...??? これは、それが私の投稿 URL に一致するものでなければならないことを意味していました! 私はそれを次のように持っていました:

'^uploaded_test/+$'

私はそれを次のように変更しました:

'^uploaded_test$'

フォームが投稿されていることを確認し、action="/uploaded_test"機能しました! urls.py を次のように変更しました。

'^uploaded_test/$'

投稿するフォームを変更したaction="/uploaded_test/"ところ、うまくいきました!多くのエントリで /loc または /loc/ のいずれかを受け入れることができるように、ほとんどの URL エントリに /+ を含めました。私は今、それらを元に戻すと思います。/+これがバグなのか、正規表現の構文について理解していないものなのかはわかりません。

于 2012-04-06T17:26:35.413 に答える
1

プレゼンテーション層で if-else ステートメントを試しましたか? たとえばif request.method == 'POST':、コードの残りの部分。

于 2012-04-06T17:15:15.237 に答える