inertia-rails-architecture

Server-driven architecture patterns for Inertia Rails + React. Load this FIRST when building any Inertia page or feature — it routes to the right skill. Decision matrix for data loading, forms, navigation, state management. NEVER useEffect+fetch, NEVER redirect_to for external URLs (use inertia_location), NEVER react-hook-form (use Form component). MUST invoke when adding pages, models with views, CRUD, or displaying data in an Inertia Rails app. ALWAYS `render inertia: { key: value }` to pass data — `@ivars` are NOT auto-passed as props.

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "inertia-rails-architecture" with this command: npx skills add inertia-rails/skills/inertia-rails-skills-inertia-rails-architecture

Inertia Rails Architecture

Server-driven architecture for Rails + Inertia.js + React when building pages, forms, navigation, or data refresh. Inertia is NOT a traditional SPA — the server owns routing, data, and auth. React handles rendering only.

The Core Mental Model

The server is the source of truth. React receives data as props and renders UI. There is no client-side router, no global state store, no API layer.

Before building any feature, ask:

  • Where does the data come from? → If server: controller prop. If user interaction: useState.
  • Who owns this state? → If it's in the URL or DB: server owns it (use props). If it's ephemeral UI: React owns it.
  • Am I reaching for a React/SPA pattern? → Check the decision matrix below first — Inertia likely has a server-driven equivalent.

Decision Matrix

NeedSolutionNOT This
Page data from serverController propsuseEffect + fetch
Global data (auth, config)inertia_share + usePage()React Context / Redux
Flash messages / toastsRails flash + usePage().flashinertia_share / React state
Form submission<Form> componentfetch/axios + useState
Navigate between pages<Link> / router.visitreact-router / window.location
Refresh specific datarouter.reload({ only: [...] })React Query / SWR
Expensive server dataInertiaRails.deferuseEffect + loading state
Infinite scrollInertiaRails.scroll + <InfiniteScroll>Client-side pagination
Stable reference dataInertiaRails.onceCache in React state
Real-time updates (core)ActionCable + router.reloadPolling with setInterval
Simple polling (MVP/prototyping)usePoll (auto-throttles in background tabs)setInterval + router.reload
URL-driven UI state (dialogs, tabs)Controller reads params → prop, router.get to updateuseEffect + window.location
Ephemeral UI stateuseState / useReducerServer props
External API callsDedicated API endpointMixing with Inertia props

Rules (by impact)

#ImpactRuleWHY
1CRITICALNever useEffect+fetch for page dataInertia re-renders the full component on navigation; a useEffect fetch creates a second data lifecycle that drifts from props and causes stale UI
2CRITICALNever check auth client-sideAuth state in React can be spoofed; server-side checks are the only real gate. Client-side "guards" give false security
3CRITICALUse <Form>, not fetch/axios<Form> handles CSRF, redirect-following, error mapping, file detection, and history state — fetch duplicates or breaks all of this
4HIGHUse <Link> and router, not <a> or window.location<a> triggers a full page reload, destroying all React state and layout persistence
5HIGHUse partial reloads, not React Query/SWRReact Query adds a second cache layer that conflicts with Inertia's page-based caching and versioning
5bHIGHUse usePoll only for MVPs; prefer ActionCable for production real-timeusePoll is convenient but wastes bandwidth — every interval hits the server even when nothing changed. ActionCable pushes only on actual changes
6HIGHUse inertia_share for global data, not React ContextContext re-renders consumers on every change; shared props are per-request and integrated with partial reloads
7HIGHUse Rails flash for notifications, not shared propsFlash auto-clears after one response; shared props persist until explicitly changed, causing stale toasts
8MEDIUMUse deferred/optional props for expensive queriesBlocks initial render otherwise — user sees blank page until slow query finishes
9MEDIUMUse persistent layouts for state preservationWithout persistent layout, layout remounts on every navigation — scroll position, audio playback, and component state are lost
10MEDIUMKeep React components as renderers, not data fetchersMixing data-fetching into components makes them untestable and breaks Inertia's server-driven model

Skill Map

Common workflows span multiple skills — load all listed for complete coverage:

WorkflowLoad these skills
New page with propsinertia-rails-controllers + inertia-rails-pages + inertia-rails-typescript
Form with validationinertia-rails-forms + inertia-rails-controllers
shadcn form inputsinertia-rails-forms + shadcn-inertia
Flash toastsinertia-rails-controllers + inertia-rails-pages + shadcn-inertia
Deferred/lazy datainertia-rails-controllers + inertia-rails-pages
URL-driven dialog/tabsinertia-rails-controllers + inertia-rails-pages
Alba serializationalba-inertia + inertia-rails-typescript
Testing controllersinertia-rails-testing + inertia-rails-controllers

References

MANDATORY — READ ENTIRE FILE before building a new Inertia page or feature: references/AGENTS.md (~430 lines) — full-stack examples for each pattern in the decision matrix above.

MANDATORY — READ ENTIRE FILE when unsure which Inertia pattern to use: references/decision-trees.md (~70 lines) — flowcharts for choosing between prop types, navigation methods, and data strategies.

Do NOT load references for quick questions about a single pattern already covered in the decision matrix above.

When You DO Need a Separate API

Not everything belongs in Inertia's request cycle. Use a traditional API endpoint when:

SignalWhyExample
Non-browser consumerInertia's JSON envelope (component, props, url, version) is designed for the frontend adapter — other consumers can't use itMobile API, CLI tools, payment webhooks
Large-dataset searchDataset is too big to load as a prop; each input needs per-keystroke server filtering. Use raw fetch for the search, let Inertia handle post-selection side effects via props.City/address autocomplete, postal code lookup
Binary/streaming responseInertia can only deliver JSON props. Use a separate route with a standard download response.PDF/CSV export, file downloads

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

General

inertia-rails-controllers

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

inertia-rails-typescript

No summary provided by upstream source.

Repository SourceNeeds Review
General

inertia-rails-pages

No summary provided by upstream source.

Repository SourceNeeds Review
General

inertia-rails-testing

No summary provided by upstream source.

Repository SourceNeeds Review