【问题标题】:Rapid API Error code 500 with Retrofit2 Android使用 Retrofit2 Android 的快速 API 错误代码 500
【发布时间】:2019-06-21 17:27:01
【问题描述】:

我在一个食谱应用项目中使用 spoonacular API。尝试向 API 发出多个 GET 请求时会出现此问题。第一个请求是带有查询参数的简单搜索。第一个请求的结果 JSON 包含一个配方 ID,我使用该 ID 发出第二个 GET 请求,出现问题。 API 仅在我第一次发出请求时才会响应,但之后它会以错误代码 500 [Internal Server Error] 进行响应。

我已经在 Postman 上测试了 GET 请求,但每次都可以正常工作。 我是使用 API 的新手,非常感谢任何帮助。

这是我的改造服务类

公共类 ServiceGenerator {

public static final String API_BASE_URL = "https://spoonacular-recipe-food-nutrition-v1.p.rapidapi.com/";

private static OkHttpClient.Builder httpClient = new OkHttpClient.Builder();

private static Retrofit.Builder builder =
        new Retrofit.Builder()
                .baseUrl(API_BASE_URL)
                .addConverterFactory(GsonConverterFactory.create());

private static Retrofit retrofit = builder.build();

public static <S> S createService(Class<S> serviceClass, final String HostName, final String KeyVal)
{
    if (!TextUtils.isEmpty(HostName) && !TextUtils.isEmpty(KeyVal))
    {
        HeadersInterceptor interceptor = new HeadersInterceptor(HostName,KeyVal);

        if (!httpClient.interceptors().contains(interceptor))
        {
            httpClient.addInterceptor(interceptor);
            builder.client(httpClient.build());
            retrofit = builder.build();
        }
    }

    return retrofit.create(serviceClass);
}

这是我用来在请求中添加标头的拦截器。

公共类 HeadersInterceptor 实现拦截器{

private String HostName,KeyVal;

HeadersInterceptor(final String HostName,final String KeyVal) {
    this.HostName = HostName;
    this.KeyVal = KeyVal;
}

@NotNull
@Override
public Response intercept(@NotNull Chain chain) throws IOException {
    Request original = chain.request();

    Request.Builder builder = original.newBuilder()
            .addHeader("X-RapidAPI-Host",HostName)
            .addHeader("X-RapidAPI-Key",KeyVal);

    Request request = builder.build();
    return chain.proceed(request);
}

}

这是我的片段,它进行搜索查询并成功返回结果[Receipe ID's]

公共类 ListSelectedFragments 扩展片段 {

private ProgressBar PreviewFragPrg;
private final String TAG = "ListSelectedFragment->";

private PreviewRecipeAdapter adapter;
private RecyclerView SelectedItemRV;
private ArrayList<RecipePreviewHolder> RecipePreviewsList = new ArrayList<>();

public ListSelectedFragments() {
    // Required empty public constructor
}


@Override
public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) {
    // Inflate the layout for this fragment
    final View view = inflater.inflate(R.layout.fragment_list_selected_fragments, container, false);
    SelectedItemRV = view.findViewById(R.id.SelectedItemRV);
    TextView DisplayNameTV = view.findViewById(R.id.DisplayNameTV);
    PreviewFragPrg = view.findViewById(R.id.PreviewFragPrg);
    ImageView BackBtn = view.findViewById(R.id.BackBtn);

    BackBtn.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            if (getFragmentManager() != null) {
                getFragmentManager().popBackStackImmediate();
            }
        }
    });

    if (getArguments() != null) {
        final String QueryTag = getArguments().getString("QueryTag");
        final String CuisineName = getArguments().getString("CuisineName");

        if(CuisineName!=null){
            DisplayNameTV.setText(CuisineName);
        }
        if(QueryTag!=null){
            ProcessQuery(QueryTag);
        }
    }

    return view;
}

private void ProcessQuery(final String QueryStr){

    String hostname = getResources().getString(R.string.spoonacular_host_name);
    String key      = getResources().getString(R.string.spoonacular_apikey_val);


    final ServiceGenerator.GetDataService mService =
            ServiceGenerator.createService(ServiceGenerator.GetDataService.class, hostname,key);

    Call<RecipeInfoModel> call = mService.getRecipes(QueryStr);

    call.enqueue(new Callback<RecipeInfoModel>() {
        @Override
        public void onResponse(@NonNull Call<RecipeInfoModel> call,
                               @NonNull Response<RecipeInfoModel> response)
        {
            Log.d(TAG, "Request Response Received");
            Log.d(TAG, response.toString());

            if (response.body() != null) {
                Results[] mRES = response.body().getResults();
                SetUpRecipePreviews(mRES);
                PreviewFragPrg.setVisibility(View.GONE);
            }
        }

        @Override
        public void onFailure(@NonNull Call<RecipeInfoModel> call, @NonNull Throwable t) {
            Log.d(TAG, "Request Failed");
            Log.d(TAG, call.toString());
            Log.d(TAG, "Throwable ->" + t);
            PreviewFragPrg.setVisibility(View.GONE);
            Toast.makeText(getActivity(),"Could not get required recipes",Toast.LENGTH_SHORT).show();
        }
    });

    Log.d(TAG, "User Inputed Request\n"+call.request().url().toString());

}

