Skip to content
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

dart:io HttpClient's request force-transforms the header names to lowercase #33501

Closed
isoos opened this issue Jun 19, 2018 · 53 comments
Closed
Assignees
Labels
area-core-library SDK core library issues (core, async, ...); use area-vm or area-web for platform specific libraries. library-io P2 A bug or feature request we're likely to work on type-enhancement A request for a change that isn't a bug

Comments

@isoos
Copy link

isoos commented Jun 19, 2018

The toLowerCase() transformation may be useful when processing the response headers, but when sending, the server could be checking the cased version and fail if it doesn't find it that way. (Yeah, bad server.)

@zoechi
Copy link
Contributor

zoechi commented Jun 20, 2018

Sounds like #25120

@isoos
Copy link
Author

isoos commented Jun 20, 2018

Yeah, very similar problem.

@anders-sandholm anders-sandholm added area-core-library SDK core library issues (core, async, ...); use area-vm or area-web for platform specific libraries. library-io labels Jun 21, 2018
@maguilar2k
Copy link

If the RFC-2616 states that header names are case-insensitive... why force the header names transformation to lowercase? shouldn't that be a decision taken by the developer?

@isoos
Copy link
Author

isoos commented Aug 26, 2019

I think the reason for the lowercase transform is HTTP/2: it standardizes all header names to be lowercase, and with that it makes somewhat sense to do the same here.

@shaxxx
Copy link

shaxxx commented Sep 6, 2019

I've spent half a day debugging to find out http client forces headers as lowercase.
Guess what, I'm stuck with 10 year old client devices with HTTP server's using case sensitive headers.
Why force anything? Let me decide what I want to send with MY request

@holly-hacker
Copy link

Same here, I ended up copying over a bunch of code from the SDK after debugging a broken WebSocket connection for a few days. I get the rationale but this isn't documented anywhere which is really annoying.

@shaxxx
Copy link

shaxxx commented Sep 9, 2019

Since I can't say to thousands of users to update their receivers or fix their HTTP server implementation, and I don't expect Dart team to change this behavior, I had no other choice but to change default HTTP client.

I took current HttpClient from master branch (requires Dart 2.5, Flutter 1.8+), changed it to leave headers alone and packed in the package. Hopefully it will save someone someone from crying and pulling up their hair. Just use it as a drop in replacement for HttpClient. Tested and works with Dio.

You can find package HERE.

@a-siva
Copy link
Contributor

a-siva commented Sep 17, 2019

There seems to be a suggestion that we should add an option to not change the case.

@zichangg
Copy link
Contributor

@lrhn Any idea? This would be a breaking change if adding one more flag.

@zichangg
Copy link
Contributor

To be more specific, A flag is probably needed to be passed from httpclient all the way down to httpheader.

_HttpClient(this._context);

