maintainer-decline
This skill guides open-source maintainers in composing respectful, clear responses when declining feature requests, closing issues as won't-fix, redirecting users to alternative solutions, or explaining intentional design choices. Use it when communicating with contributors about deprecations, scope limitations, effort tradeoffs, or misunderstandings, ensuring the recipient understands the reasoning behind the decision while remaining unblocked and valued.
git clone --depth 1 https://github.com/posit-dev/skills /tmp/maintainer-decline && cp -r /tmp/maintainer-decline/open-source/maintainer-decline ~/.claude/skills/maintainer-declineSKILL.md
# Maintainer Decline Help an open-source maintainer compose responses that close, decline, or redirect issues while leaving the contributor informed and respected. ## Core Philosophy Closure is not dismissal. A well-crafted decline educates, unblocks, or redirects the contributor productively. The goal is that someone reading your response walks away understanding *why* — even if they didn't get what they asked for. ## Principles **Explain the why, not the what.** Contributors understand they're being told "no." What they need is the reasoning: strategic direction, design philosophy, effort/benefit tradeoff, or the actual technical root cause. **Validate before declining.** Acknowledge the contributor's point before the redirect. Phrases like "Yeah," "This is a reasonable idea," or "I think..." signal that you heard them before delivering the decision. **Offer a path forward when possible.** Even a "no" can come with "but you could..." or "use X instead." Leave the contributor unblocked or pointed in the right direction. **Use "we" for design decisions.** "We believe," "we've decided," "we experimented with..." positions decisions as team choices, not personal gatekeeping. **One decision per response.** No hedging, no "maybe later" unless you mean it. Clear closure — even if that closure is "this will happen after X." **Brevity signals respect.** Most declines land in 1–3 sentences. Go longer only when the misunderstanding is significant or teaching is genuinely needed. **Acknowledge effort.** If the contributor did real work (wrote a reprex, drafted code, did research), name it. "Thanks for the detailed report" costs nothing and signals you value their time. ## Response Patterns Choose the pattern that fits the situation. These are not templates — they're structural moves you can adapt. ### Duplicate One line. Link the original issue. No elaboration needed. > Duplicate of #7651 ### Fixed / Resolved Link the fix. Nothing else required. > Fixed in #1430 ### Won't Fix — Superseded or Deprecated Explain the strategic direction, reference the replacement, thank the user. > gather() is no longer under active development because it's been superseded by pivot_longer(). Sorry we can't help more and thanks for using tidyr! > Given that we're moving away from this warning (in favour of using .by), I think unfortunately it's more effort than it's worth. **Structure:** [What's happening strategically] + [migration path or replacement] + [brief thanks or acknowledgment] ### Design Philosophy When something looks like a bug but is working as intended. Frame as a principle, explain the consequence chain. > In general, we believe this liberal type casting is a net negative because it can mean that data quality problems are silently propagated, eventually causing uninformative errors. > We experimented with this in the past but because many tidyr functions produce list columns and the data.frame method for print() doesn't do a great job with them, we've decided to stick with tibbles everywhere. **Structure:** [The principle: "we believe..." / "we've decided..."] + [What goes wrong if you allow it] + [Optionally: past exploration that led here] ### Workaround / Unblocking Alternative The request isn't happening, but there's a path forward. Lead with validation, offer the concrete alternative. Position the workaround as the *best* approach, not a consolation prize. > Yeah, I think you're best off creating the formulas outside of expand_grid(). > You could make your code a bit simpler by using coalesce(): [code example]. We don't have many functions that work across columns because of the challenge of what to do if they're incompatible types. **Structure:** [Validation: "Yeah," / "I think..."] + [Concrete alternative, possibly with code] + [Optionally: why the feature doesn't exist] ### Clarification / Teaching Root Cause The contributor misunderstands the behavior. Show what's actually happening — with a reprex, a documentation quote, or a one-sentence explanation of the mechanic they missed. > The problem is that your pattern doesn't match, and the expected behaviour in that case is that you get an NA at the output. Unfortunately there's no way for us to fill in the result from some of the match groups, because none of them get populated by the regexp engine. > It's not showing up because instead of one long map() call, you're creating many short ones. > I think this is pretty clearly documented: [quotes docs]. It's pretty hard to change this sort of behaviour without affecting a lot of existing code and given the seeming rarity of it being a problem, I unfortunately don't think it's worth the effort. **Structure:** [What's actually happening] + [Why it works this way] + [Optionally: why it can't easily change] ### Effort Assessment The request is valid but the effort/benefit doesn't justify it. Acknowledge the idea, explain the barrier, optionally leave the door open for external solutions. > This seems like a reasonable optimisation, but it's not clear to me where it should live. Unfortunately I think there are just too many unknowns for us to get traction on this problem, but it would be great to have a tidyverse-wide solution. **Structure:** [Acknowledge validity] + [Name the real barrier] + [Optionally: where it could live instead] ### Scope / "Not Here Yet" The idea has merit but doesn't belong in this package or isn't ready. > Closing this for the above reasons; I think it's an interesting idea, but doesn't feel right for tidyr yet. **Structure:** [Acknowledge merit] + [Question fit: "doesn't feel right for X yet" / "is Y the right place?"] ### Redirect Diagnosis The contributor is looking at the wrong layer or package. Gently suggest a different direction. > This looks more like a stringr issue? **Structure:** [Suggest alternative diagnosis, often as a question] ## Choosing the Right Pattern Use these heuristics to identify which pattern fits:
>
Create and use brand.yml files for consistent branding across Shiny apps and Quarto documents. Covers: (1) Creating new _brand.yml files, (2) Applying to Shiny (R and Python), (3) Using in Quarto, (4) Modifying existing files, and (5) Troubleshooting. Includes complete specifications and integration guides.
Write ggsql queries — a grammar of graphics for SQL. Use when the user wants to create, modify, or understand a ggsql visualization query.
Creates a pull request from current changes, monitors GitHub CI, and debugs any failures until CI passes. Activate when the user says "create pr", "make a pr", "open pull request", "submit pr", "pr for these changes", or wants to get their current work into a reviewable PR. Assumes the project uses git, is hosted on GitHub, and has GitHub Actions CI with automated checks (lint, build, tests, etc.). Does NOT merge - stops when CI passes and provides the PR link.
Address PR review feedback by systematically working through every unresolved PR review thread on the current branch's PR - analyze each comment, make the requested code changes (with tests where useful), commit, and optionally reply and resolve.
Bulk resolve unresolved PR review threads on the current branch’s PR — typically after threads have been addressed manually or via /pr-threads-address
>
>