fastapi(starlette)RedirectResponse 重定向到 post 而不是获取方法 [重复]

2025-03-26 09:08:00
admin
原创
16
摘要:问题描述:返回 RedirectResponse 对象后,我遇到了奇怪的重定向行为事件.pyrouter = APIRouter() @router.post('/create', response_model=EventBase) async def event_create( reque...

问题描述:

返回 RedirectResponse 对象后,我遇到了奇怪的重定向行为

事件.py

router = APIRouter()

@router.post('/create', response_model=EventBase)
async def event_create(
        request: Request,
        user_id: str = Depends(get_current_user),
        service: EventsService = Depends(),
        form: EventForm = Depends(EventForm.as_form)
):
    event = await service.post(
       ...
   )
    redirect_url = request.url_for('get_event', **{'pk': event['id']})
    return RedirectResponse(redirect_url)


@router.get('/{pk}', response_model=EventSingle)
async def get_event(
        request: Request,
        pk: int,
        service: EventsService = Depends()
):
    ....some logic....
    return templates.TemplateResponse(
        'event.html',
        context=
        {
            ...
        }
    )

路由器.py

api_router = APIRouter()

...
api_router.include_router(events.router, prefix="/event")

此代码返回结果

127.0.0.1:37772 - "POST /event/22 HTTP/1.1" 405 Method Not Allowed

好的,我看到出于某种原因调用了 POST 请求而不是 GET 请求。我搜索了解释,发现 RedirectResponse 对象默认为代码 307 并调用 POST链接

我按照建议添加了状态

redirect_url = request.url_for('get_event', **{'pk': event['id']}, status_code=status.HTTP_302_FOUND)

并得到

starlette.routing.NoMatchFound

为了实验,我将@router.get('/{pk}', response_model=EventSingle)改为@router.post('/{pk}', response_model=EventSingle)

重定向成功完成,但此处的发布请求不适合我。我做错了什么?

更新型多巴胺

用于运行事件/创建逻辑的 html 表单

基础.html

<form action="{{ url_for('event_create')}}" method="POST">
...
</form>

基础视图.py

@router.get('/', response_class=HTMLResponse)
async def main_page(request: Request,
                    activity_service: ActivityService = Depends()):
    activity = await activity_service.get()
    return templates.TemplateResponse('base.html', context={'request': request,
                                                            'activities': activity})

解决方案 1:

当您想要在 POST 之后重定向到 GET 时,最佳做法是使用状态代码进行重定向303,因此只需将代码更新为:

    # ...
    return RedirectResponse(redirect_url, status_code=303)

正如您所注意到的,重定向307会保留 HTTP 方法和主体。

完整工作示例:

from fastapi import FastAPI, APIRouter, Request
from fastapi.responses import RedirectResponse, HTMLResponse


router = APIRouter()

@router.get('/form')
def form():
    return HTMLResponse("""
    <html>
    <form action="/event/create" method="POST">
    <button>Send request</button>
    </form>
    </html>
    """)

@router.post('/create')
async def event_create(
        request: Request
):
    event = {"id": 123}
    redirect_url = request.url_for('get_event', **{'pk': event['id']})
    return RedirectResponse(redirect_url, status_code=303)


@router.get('/{pk}')
async def get_event(
        request: Request,
        pk: int,
):
    return f'<html>oi pk={pk}</html>'

app = FastAPI(title='Test API')

app.include_router(router, prefix="/event")

要运行,请安装pip install fastapi uvicorn并运行:

uvicorn --reload --host 0.0.0.0 --port 3000 example:app

然后,将浏览器指向:http://localhost:3000/event/form

解决方案 2:

您在此处提到的错误是因为你试图event_create通过http://127.0.0.1:8000/event/create访问端点而引发的。但是,由于event_create路由处理POST请求,您的请求最终会到达get_event端点(并引发value is not a valid integer错误,因为您传递的是字符串而不是整数),因为当您在浏览器的地址栏中键入 URL 时,它会执行GET请求

因此,您需要一个HTML<form>,例如,POSTevent_create端点提交请求。下面是一个工作示例,您可以使用它访问 HTML (根据需要调整端口号)以发送<form>请求,然后触发。http://127.0.0.1:8000/event/`POST`RedirectResponse

正如 @tiangolo在这里提到的,当执行RedirectResponsePOST请求路由到GET请求路由时,响应状态代码必须更改303 See Other。例如:

return RedirectResponse(redirect_url, status_code=status.HTTP_303_SEE_OTHER) 

工作示例:

from fastapi import APIRouter, FastAPI, Request, status
from fastapi.responses import RedirectResponse, HTMLResponse

router = APIRouter()

# This endpoint can be accessed at http://127.0.0.1:8000/event/
@router.get('/', response_class=HTMLResponse)
def event_create_form(request: Request):
    return """
    <html>
       <body>
          <h1>Create an event</h1>
          <form method="POST" action="/event/create">
             <input type="submit" value="Create Event">
          </form>
       </body>
    </html>
    """
    
@router.post('/create')
def event_create(request: Request):
    event = {"id": 1}
    redirect_url = request.url_for('get_event', **{'pk': event['id']})
    return RedirectResponse(redirect_url, status_code=status.HTTP_303_SEE_OTHER)    

@router.get('/{pk}')
def get_event(request: Request, pk: int):
    return {"pk": pk}


app = FastAPI()
app.include_router(router, prefix="/event")
相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   2079  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1459  
  建筑行业正处于数字化转型的关键时期,建筑产品生命周期管理(PLM)系统的实施对于提升项目效率、质量和协同性至关重要。特别是在 2025 年,基于建筑信息模型(BIM)的项目进度优化工具成为众多建筑企业关注的焦点。这些工具不仅能够整合项目全生命周期的数据,还能通过精准的分析和模拟,为项目进度管理提供强大支持。BIM 与建...
plm是什么软件   0  
  PLM系统开发的重要性与现状PLM(产品生命周期管理)系统在现代企业的产品研发、生产与管理过程中扮演着至关重要的角色。它贯穿产品从概念设计到退役的整个生命周期,整合了产品数据、流程以及人员等多方面的资源,极大地提高了企业的协同效率和创新能力。通过PLM系统,企业能够实现产品信息的集中管理与共享,不同部门之间可以实时获取...
国产plm软件   0  
  PLM(产品生命周期管理)系统在企业产品研发与管理过程中扮演着至关重要的角色。随着市场竞争的加剧和技术的飞速发展,企业对PLM系统的迭代周期优化需求日益迫切。2025年敏捷认证对项目管理提出了新的要求,其中燃尽图作为一种强大的可视化工具,在PLM系统迭代周期优化中有着广泛且重要的应用。深入探讨这些应用,对于提升企业的项...
plm系统主要干什么的   0  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用