-
-
Notifications
You must be signed in to change notification settings - Fork 1.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
request.form empty for certain form POST request payloads #2734
Comments
Update: I recently updated the already-running weather station and now it has the same problem there with Werkzeug v2.0.3. |
In your second request example, you have I can't reproduce this issue, although I don't have a way to reproduce that typo. |
I have the same issue (which I logged here pallets/flask#5168 and also discussed with another weather station hacker here bentasker/Ecowitt_to_InfluxDB#1). I'm on Python v3.11.3, Flask v2.3.2 and Werkzeug v2.3.6. It seems we can deduce that something has changed with the format that the Ecowitt is sending in, but it also looks like this should be parsed correctly by Flask/Werkzeug I'm running an Ecowitt WS2910 with firmware version v5.1.1, and the top item in the firmware changelog is "Optimize the http server" 🤦♂️ |
Just to add, I've worked around this issue by placing an NGINX reverse proxy in front of Flask, with no special config, and it seems to magically fix the dodgy header in flight. Both NGINX and the http echo server I used for testing were able to accept and correct the header, so it feels like Flask should be able to, too. |
You still haven't provided a way to reproduce the issue locally, so I still can't do anything. Saying you "feel like Flask should be able" to handle it isn't enough. It sounds like you might be running the development server, and assuming that is "Flask". This would only be an issue with Werkzeug if it also happens when running a production server such as Gunicorn, Waitress, or uWSGI. And you'd still need to provide a reproducible example. The development server is a thin wrapper around Python's built-in |
Based on the description, this should be quite easy to repro - you just need to use something like netcat to place the request manually rather than However, I'm not so sure it is that missing space (and if it is, the problem is in the header parsing as spaces are optional, per RFC 7231). If it were the space, this should repro it Stand up flask from flask import Flask, request
app = Flask(__name__)
@app.route('/', methods=["POST"])
def hello():
return request.get_data(as_text=True)
app.run(host="0.0.0.0", port=8090, debug=True) If the issue were the space, we should get an empty response, but
That being said, I'm running Python 3.10.6, so perhaps that's why
@djjudas21 be worth seeing if that repro works for you though, there might be something different in my env. Just run a capture of traffic from my Ecowitt and minne also lacks the space on |
Got it! It's not visible in the description (for obvious reasons) but that content length header is not We have a repro:
So, the header parsing is non-RFC compliant (7231 says the value can be prefixed and suffixed by optional whitespace). Unless I've gone awry, Python's header parsing relies on the parsing in it's email class (here, originally called from here). Once we get down the callchain, the parsing class is here, but it ultimately calls If we take a copy of that and pass headers into it:
Notice that it hasn't stripped the trailing whitespace. This'll bubble all the way back up through the callchain until we're back in Werkzeug. When handling form parsing, Werkzeug calls That applies the following regex
Which will fail to match because of that trailing space
As a result, the response body isn't actually read. So, there are two things that can/could be done
|
I'll look at raising a PR for the first in a bit. @davidism let me know if you'd be willing to accept a PR for the second |
Considering this regex is meant to remove stuff that's allowed in |
PR to fix that is fine, probably just do |
…type This mitigates pallets#2734
…type This mitigates pallets#2734
PR posted here: #2738 I'll look at getting one into Python for the root cause soon (or possibly tomorrow) |
…type This mitigates pallets#2734
…type This mitigates pallets#2734
Closed by #2764 |
I have a weird one, and I admit I am not entirely sure if it's a problem in Werkzeug or maybe somewhere deeper:
I have an ecowitt weather station and am happily getting data pushed to a python script that works great. Recently someone else wanted to do the same, installed my script but... it just didn't work. We found that the only difference was that I was still on python 3.6.9 and Werkzeug 2.0.3, whereas he was on Python 3.9.2 and Werkzeug 2.3.6
What happens is that in the route handler, request.form is empty depending on how the form request looks in detail.
Here's the code snippet:
So far so easy, right? Now sending a POST request using a simple html form works fine as long as it looks like this:
With that request, request.form is filled and the data can be parsed normally.
However, the Ecowitt station sends requests that have a different set of header fields, which looks like this:
Sending that request results in an empty request.form. It's also no use to try and use request.get_data() or stuff like that. The method in both requests is POST.
What should happen: Both requests should work and request.form should be filled in both cases.
This works fine in Python 3.6.9 and Werkzeug 2.0.3
Environment:
Python v3.9.2
Werkzeug v2.3.6
The text was updated successfully, but these errors were encountered: