„Wenn ein Arbeiter seine Arbeit gut machen will, muss er zuerst seine Werkzeuge schärfen.“ – Konfuzius, „Die Gespräche des Konfuzius. Lu Linggong“
Titelseite > Programmierung > Wie werden sowohl Formular- als auch JSON-Daten in einem einzigen FastAPI-Endpunkt verarbeitet?

Wie werden sowohl Formular- als auch JSON-Daten in einem einzigen FastAPI-Endpunkt verarbeitet?

Veröffentlicht am 08.11.2024
Durchsuche:947

How to Handle Both Form and JSON Data in a Single FastAPI Endpoint?

Wie erstelle ich einen FastAPI-Endpunkt, der entweder ein Formular oder einen JSON-Körper akzeptieren kann?

In FastAPI können Sie einen Endpunkt erstellen, der entweder ein Formular oder einen JSON-Körper akzeptieren kann mit unterschiedlichen Methoden. Hier sind einige Optionen:

Option 1: Verwenden einer Abhängigkeitsfunktion

Diese Option beinhaltet das Erstellen einer Abhängigkeitsfunktion, die den Wert des Content-Type-Anfrageheaders überprüft und den Text mithilfe der Methoden von Starlette analysiert , entsprechend.

from fastapi import FastAPI, Depends, Request
from starlette.datastructures import FormData

app = FastAPI()

async def get_body(request: Request):
    content_type = request.headers.get('Content-Type')
    if content_type is None:
        raise HTTPException(status_code=400, detail='No Content-Type provided!')
    elif content_type == 'application/json':
        return await request.json()
    elif (content_type == 'application/x-www-form-urlencoded' or
          content_type.startswith('multipart/form-data')):
        try:
            return await request.form()
        except Exception:
            raise HTTPException(status_code=400, detail='Invalid Form data')
    else:
        raise HTTPException(status_code=400, detail='Content-Type not supported!')

@app.post('/')
def main(body = Depends(get_body)):
    if isinstance(body, dict):  # if JSON data received
        return body
    elif isinstance(body, FormData):  # if Form/File data received
        msg = body.get('msg')
        items = body.getlist('items')
        return msg

Option 2: Separate Endpunkte definieren

Eine weitere Option besteht darin, einen einzelnen Endpunkt zu haben und Ihre Datei(en)- und/oder Formulardatenparameter als optional zu definieren. Wenn an einen der Parameter Werte übergeben werden, bedeutet dies, dass die Anfrage entweder application/x-www-form-urlencoded oder multipart/form-data war. Andernfalls handelte es sich wahrscheinlich um eine JSON-Anfrage.

from fastapi import FastAPI, UploadFile, File, Form
from typing import Optional, List

app = FastAPI()

@app.post('/')
async def submit(items: Optional[List[str]] = Form(None),
                    files: Optional[List[UploadFile]] = File(None)):
    # if File(s) and/or form-data were received
    if items or files:
        filenames = None
        if files:
            filenames = [f.filename for f in files]
        return {'File(s)/form-data': {'items': items, 'filenames': filenames}}
    else:  # check if JSON data were received
        data = await request.json()
        return {'JSON': data}

Option 3: Verwenden einer Middleware

Sie können auch eine Middleware verwenden, um die eingehende Anfrage zu überprüfen und sie je nach dem Endpunkt /submitJSON oder /submitForm umzuleiten auf dem Inhaltstyp der Anfrage.

from fastapi import FastAPI, Request
from fastapi.responses import JSONResponse

app = FastAPI()

@app.middleware("http")
async def some_middleware(request: Request, call_next):
    if request.url.path == '/':
        content_type = request.headers.get('Content-Type')
        if content_type is None:
            return JSONResponse(
                content={'detail': 'No Content-Type provided!'}, status_code=400)
        elif content_type == 'application/json':
            request.scope['path'] = '/submitJSON'
        elif (content_type == 'application/x-www-form-urlencoded' or
              content_type.startswith('multipart/form-data')):
            request.scope['path'] = '/submitForm'
        else:
            return JSONResponse(
                content={'detail': 'Content-Type not supported!'}, status_code=400)

    return await call_next(request)

@app.post('/')
def main():
    return

@app.post('/submitJSON')
def submit_json(item: Item):
    return item

@app.post('/submitForm')
def submit_form(msg: str = Form(...), items: List[str] = Form(...),
                    files: Optional[List[UploadFile]] = File(None)):
    return msg

Testen der Optionen

Sie können die oben genannten Optionen mithilfe der Anforderungsbibliothek von Python testen:

import requests

url = 'http://127.0.0.1:8000/'
files = [('files', open('a.txt', 'rb')), ('files', open('b.txt', 'rb'))]
payload ={'items': ['foo', 'bar'], 'msg': 'Hello!'}
 
# Send Form data and files
r = requests.post(url, data=payload, files=files)  
print(r.text)

# Send Form data only
r = requests.post(url, data=payload)              
print(r.text)

# Send JSON data
r = requests.post(url, json=payload)              
print(r.text)
Neuestes Tutorial Mehr>

Haftungsausschluss: Alle bereitgestellten Ressourcen stammen teilweise aus dem Internet. Wenn eine Verletzung Ihres Urheberrechts oder anderer Rechte und Interessen vorliegt, erläutern Sie bitte die detaillierten Gründe und legen Sie einen Nachweis des Urheberrechts oder Ihrer Rechte und Interessen vor und senden Sie ihn dann an die E-Mail-Adresse: [email protected] Wir werden die Angelegenheit so schnell wie möglich für Sie erledigen.

Copyright© 2022 湘ICP备2022001581号-3