fail-early-pattern
À propos
Cette compétence met en œuvre le modèle "fail-fast" pour valider les entrées et détecter les erreurs immédiatement en utilisant des clauses de garde et des assertions. Elle fournit des exemples pratiques en R avec des conseils polyglottes pour écrire des fonctions robustes et renforcer les API. Utilisez-la lors de la réception d'entrées externes, de la refactorisation de pannes silencieuses ou de l'examen du code pour la qualité de la gestion des erreurs.
Installation rapide
Claude Code
Recommandénpx skills add pjt222/agent-almanac -a claude-code/plugin add https://github.com/pjt222/agent-almanacgit clone https://github.com/pjt222/agent-almanac.git ~/.claude/skills/fail-early-patternCopiez et collez cette commande dans Claude Code pour installer cette compétence
Documentation
早敗
若物將敗,宜盡早、盡響、盡帶語境而敗。此技定早敗之模式:於系統邊界驗輸入、以 guard 子句於壞態擴前拒之、書答何敗、於何處、何以、何以修之錯誤訊息。
適用時機
- 書或審受外部輸入之函數(使用者數據、API 回應、檔內容)
- CRAN 提交前為包函數加輸入驗
- 重構默生誤結而非報錯之代碼
- 審 pull request 之錯處品質
- 加固內部 API 以禦誤參
輸入
- 必要:欲施此模式之函數或模組
- 必要:信任邊界之識(外部數據入處)
- 選擇性:待重構之既有錯處代碼
- 選擇性:目標語言(預設:R;亦適於 Python、TypeScript、Rust)
步驟
步驟一:識信任邊界
映外部數據入系統之處。此為需驗之點:
- 公 API 函數(R 包中所匯出之函數)
- 面對使用者之參數
- 檔 I/O(讀配置、數據檔、使用者上傳)
- 網絡回應(API 呼、DB 查詢)
- 環境變量與系統配置
僅由爾自身已驗代碼呼之內部輔助函數通常無需冗餘之驗。
預期: 不信任數據跨入爾代碼之入口清單。
失敗時: 若邊界不明,自日誌或錯報中之錯倒追至壞數據首入處。
步驟二:於入口加 guard 子句
於各公函數之頂驗輸入,先於任何工。
R(base):
calculate_summary <- function(data, method = c("mean", "median", "trim"), trim_pct = 0.1) {
# Guard: type check
if (!is.data.frame(data)) {
stop("'data' must be a data frame, not ", class(data)[[1]], call. = FALSE)
}
# Guard: non-empty
if (nrow(data) == 0L) {
stop("'data' must have at least one row", call. = FALSE)
}
# Guard: argument matching
method <- match.arg(method)
# Guard: range check
if (!is.numeric(trim_pct) || trim_pct < 0 || trim_pct > 0.5) {
stop("'trim_pct' must be a number between 0 and 0.5, got: ", trim_pct, call. = FALSE)
}
# --- All guards passed, begin real work ---
# ...
}
R(rlang/cli — 於包所偏好):
calculate_summary <- function(data, method = c("mean", "median", "trim"), trim_pct = 0.1) {
rlang::check_required(data)
if (!is.data.frame(data)) {
cli::cli_abort("{.arg data} must be a data frame, not {.cls {class(data)}}.")
}
if (nrow(data) == 0L) {
cli::cli_abort("{.arg data} must have at least one row.")
}
method <- rlang::arg_match(method)
if (!is.numeric(trim_pct) || trim_pct < 0 || trim_pct > 0.5) {
cli::cli_abort("{.arg trim_pct} must be between 0 and 0.5, not {.val {trim_pct}}.")
}
# ...
}
通用(TypeScript):
function calculateSummary(data: DataFrame, method: Method, trimPct: number): Summary {
if (data.rows.length === 0) {
throw new Error(`data must have at least one row`);
}
if (trimPct < 0 || trimPct > 0.5) {
throw new RangeError(`trimPct must be between 0 and 0.5, got: ${trimPct}`);
}
// ...
}
預期: 每公函數以 guard 子句啟,於任何副作用或算前拒誤輸入。
失敗時: 若驗邏輯漸長(>15 行 guard),抽 validate_* 輔或用 stopifnot() 為簡類斷言。
步驟三:書有意之錯誤訊息
每錯訊息宜答四問:
- 何敗——何參數或操作
- 於何處——函數名或語境(用
cli::cli_abort自動) - 何以——期何與受何
- 何以修——於修不顯時
善訊:
# What + Why (expected vs. actual)
stop("'n' must be a positive integer, got: ", n, call. = FALSE)
# What + Why + How to fix
cli::cli_abort(c(
"{.arg config_path} does not exist: {.file {config_path}}",
"i" = "Create it with {.run create_config({.file {config_path}})}."
))
# What + context
cli::cli_abort(c(
"Column {.val {col_name}} not found in {.arg data}.",
"i" = "Available columns: {.val {names(data)}}"
))
惡訊:
stop("Error") # What failed? No idea
stop("Invalid input") # Which input? What's wrong with it?
stop(paste("Error in step", i)) # No actionable information
預期: 錯訊自文檔——初見錯之開發者不讀源碼即可診並修。
失敗時: 審近三 bug 報告。若有需讀源碼方解者,其錯訊需改。
步驟四:偏 stop() 而避 warning()
於函數不能生正確結果時用 stop()(或 cli::cli_abort())。唯於函數仍可生有意之結果而呼者宜知之慮時用 warning()。
約則: 若使用者可默得誤答,則此為 stop(),非 warning()。
# CORRECT: stop when result would be wrong
read_config <- function(path) {
if (!file.exists(path)) {
stop("Config file not found: ", path, call. = FALSE)
}
yaml::read_yaml(path)
}
# CORRECT: warn when result is still usable
summarize_data <- function(data) {
if (any(is.na(data$value))) {
warning(sum(is.na(data$value)), " NA values dropped from 'value' column", call. = FALSE)
data <- data[!is.na(data$value), ]
}
# proceed with valid data
}
預期: stop() 用於將生誤結之條件;warning() 保於降級而仍有效之結果。
失敗時: 審既有之 warning() 呼。若函數於警後返無義之物,改為 stop()。
步驟五:以斷言持內部不變
於「正確代碼中不當生」之條件,用斷言。此於開發中捕程式員之錯:
# R: stopifnot for internal invariants
process_chunk <- function(chunk, total_size) {
stopifnot(
is.list(chunk),
length(chunk) > 0,
total_size > 0
)
# ...
}
# R: explicit assertion with context
merge_results <- function(left, right) {
if (ncol(left) != ncol(right)) {
stop("Internal error: column count mismatch (", ncol(left), " vs ", ncol(right),
"). This is a bug — please report it.", call. = FALSE)
}
# ...
}
預期: 內部不變已斷,故 bug 於違處立顯,非三呼後帶隱晦錯。
失敗時: 若 stopifnot() 之訊過隱,改用附語境之顯 if/stop。
步驟六:重構反模式
識並修此常反模式:
反模式一:空 tryCatch(吞錯)
# BEFORE: Error silently disappears
result <- tryCatch(
parse_data(input),
error = function(e) NULL
)
# AFTER: Log, re-throw, or return a typed error
result <- tryCatch(
parse_data(input),
error = function(e) {
cli::cli_abort("Failed to parse input: {e$message}", parent = e)
}
)
反模式二:預設值遮誤輸入
# BEFORE: Caller never knows their input was ignored
process <- function(x = 10) {
if (!is.numeric(x)) x <- 10 # silently replaces bad input
x * 2
}
# AFTER: Tell the caller about the problem
process <- function(x = 10) {
if (!is.numeric(x)) {
stop("'x' must be numeric, got ", class(x)[[1]], call. = FALSE)
}
x * 2
}
反模式三:以 suppressWarnings 為修
# BEFORE: Hiding the symptom instead of fixing the cause
result <- suppressWarnings(as.numeric(user_input))
# AFTER: Validate explicitly, handle the expected case
if (!grepl("^-?\\d+\\.?\\d*$", user_input)) {
stop("Expected a number, got: '", user_input, "'", call. = FALSE)
}
result <- as.numeric(user_input)
反模式四:通捕異常處理
# BEFORE: Every error treated the same
tryCatch(
complex_operation(),
error = function(e) message("Something went wrong")
)
# AFTER: Handle specific conditions, let unexpected ones propagate
tryCatch(
complex_operation(),
custom_validation_error = function(e) {
cli::cli_warn("Validation issue: {e$message}")
fallback_value
}
# Unexpected errors propagate naturally
)
預期: 反模式已換為顯驗或具體錯處。
失敗時: 若移 tryCatch 致連鎖敗,則上游代碼有驗之罅。修源,非症。
步驟七:驗早敗之重構
執測試以確錯路正行:
# Verify error messages are triggered
testthat::expect_error(calculate_summary("not_a_df"), "must be a data frame")
testthat::expect_error(calculate_summary(data.frame()), "at least one row")
testthat::expect_error(calculate_summary(mtcars, trim_pct = 2), "between 0 and 0.5")
# Verify valid inputs still work
testthat::expect_no_error(calculate_summary(mtcars, method = "mean"))
# Run full test suite
Rscript -e "devtools::test()"
預期: 所有測試過。錯路測試確誤輸入觸發所期之錯訊。
失敗時: 若既有測試倚默敗(如於誤輸入返 NULL),更之以期新之錯。
驗證
- 每公函數於工前驗其輸入
- 錯訊答:何敗、於何處、何以、何以修
-
stop()用於生誤結之條件 -
warning()唯用於降級而仍有效之結果 - 無默吞錯之空
tryCatch塊 - 無以
suppressWarnings()代正驗 - 無默遮誤輸入之預設值
- 內部不變用
stopifnot()或顯斷言 - 每驗 guard 有錯路測試
- 重構後測試套件過
常見陷阱
- 驗過深:於信任邊界(公 API)驗,非於每內部輔助。過驗加噪害效能。
- 無語境之錯訊:
"Invalid input"迫呼者猜。恒含參名、期類/範圍、實所受之值。 - 本意 stop() 而用 warning():若函數於警後返垃圾,則呼者默得誤答。用
stop()令呼者決如何處。 - 於 tryCatch 吞錯:
tryCatch(..., error = function(e) NULL)藏 bug。若必捕,記或以加語境再拋。 - 忘 call. = FALSE:於 R,
stop("msg")預設含呼,於終端使用者為噪。於面對使用者之函數用call. = FALSE。cli::cli_abort()自動為之。 - 於測試而非代碼中驗:測試驗行為而不護生產呼者。驗屬函數本身。
- 混合系統之誤 R 二進制:於 WSL 或 Docker,
Rscript或解為跨平台包裝而非原生 R。以which Rscript && Rscript --version察。偏原生 R 二進制(如 Linux/WSL 之/usr/local/bin/Rscript)以求可靠。見 設環境 查 R 路徑配置。
相關技能
write-testthat-tests- 書驗錯路之測試review-pull-request- 審代碼缺驗與默敗review-software-architecture- 於系統層評錯處策略create-skill- 循 agentskills.io 標準建新技能security-audit-codebase- 與輸入驗重疊之以安全為焦之審
Dépôt GitHub
Compétences associées
evaluating-llms-harness
TestsCette compétence Claude exécute le lm-evaluation-harness pour évaluer les modèles de langage sur plus de 60 tâches académiques standardisées telles que MMLU et GSM8K. Elle est conçue pour permettre aux développeurs de comparer la qualité des modèles, de suivre les progrès de l'entraînement ou de rapporter des résultats académiques. L'outil prend en charge différents backends, incluant les modèles HuggingFace et vLLM.
cloudflare-cron-triggers
TestsCette compétence fournit une connaissance complète pour la mise en œuvre de Déclencheurs Cron Cloudflare afin de planifier des Workers à l'aide d'expressions cron. Elle couvre la configuration de tâches périodiques, de travaux de maintenance et de flux de travail automatisés, tout en traitant des problèmes courants tels que les expressions cron non valides et les problèmes de fuseau horaire. Les développeurs peuvent l'utiliser pour configurer des gestionnaires planifiés, tester des déclencheurs cron et intégrer avec Workflows et Green Compute.
webapp-testing
TestsCette Compétence Claude fournit une boîte à outils basée sur Playwright pour tester des applications web locales via des scripts Python. Elle permet la vérification frontend, le débogage d'interface utilisateur, la capture d'écrans et la consultation des journaux, tout en gérant les cycles de vie du serveur. Utilisez-la pour les tâches d'automatisation de navigateur, mais exécutez les scripts directement plutôt que de lire leur code source pour éviter la pollution du contexte.
finishing-a-development-branch
TestsCette compétence aide les développeurs à finaliser leur travail en vérifiant que les tests passent, puis en présentant des options d'intégration structurées. Elle guide le processus de fusion, de création de PRs ou de nettoyage des branches une fois l'implémentation terminée. Utilisez-la lorsque votre code est prêt et testé pour finaliser systématiquement le cycle de développement.
