我已经经历了所有与此相关的问题,但我还没有找到一个对我有用的解决方案。
我在使用
retrofit 2.8.1
和
OkHttp 4.5.0
。
我的服务接口看起来如下
public interface MlApiService
@POST
@Multipart
Call<List<PreprocessedDocument>> postDocument( @Url String apiUrl, @Part MultipartBody.Part document,
@Part ( "document_id") RequestBody documentId );
}
然后像下面这样构建客户机,并将
requestTimeoutInSeconds
设置为90秒。
public void init()
GsonBuilder gson = new GsonBuilder();
gson.registerTypeAdapter( new TypeToken<List<PreprocessedDocument>>() {}.getType(), new CustomResponseDeserializer() );
HttpLoggingInterceptor logInterceptor = new HttpLoggingInterceptor();
logInterceptor.setLevel( HttpLoggingInterceptor.Level.HEADERS );
OkHttpClient client = new OkHttpClient.Builder().retryOnConnectionFailure( true ).addInterceptor( logInterceptor )
.readTimeout( requestTimeoutInSeconds, TimeUnit.SECONDS ).build();
//Dummy Base URL must be provided. otherwise client won't get initialized
Retrofit retrofit = new Retrofit.Builder().baseUrl( "http://thisIsJustDummyUrlForTheSakeOfAddingBaseUrl.com/" )
.client( client ).addConverterFactory( GsonConverterFactory.create( gson.setLenient().create() ) ).build();
mlApiService = retrofit.create( MlApiService.class );
}
请求到达服务器,当服务器响应时,我得到以下错误:
Caused by: java.io.IOException: unexpected end of stream on Connection{34.XXX.XXX.9:8085, proxy=DIRECT hostAddress=/34.XXX.XXX.9:8085 cipherSuite=none protocol=http/1.1}
at okhttp3.internal.http1.Http1Codec.readResponseHeaders(Http1Codec.java:203)
at okhttp3.internal.http.CallServerInterceptor.intercept(CallServerInterceptor.java:88)
at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:147)
at okhttp3.internal.connection.ConnectInterceptor.intercept(ConnectInterceptor.java:45)
Caused by: java.io.EOFException: \n not found: limit=0 content=…
at okio.RealBufferedSource.readUtf8LineStrict(RealBufferedSource.java:227)
at okio.RealBufferedSource.readUtf8LineStrict(RealBufferedSource.java:211)
at okhttp3.internal.http1.Http1Codec.readResponseHeaders(Http1Codec.java:187)
我迄今尝试过的几件事
API在postman中运行良好,但当我尝试使用代码时,它就失败了。所以我尝试了邮递员发送的同样的标题。还是没有运气。
很少有意见:
,
后,我立即得到错误信息。
编辑1: 我点击的服务器是由gunicorn/20.0.4支持的,并且使用了烧瓶。我没有访问服务器代码的权限。我怀疑接收到的响应是否存在导致错误的垃圾字符。我不知道如何记录原始响应,然后才能被retrofit/okhttp读取。
编辑2:
我详细地运行了Curl命令,这就是我得到的结果。
从server
的空回复
你会得到一个没有状态行的空回复。这就是问题所在。HTTP-请求通常返回一个状态行(例如
HTTP/1.1 200 OK\r\n
),其中包含状态代码(参见
https://www.ietf.org/rfc/rfc2616.txt
第6.1章)。这通常是服务器错误。
短篇小说
问题是我撞到的服务器。它没有发出任何回应(字面意思是什么都没有。)没有标题,没有身体,什么都没有)。
长话短说
因此,在翻阅了堆叠溢出、其他好网站的所有相关答案,并尝试了我在问题中提到的这么多解决方案之后,它并没有解决我的问题。
在仔细阅读堆栈跟踪之后,我遇到了下面的一行。
okhttp3.internal.http1.Http1Codec.readResponseHeaders(Http1Codec.java:203)
客户端(我的代码)正在尝试读取响应头,这时抛出了错误
java.io.EOFException: \n not found: limit=0 content=
。
这给了我一个提示,问题可能是服务器而不是客户端的问题。因此,我想我应该尝试与另一个客户,看看我是否能看到原始的反应。
我想到的第一个工具是
Curl
(邮递员用来给出通用的
无法得到任何响应
,而且这种情况并没有持续发生)。我点击服务器使用卷曲和详细的选项和砰!我得到了以下答复:
curl -v --location --request POST 'http://XX.XXX.XXX.9:8085/psc/document_upload' --form 'document=@/home/user376/Downloads/test-1.pdf' --form 'document_id=22004494_ae7f_4998_a1d8_73249bda9905'
Note: Unnecessary use of -X or --request, POST is already inferred.
* Trying XX.XXX.XXX.9...
* Connected to XX.XXX.XXX.9 (XX.XXX.XXX.9) port 8085 (#0)
> POST /psc/document_upload HTTP/1.1
> Host: XX.XXX.XXX.9:8085
> User-Agent: curl/7.49.0
> Accept: */*
> Content-Length: 4684053
> Expect: 100-continue