_HttpOutboundMessage(Uri uri, String protocolVersion, _HttpOutgoing outgoing,

_HttpClientRequest(_HttpOutgoing outgoing, Uri uri, this.method, this._proxy,

Inside headers,

List<String> operator [](String name) => _headers[name.toLowerCase()];

name = name.toLowerCase();

name = name.toLowerCase();

@shaxxx
Copy link

shaxxx commented Sep 18, 2019

Thats the one way to look at it.
Why would HttpClient assume anything?
You have bunch of Abstract classes in HTTP client that are used only as internal implementations. Take HttpHeaders for instance. It's abstract, and HttpClient uses one single concrete implementation, which is internal, and has no way to use some other implementation.
Why making abstract class in a first place if you're forcing it to be used in a single way?
Good and probably non breaking start would be to

  • move internal implementations to separate public classes/files
  • refactor existing calls to new classes
  • add a way for user to provide specific implementation of abstract class (not just HttpHeaders)
  • if no specific implementation is provided use default one

Last two can be done as optional class constructor with default value params (better) or as public properties (less transparent IMHO).
I'm speaking from top of my head now, but this shouldn't breake anything.
Dart has no reflection, and internal calls can be easily refactored.
And once you're there, in default HttpHeaders expose map of all headers as public property, leave current behaviour, as is, but let user modify the map directly however he/she wants. That's extra, I'd be fine if I could make HttpClient use my HttpHeaders instead default ones.

Just my 2 cents.

@songfei
Copy link

songfei commented Nov 25, 2019

any update?

@zichangg
Copy link
Contributor

Still in progress. I expect this cl to be landed within next few weeks.

@sortie sortie added P2 A bug or feature request we're likely to work on type-enhancement A request for a change that isn't a bug labels Dec 10, 2019
@shaxxx
Copy link

shaxxx commented Dec 27, 2019

Is there any ETA when this will land in a Dart release?

@zichangg
Copy link
Contributor

The cl is pretty much done. But it has to wait for breaking change request getting approved.

@cdm2012
Copy link

cdm2012 commented Dec 29, 2019

Creating an API Client using HTTPClient with the following request headers:

I/flutter (32611): headers: {
I/flutter (32611):   "Content-type": "application/json",
I/flutter (32611):   "Accept": "application/json",
I/flutter (32611):   "Authorization": "Bearer: E7HlYmfiXwhLQT1nYApOAE4wqhvCZZXndrT7kiRv8tUBycj3npVnYwnQnH34"
I/flutter (32611): }
I/flutter (32611): 
I/flutter (32611): 
I/flutter (32611): 
I/flutter (32611): request.headers: 
I/flutter (32611): content-type: application/json
I/flutter (32611): accept: application/json
I/flutter (32611): authorization: Bearer: E7HlYmfiXwhLQT1nYApOAE4wqhvCZZXndrT7kiRv8tUBycj3npVnYwnQnH34

And the server (nginx/1.15.8) is returning a 400 error:

I/flutter (32611): <html>
I/flutter (32611): <head><title>400 Bad Request</title></head>
I/flutter (32611): <body>
I/flutter (32611): <center><h1>400 Bad Request</h1></center>
I/flutter (32611): <hr><center>nginx/1.15.8</center>
I/flutter (32611): </body>
I/flutter (32611): </html>

Using the same headers in android with the following request headers:

E: Called doInBackground w/ headers: [Accept: application/json, Content-type: application/json, Authorization: Bearer E7HlYmfiXwhLQT1nYApOAE4wqhvCZZXndrT7kiRv8tUBycj3npVnYwnQnH34]

And the same server is returning a 200 response, so I'm really looking forward to integrating the change by @zichangg : #39657 (comment)

@accek
Copy link

accek commented Dec 29, 2019

I/flutter (32611):   "Authorization": "Bearer: E7HlYmfiXwhLQT1nYApOAE4wqhvCZZXndrT7kiRv8tUBycj3npVnYwnQnH34"

It looks like the colon : after Bearer should not be there.

@cdm2012
Copy link

cdm2012 commented Dec 29, 2019

Thanks @accek, that was definitely an oversight. I just tried it again with these headers:

I/flutter ( 3473): headers: {
I/flutter ( 3473):   "Content-type": "application/json",
I/flutter ( 3473):   "Accept": "application/json",
I/flutter ( 3473):   "Authorization": "Bearer E7HlYmfiXwhLQT1nYApOAE4wqhvCZZXndrT7kiRv8tUBycj3npVnYwnQnH34"
I/flutter ( 3473): }
I/flutter ( 3473): 
I/flutter ( 3473): 
I/flutter ( 3473): 
I/flutter ( 3473): request.headers: 
I/flutter ( 3473): content-type: application/json
I/flutter ( 3473): accept: application/json
I/flutter ( 3473): authorization: Bearer E7HlYmfiXwhLQT1nYApOAE4wqhvCZZXndrT7kiRv8tUBycj3npVnYwnQnH34

But the result was still the same:

I/flutter ( 3473): responseString: <html>
I/flutter ( 3473): <head><title>400 Bad Request</title></head>
I/flutter ( 3473): <body>
I/flutter ( 3473): <center><h1>400 Bad Request</h1></center>
I/flutter ( 3473): <hr><center>nginx/1.15.8</center>
I/flutter ( 3473): </body>
I/flutter ( 3473): </html>

I didn't think to add the nginx logs before, but here are all the nginx logging from last night:

2019/12/29 03:15:05 [error] 3653#3653: *65 open() "/var/www/reely-api/testing/public/api/version" failed (2: No such file or directory), client: 10.0.0.5, server: reely.the-mac.local, request: "POST /api/version HTTP/1.1", host: "reely.the-mac.local"

Currently there are no logs showing, but I think that error was from using the wrong url protocol.

@sortie
Copy link
Contributor

sortie commented Jan 6, 2020

@accek We're working to make the breaking change, but it's a workaround for bad servers. The standard requires the protocol headers are case insensitive and you get the server fixed instead of debugging the issue here. I assume those bearer tokens are for debugging and have no actual access.

@cdm2012
Copy link

cdm2012 commented Jan 6, 2020

Yes the token is not usable @accek , it only worked for a private internal server. Could the solution be to use the HTTP/2 standard optionally @isoos , similar to @zichangg 's solution (but an opt in solution)?

@sortie I looked into the Server side, and according to the creator of Laravel, Laravel literally does not touch request headers in any form or fashion.

dart-bot pushed a commit that referenced this issue Jan 31, 2020
This is a breaking change. Request: #33501

HttpHeaders use lowercase by default for all headers, since it is supposed to be case insensitive. Some servers incorrectly treat case as significant, however, and expect headers with capitalization or in uppercase. The current implementation forces headers to be lower cases when adding values. Users cannot even manually modify the headers.

This change removes this restriction here so that users can modify the headers to whatever form they want. The new behavior is backwards compatible except if class was implemented. All headers inside http.dart are written as lower cases, adding values to HttpHeaders is still receiving lower cases input.

The other cl (https://dart-review.googlesource.com/c/http_multi_server/+/121411) migrates multi_headers.dart to be compatible with this change.

Bug: #33501
Change-Id: I6f7f2ef907b229773c283140c07f2de4cd500981
Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/119100
Commit-Queue: Zichang Guo <[email protected]>
Reviewed-by: Lasse R.H. Nielsen <[email protected]>
dart-bot pushed a commit that referenced this issue Jan 31, 2020
This reverts commit b2b7337.

Reason for revert: flutter build broke!!

Original change's description:
> [dart:io] Stop forcing lower case on HttpHeaders
> 
> This is a breaking change. Request: #33501
> 
> HttpHeaders use lowercase by default for all headers, since it is supposed to be case insensitive. Some servers incorrectly treat case as significant, however, and expect headers with capitalization or in uppercase. The current implementation forces headers to be lower cases when adding values. Users cannot even manually modify the headers.
> 
> This change removes this restriction here so that users can modify the headers to whatever form they want. The new behavior is backwards compatible except if class was implemented. All headers inside http.dart are written as lower cases, adding values to HttpHeaders is still receiving lower cases input.
> 
> The other cl (https://dart-review.googlesource.com/c/http_multi_server/+/121411) migrates multi_headers.dart to be compatible with this change.
> 
> Bug: #33501
> Change-Id: I6f7f2ef907b229773c283140c07f2de4cd500981
> Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/119100
> Commit-Queue: Zichang Guo <[email protected]>
> Reviewed-by: Lasse R.H. Nielsen <[email protected]>

[email protected],[email protected],[email protected],[email protected]

Change-Id: I4d4299393ad6549b250053df8823e726855e2baf
No-Presubmit: true
No-Tree-Checks: true
No-Try: true
Bug: #33501
Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/134102
Reviewed-by: Zichang Guo <[email protected]>
Commit-Queue: Zichang Guo <[email protected]>
@MiriDevAndro
Copy link

The flutter channel dev dart sdk 2.8.0-dev.15.0 has to much bug and stable has ver dart sdk 2.7.0 max

@mit-mit
Copy link
Member

mit-mit commented Mar 20, 2020

Sorry, you will have to wait for the dev channel to stabilize then.

@MiriDevAndro
Copy link

ok thnx for explanation

@eirikb
Copy link

eirikb commented Mar 31, 2020

I'm also struggling with a server only accepting cased headers.
How can I use the above solution?

Tried:

final body = '{"hello":"world"}';
final client = new HttpClient();
final req = await client.put('localhost', 8000, '/');
req.headers.add('Content-Type', 'application/json', preserveHeaderCase: true);
req.headers.add('Content-Length', body.length, preserveHeaderCase: true);
req.write(body);
await req.close();

Versions:

> flutter --version
Flutter 1.15.21 • channel master • /home/eirikb/.cache/yay/flutter-git/flutter
Framework • revision e2b4edd286 (3 weeks ago) • 2020-03-13 02:46:02 -0400
Engine • revision d7a00b8b09
Tools • Dart 2.8.0 (build 2.8.0-dev.14.0 eb9c26bd37)

But headers are still lower-cased.
Tested by running Flutter in Android Emulator and dart in terminal:

> /opt/flutter/bin/cache/dart-sdk/bin/dart --version
Dart VM version: 2.8.0-dev.14.0.flutter-eb9c26bd37 (Fri Mar 13 01:41:41 2020 +0000) on "linux_x64"
> /opt/flutter/bin/cache/dart-sdk/bin/dart api_test.dart

Output from nc (also see same result in Wireshark):

> nc -l -p 8000
PUT / HTTP/1.1
user-agent: Dart/2.8 (dart:io)
content-type: application/json
accept-encoding: gzip
content-length: 17
host: localhost:8000

{"hello":"world"}

Printing headers show they were correctly set in HttpHeaders, but not sent correctly:

print(req.headers);

user-agent: Dart/2.8 (dart:io)
Content-Type: application/json
accept-encoding: gzip
Content-Length: 17
host: localhost:8000

dart-bot pushed a commit that referenced this issue Apr 9, 2020
Bug: #33501
Change-Id: I57d9bba251b76314bf40b81d1b09bd4643dce4d2
Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/141911
Commit-Queue: Zichang Guo <[email protected]>
Reviewed-by: Siva Annamalai <[email protected]>
Reviewed-by: Jonas Termansen <[email protected]>
dart-bot pushed a commit that referenced this issue Apr 9, 2020
This reverts commit af19f96.

Reason for revert: Breaks package:shelf tests.

Original change's description:
> [dart:io] Preserve header case in http header _builds()
> 
> Bug: #33501
> Change-Id: I57d9bba251b76314bf40b81d1b09bd4643dce4d2
> Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/141911
> Commit-Queue: Zichang Guo <[email protected]>
> Reviewed-by: Siva Annamalai <[email protected]>
> Reviewed-by: Jonas Termansen <[email protected]>

[email protected],[email protected],[email protected]

Change-Id: I9c0418a256cb53e415ed0d5aeab84c4b0b4a161d
No-Presubmit: true
No-Tree-Checks: true
No-Try: true
Bug: #33501
Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/142980
Reviewed-by: David Morgan <[email protected]>
Commit-Queue: David Morgan <[email protected]>
@shaxxx
Copy link

shaxxx commented May 19, 2020

@mit-mit I'm having exactly the same issue as @eirikb .
preserveHeaderCase as far as I can see is currently not working.
Using

Dart VM version: 2.8.2 (stable) (Mon May 11 15:06:42 2020 +0200) on "windows_x64"

Can someone please explain what's current status on this, is it working, if yes how, if not what's happening with it? Should this be re-opened?

@zichangg
Copy link
Contributor

@shaxxx Sorry for the delayed reply. Code reviews for my fix took some time, I'll land the fix today.

@shaxxx
Copy link

shaxxx commented May 19, 2020

@zichangg Thanks for the update, as far I can understand preserveHeaderCase should work in the next flutter stable?
It wasn't really clear from this thread if this is currently working or not since no one confirmed this is still an issue and this bug is closed. Is there another bug tracking progress on this?

@a-siva a-siva reopened this May 19, 2020
@a-siva
Copy link
Contributor

a-siva commented May 19, 2020

reopening issue so that we can track when the bug fix CL lands and update on when it gets rolled into Flutter. You will probably see it in the dev channel first.

dart-bot pushed a commit that referenced this issue May 19, 2020
Outgoing HttpHeaders will use _build() to build headers. This cl make
_build() uses case-sensitive HttpHeaders.

Original change's description:
> [dart:io] Preserve header case in http header _builds()
>
> Bug: #33501
> Change-Id: I57d9bba251b76314bf40b81d1b09bd4643dce4d2
> Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/141911
> Commit-Queue: Zichang Guo <[email protected]>
> Reviewed-by: Siva Annamalai <[email protected]>
> Reviewed-by: Jonas Termansen <[email protected]>

Bug: #33501
Change-Id: I2d42351b9593d86ad1b2ea248faaea625d321e96
Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/143081
Commit-Queue: Zichang Guo <[email protected]>
Reviewed-by: Siva Annamalai <[email protected]>
@talregev
Copy link

talregev commented Jul 1, 2021

reopening this issue because force lowercase http header make it breakable from legacy server that don't check for small letter. and http1 header is sensitive case.
Please make by default non force lowercase http header, and add an option to people to want to force it.
It break my communication with the server.
Also http in web, doing as http should. it non force lower case and adding automatically content-length in the header.
But, http in dart-io force you lowercase http header and not adding automatically content-length.
This is bad inconsistency. http request should be the same across all the platform. and doing the same as http in web.

@MiriDevAndro
Copy link

MiriDevAndro commented Jul 1, 2021

@talregev im okay with you , they should insert support for http 1.1

@cesuper
Copy link

cesuper commented Sep 4, 2021

Hey,

I also miss the case sensitive version of the headers. Any update in this topic?

@talregev

This comment has been minimized.

@cesuper

This comment has been minimized.

@talregev

This comment has been minimized.

@mraleph
Copy link
Member

mraleph commented Sep 6, 2021

@cesuper @talregev this issue was closed because HttpHeaders add and set methods were expanded with explicit support for case-sensitive header via preserveHeaderCase: true named argument.

If this does not work for you (meaning: you set preserveHeaderCase: true but headers are sent in forced lowercase) then you should file a separate issue and provide a minimal reproduction of the issue.

There are no plans to switch to case-sensitive headers by default.

@MiriDevAndro
Copy link

Any update on this?
I think that this bug is on native compiled library, any idea ?

@mraleph
Copy link
Member

mraleph commented Nov 8, 2021

@MiriDevAndro

This comment has been minimized.

@mraleph

This comment has been minimized.

@MiriDevAndro

This comment has been minimized.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area-core-library SDK core library issues (core, async, ...); use area-vm or area-web for platform specific libraries. library-io P2 A bug or feature request we're likely to work on type-enhancement A request for a change that isn't a bug
Projects
None yet
Development

No branches or pull requests