使用 Next.js 发出 API POST 请求时遇到问题

问题描述 投票:0回答:1

对于与后端相关的任何事物,我都是新的,对 Next.js 和 TypeScript 也是新的。 我正在尝试向 API 发出 POST 请求,该请求应该接收

formData
并基于该
formData
创建一个新列表。我正在尝试使用 Next.js 服务器操作和路由处理程序来做到这一点。

这是我的代码:

这是来自表单组件:

const handleSubmit = async (e: React.FormEvent<HTMLFormElement>) => {
    e.preventDefault();

    const form = e.currentTarget as HTMLFormElement;
    const formData = new FormData(form);
    
    const address = formData.get("address");
    const mail = formData.get("mail");
    const region_id = selectedRegion?.id; //these come from states
    const agent_id = selectedAgent?.id;  //these come from states
    const city_id = selectedCity?.id;   //these come from states
    const area = formData.get("area");
    const price = formData.get("price");
    const type = formData.get("type");
    const cover = formData.get("cover"); // This is the file input

    try {
        console.log(address, mail, region_id, agent_id, city_id, area, price, type, cover);

        const newFormData = new FormData();

        newFormData.append("address", address as string);
        newFormData.append("mail", mail as string);
        newFormData.append("region_id", region_id as any);
        newFormData.append("agent_id", agent_id as any);
        newFormData.append("city_id", city_id as any);
        newFormData.append("area", area as string);
        newFormData.append("price", price as string);
        newFormData.append("type", type as string);
        newFormData.append("cover", cover as File); 
        
        await addListingAction(newFormData)
    } catch (error) {
        alert(`Error: ${error}`);
    }
};

这是我的服务器操作:

export async function addListingAction(formData: FormData) {
  try {
    await addListing(formData);
  } catch (error) {
    console.log(error);
  } finally {
    revalidatePath("/")
  }
}

我的 api.ts 文件:

export async function addListing(formData: FormData) {

  return await fetch(`${process.env.NEXT_PUBLIC_API_URL}/api/add-listing`, {
    method: 'POST',
    body: formData,
  })
    .then(response => response.json())
    .then(data => console.log(data))
    .catch(error => console.error('Error:', error));
}

和我的route.ts文件:

import { NextResponse } from 'next/server';

export async function POST(req: Request) {
  try {
    const formData = await req.formData();

    const address = formData.get("address");
    const mail = formData.get("mail");
    const region_id = formData.get("region_id");
    const agent_id = formData.get("agent_id");
    const city_id = formData.get("city_id");
    const area = formData.get("area");
    const price = formData.get("price");
    const type = formData.get("type");
    const cover = formData.get("cover") as File | null;

    const form = new FormData();
    if (address) form.append("address", address);
    if (mail) form.append("mail", mail);
    if (region_id) form.append("region_id", region_id);
    if (agent_id) form.append("agent_id", agent_id);
    if (city_id) form.append("city_id", city_id);
    if (area) form.append("area", area);
    if (price) form.append("price", price);
    if (type) form.append("type", type);
    if (cover instanceof File) form.append("cover", cover);

    const REDBERRY_API_TOKEN = process.env.REDBERRY_API_TOKEN;

    if (!REDBERRY_API_TOKEN) {
      throw new Error("Missing API token");
    }


    const response = await fetch('https://api.real-estate-manager.redberryinternship.ge/api/real-estates', {
      method: 'POST',
      headers: {
        'Authorization': `Bearer ${REDBERRY_API_TOKEN}`,
      },
      body: form,
    });


if (!response.ok) {
  let errorResponse;
  try {
    errorResponse = await response.json();
  } catch {
    errorResponse = await response.text();
  }
  console.error('HTTP error! Status:', response.status, 'Response:', errorResponse);
  return NextResponse.json(
    { message: 'API request failed', error: errorResponse },
    { status: response.status }
  );
}


  const listing = await response.json();

  return NextResponse.json({ listing }, { status: 201 });

  } catch (error) {
    const message = error instanceof Error ? error.message : 'An unexpected error occurred';
    console.error('Error creating listing:', message);
  return NextResponse.json(
    { message: 'Failed to create listing', error: message },
    { status: 500 }
  );
  }
}

API文档说curl请求看起来像这样:

curl -X 'POST' \
  'https://api.real-estate-manager.redberryinternship.ge/api/real-estates' \
  -H 'accept: application/json' \
  -H 'Authorization: Bearer API_TOKEN' \
  -H 'Content-Type: multipart/form-data' \
  -F 'region_id=1' \
  -F 'price=100000' \
  -F 'zip_code=0101' \
  -F 'area=100.5' \
  -F 'city_id=1' \
  -F 'address=example address' \
  -F 'agent_id=371' \
  -F 'bedrooms=3' \
  -F 'is_rental=0' \
  -F '[email protected];type=image/png' \
  -F 'description=house near metro station'

响应如下所示:

"price": "100000",
  "zip_code": "0101",
  "area": "100.5",
  "city_id": "1",
  "address": "example description",
  "agent_id": "371",
  "bedrooms": "3",
  "is_rental": "0",
  "description": "house near metro station",
  "image": "https://api.real-estate-manager.redberryinternship.ge/storage/images/vZ6KRjLiBIoVqwl4GMniE598mAzno8wqzCkZyg3f.png",
  "created_at": "2024-09-15T16:43:21.000000Z",
  "id": 450

我收到的错误是:

创建列表时出错:意外的令牌'<', " POST /api/add-listing 500 in 931ms
{
消息:“创建列表失败”,
错误:“意外的令牌”<', " }

任何帮助将不胜感激。

javascript reactjs typescript next.js
1个回答
0
投票

能够通过在我的表单中添加不可见的输入来解决问题,这将从

region_id
city_id
.... 的状态中获取值,而不是将这些值附加到
formData

© www.soinside.com 2019 - 2024. All rights reserved.