private void SetUpRecipePreviews(final Results[] mRES) {

    RecipePreviewsList.clear();
    adapter = new PreviewRecipeAdapter(getActivity(),RecipePreviewsList);
    SelectedItemRV.setLayoutManager(new GridLayoutManager(getActivity(), 2));
    SelectedItemRV.setAdapter(adapter);

    for (Results mRE : mRES) {
        String ImgUrls = mRE.getImage();
        RecipePreviewHolder obj = new RecipePreviewHolder(Integer.valueOf(mRE.getId()),
                mRE.getTitle(), ImgUrls);
        Log.d("GlideLogs->","Rid->"+mRE.getId());
        Log.d("GlideLogs->","Img URL->"+ ImgUrls);
        Log.d("GlideLogs->","Name->"+mRE.getTitle());

        RecipePreviewsList.add(obj);
    }

    if(RecipePreviewsList.size()>1){
        adapter.notifyDataSetChanged();
    }
}

这是我在单击食谱卡后从我的片段转换到的活动...在附加组件中发送食谱 ID。此函数在收到 Intent Extras 后立即调用。

private void RetrieveRecipeInfo(final int recipeID) {

    String hostname = getResources().getString(R.string.spoonacular_host_name);
    String key   = getResources().getString(R.string.spoonacular_apikey_val);


    final ServiceGenerator.GetDataService mService =
            ServiceGenerator.createService(ServiceGenerator.GetDataService.class, hostname,key);

    Call<RecipeDetailedInfo> call = mService.getInformation(185071);
    Log.d(TAG , "Your GET Request:\n"+call.request().url().toString());

    call.enqueue(new Callback<RecipeDetailedInfo>() {
        @Override
        public void onResponse(@NonNull Call<RecipeDetailedInfo> call, @NonNull Response<RecipeDetailedInfo> response)
        {
            Log.d(TAG,"OnResponse()  Called\n");
            Log.d(TAG,"Response = "+ response);

            if(response.body()!=null) {
                String obj = response.body().getSourceUrl();
                Log.d(TAG,"Getting Recipe Info\n");
                Log.d(TAG, String.valueOf(obj));
            }
        }

        @Override
        public void onFailure(@NonNull Call<RecipeDetailedInfo> call, @NonNull Throwable t){
        }
    });
}

使用邮递员我每次都能得到结果,但在我的应用程序中,API 在第一次请求后停止响应。我包含标题的方式有问题吗?

【问题讨论】:

  • 第一个请求有效,但第二个请求失败。尝试使用硬编码的 ID 发出第二个请求,看看它是否有效。
  • 不走运。仍然得到同样的错误。这是响应 Response{protocol=http/1.1, code=500, message=Internal Server Error, url=spoonacular-recipe-food-nutrition-v1.p.rapidapi.com/recipes/…}
  • 您是否使用代理来检查来自您设备的请求?标题是否存在?请求正文是否正确(如果存在)等。
  • 我认为使用像Stetho 这样强大的拦截器来监控请求和响应的细节可以让你摆脱这个问题。这是一个逻辑问题。不是技术问题!
  • @MartinMarconcini 我正在使用简单的日志来查看来自我的设备的请求,并使用邮递员仔细检查响应。至于标头,我也很确定它们存在,因为如果它们没有随请求一起发送,API 会以 [headers not included] 响应。

标签: android retrofit2 internal-server-error http-status-code-500


【解决方案1】:

所以我终于把事情搞定了。问题出在 HeadersInterceptor.java 上。我正在使用拦截器在调用中添加标头,但我发现了一种更简单的方法,它就像一个魅力。

只需添加@Header 调用即可在Retrofit 中添加不带拦截器的标头。

public interface GetDataService {
    @GET("recipes/complexSearch?")
    Call<RecipeInfoModel> getRecipes(
            @Header("X-RapidAPI-Host") String api,
            @Header("X-RapidAPI-Key") String apiKey,
            @Query("query") String query_str);
}

【讨论】:

    猜你喜欢
    • 2023-04-05
    • 1970-01-01
    • 2016-01-07
    • 2015-03-05
    • 1970-01-01
    • 1970-01-01
    • 2014-08-22
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多