You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

3.7 KiB

Body - 更新

使用 PUT 取代式更新

要更新一個項目,你可以使用 HTTP PUT 操作。

你可以使用 jsonable_encoder 將輸入資料轉換為可儲存為 JSON 的資料(例如用於 NoSQL 資料庫)。例如把 datetime 轉成 str

{* ../../docs_src/body_updates/tutorial001_py310.py hl[28:33] *}

PUT 用於接收應該取代現有資料的資料。

關於取代的警告

這表示,如果你想用 PUT 並在 body 中包含以下內容來更新項目 bar

{
    "name": "Barz",
    "price": 3,
    "description": None,
}

由於這裡沒有包含已儲存的屬性 "tax": 20.2,輸入的模型會採用預設值 "tax": 10.5

最終資料會以這個「新的」 tax10.5 被儲存。

使用 PATCH 進行部分更新

你也可以使用 HTTP PATCH 操作來進行部分更新。

這表示你只需傳送想要更新的資料,其餘保持不變。

/// note | 注意

PATCH 相較於 PUT 較少被使用、也較不為人知。

許多團隊甚至在部分更新時也只用 PUT

你可以依需求自由選用,FastAPI 不會強制規範。

但本指南會大致示範它們各自的設計用法。

///

使用 Pydantic 的 exclude_unset 參數

如果要接收部分更新,在 Pydantic 模型的 .model_dump() 中使用 exclude_unset 參數非常實用。

例如 item.model_dump(exclude_unset=True)

這會產生一個只包含建立 item 模型時實際設定過之欄位的 dict,不含預設值。

接著你可以用它來生成只包含實際設定(請求中傳來)的資料之 dict,省略預設值:

{* ../../docs_src/body_updates/tutorial002_py310.py hl[32] *}

使用 Pydantic 的 update 參數

接著,你可以用 .model_copy() 建立現有模型的副本,並傳入含有要更新資料之 dictupdate 參數。

例如 stored_item_model.model_copy(update=update_data)

{* ../../docs_src/body_updates/tutorial002_py310.py hl[33] *}

部分更新摘要

總結一下,若要套用部分更新,你可以:

*(可選)使用 PATCH 取代 PUT

  • 取回已儲存的資料。
  • 將該資料放入一個 Pydantic 模型。
  • 從輸入模型產生一個不含預設值的 dict(使用 exclude_unset)。
    • 如此即可只更新使用者實際設定的值,而不會以模型的預設值覆寫已儲存的值。
  • 建立已儲存模型的副本,並以收到的部分更新值更新其屬性(使用 update 參數)。
  • 將該副本模型轉成可儲存到資料庫的型別(例如使用 jsonable_encoder)。
    • 這與再次使用模型的 .model_dump() 類似,但它會確保(並轉換)所有值為可轉為 JSON 的資料型別,例如把 datetime 轉為 str
  • 將資料儲存到資料庫。
  • 回傳更新後的模型。

{* ../../docs_src/body_updates/tutorial002_py310.py hl[28:35] *}

/// tip | 提示

其實你也可以在 HTTP PUT 操作中使用同一套技巧。

但此處示例使用 PATCH,因為它正是為這類情境設計的。

///

/// note | 注意

請注意,輸入的模型依然會被驗證。

因此,如果你希望接收可以省略所有屬性的部分更新,你需要一個所有屬性皆為可選(具預設值或為 None)的模型。

為了區分用於更新(全部可選)與用於建立(欄位為必填)的模型,你可以參考 額外模型 中的做法。

///