3

wtforms を使用して登録フォームを作成しました。フォームの一部の要素を再度繰り返す必要がないように、その中で FormField を使用しています。しかし、送信ボタンをクリックするたびに、validate_on_submit メソッドの呼び出しで常に false になります。なぜこれが起こっているのかわかりません。

私のform.pyは次のとおりです。

class ProfileInfoForm(Form):
    firstname = TextField('firstname', validators=
                          [validators.Required("Please enter First name.")])
    lastname = TextField('lastname', validators=
                         [validators.Required("Please enter Last name.")])
    email = EmailField('email', validators=
                       [validators.Required("Please enter your valid email.")])
    gender = RadioField('gender', validators=
                        [validators.Required("Please select gender")],
                        choices=[('female', 'Female'), ('male', 'Male')])
    dob = TextField('dob', validators=
                    [validators.Required("Please select date of birth.")])
    languages = SelectMultipleField('languages', choices=[('', '')],
                                    validators=
                                    [validators.Required("Please select\
                                                         atleast one \
                                                         language.")])


class RegistrationForm(Form):
    profilefield = FormField(ProfileInfoForm)
    password = PasswordField('password',
                             validators=
                             [validators.Required("Please enter password."),
                              validators.Length(min=8),
                              validators.EqualTo('confirm_password',
                                                 message='Password and confirm\
                                                 password must match')])
    confirm_password = PasswordField('confirm_password',
                                     validators=
                                     [validators.Required("Please enter\
                                                          confirm password.")])
    tnc = BooleanField('tnc', validators=
                       [validators.Required("Please select Terms and \
                                            Conditions")], default=False)

    submit = SubmitField('Create My Account')

Signup方法は次のとおりです。

@module.route('/signup', methods=['GET', 'POST'])
  @handle_error
  def signup():
      if hasattr(g, 'user') and g.user:
          # TODO: do some operations if needed else keep it blank
          return redirect(url_for('index'))
      else:
          signup_form = RegistrationForm()
          # Add choices for the user
          signup_form.profilefield.languages.choices = getLanguages()
          if signup_form.validate_on_submit():
              firstname = signup_form.profilefield.firstname.data
              lastname = signup_form.profilefield.lastname.data
              email = signup_form.profilefield.email.data
              password = signup_form.password.data
              #  confirm_password = signup_form.confirm_password.data
              gender = signup_form.profilefield.gender.data
              dob = signup_form.profilefield.dob.data
              languages = signup_form.profilefield.languages.data
              tnc = signup_form.tnc.data

              payload = {'firstname': firstname, 'lastname': lastname,
                         'email': email, 'password': password, 'gender': gender,
                         'dob': dob, 'languages': languages,
                         'tnc': ('1' if tnc else '0')}
              try:
                  buildApiUrl = BuildApiUrl()
                  response = requests.post(buildApiUrl.getUrl("user", "signup"),
                                           data=payload)

                  if response.status_code == requests.codes.ok:
                      data = json.loads(response.text)
                      if 'status' in data and data['status'] != 200:
                          flash(data['message'], category="error")
                      else:
                          flash(data['message'] +
                                ': Your account is created successfully! ' +
                                'Please login to your account!',
                                category="success")
                          return redirect(url_for('index'))
              except requests.exceptions.RequestException:
                  flash('Internal Server side error occured', category="error")
                  return redirect(url_for('server_error', e='500'))

      return render_template('public/index.html',
                             signup_form=signup_form, login_form=LoginForm())

HTMLフォームはgist hereにあります

参考までに: 必要なすべてのフィールドに実際に必要なデータを入力しています。validate_on_submit() を呼び出すと、まだ false になります。私のコードで何が間違っていますか?

編集: getLanguages は、データベースから言語を取得し、選択リストに入れるメソッドです。この機能は期待どおりに機能しており、言語のリストを取得できます。

編集 2: ここで 1 つのことを実現します。ProfileInfoForm() のすべてのフィールドを RegistrationForm() メソッドに追加してテストしたため、これは FormField が原因で発生しており、すべてが正常に機能し、サインアップできました。したがって、FormField または使用方法に問題がありますが、どこが間違っているのかわかりません。

問題は FormField ではなく、私の ProfileInfoForm() にあることがわかりました。常に false を返します。まだ理由はありませんが、その点については独自の検証を作成する必要があると思います。何かご意見は?

編集:

ダンプで次のようになりました(ここではpprintを使用):

{'SECRET_KEY': '1e4c35233e50840483467e8d6cfe556c',
 '_errors': None,
 '_fields': {'csrf_token': <wtforms.ext.csrf.fields.CSRFTokenField object at 0x2207290>,
             'dob': <wtforms.fields.simple.TextField object at 0x2207650>,
             'email': <flask_wtf.html5.EmailField object at 0x22074d0>,
             'firstname': <wtforms.fields.simple.TextField object at 0x2207350>,
             'gender': <wtforms.fields.core.RadioField object at 0x2207590>,
             'languages': <wtforms.fields.core.SelectMultipleField object at 0x2207710>,
             'lastname': <wtforms.fields.simple.TextField object at 0x2207410>},
 '_prefix': u'profilefield-',
 'csrf_enabled': True,
 'csrf_token': <wtforms.ext.csrf.fields.CSRFTokenField object at 0x2207290>,
 'dob': <wtforms.fields.simple.TextField object at 0x2207650>,
 'email': <flask_wtf.html5.EmailField object at 0x22074d0>,
 'firstname': <wtforms.fields.simple.TextField object at 0x2207350>,
 'gender': <wtforms.fields.core.RadioField object at 0x2207590>,
 'languages': <wtforms.fields.core.SelectMultipleField object at 0x2207710>,
 'lastname': <wtforms.fields.simple.TextField object at 0x2207410>}

編集:

少し掘り下げたところ、csrfトークンが見つからないためにエラーが生成されることがわかりました。しかし {{ signup_form.hidden_tag() }}、HTML のフォーム テンプレートに含めました。要素を検査すると生成されたhtmlの隠しタグが表示され、ハッシュ値を持つcsrf_tokenフィールドが表示されます。では、ここで何が問題なのですか?

4

2 に答える 2

5

次の関数で問題を解決しました。

def __init__(self, *args, **kwargs):
    kwargs['csrf_enabled'] = False
    super(ProfileInfoForm, self).__init__(*args, **kwargs)

この機能を追加しましたProfileInfoForm()

この問題は、実際のフォームだけでなくフィールドもFormField含まれていました。つまり、 csrf_token も含まれていたため、検証する必要があるのは 2 つあり、実際にフォームにレンダリングされたのは 1 つだけでした。そのため、無効にしたため、FormField がレンダリングしたときに.csrf_tokenRegistrationFormcsrf_tokencsrf_tokenProfileInfoFormcsrf_token = False

RegistrationForm現在も有効になっているため、csrf_tokenフォームはまだ安全です。

私の推測では、これも同様に行う必要がありFormFieldます。

参考: FormField コードの私の解釈により、この解決策は間違っている可能性があります。上記の解決策で間違っている場合は修正してください。

于 2013-09-13T14:19:21.723 に答える