---
title: Keep Provide/Inject Mutations in the Provider Component
impact: MEDIUM
impactDescription: Allowing injectors to mutate provided state leads to unpredictable data flow and difficult debugging
type: best-practice
tags: [vue3, provide-inject, state-management, architecture, debugging]
---
# Keep Provide/Inject Mutations in the Provider Component
**Impact: MEDIUM** - When using reactive provide/inject values, mutations should be kept inside the provider component whenever possible. Allowing child components to mutate injected state directly leads to confusion about where changes originate, making debugging and maintenance difficult.
## Task Checklist
- [ ] Keep all mutations to provided state in the provider component
- [ ] Provide update functions alongside reactive data for child modifications
- [ ] Use `readonly()` to prevent accidental mutations from injectors
- [ ] Document provided values and their update patterns
## The Problem: Scattered Mutations
**Wrong - Injector mutates provided state directly:**
```vue
```
```vue
```
**Problems:**
1. Hard to trace where state changes originate
2. Multiple components might mutate the same data inconsistently
3. No centralized validation or side effects
4. Debugging becomes a nightmare in large apps
## Solution: Provide Update Functions
**Correct - Provider controls all mutations:**
```vue
```
```vue
Theme: {{ user.preferences.theme }}
```
## Pattern: Provide/Inject with Readonly Protection
Use `readonly()` to enforce the pattern at runtime:
```vue
```
```vue
{{ item.name }}
```
## Benefits of This Pattern
1. **Traceability**: All mutations go through known functions
2. **Validation**: Centralized validation in provider
3. **Side Effects**: Consistent side effects (logging, storage, API calls)
4. **Testing**: Easier to test mutation logic in isolation
5. **Debugging**: Clear mutation source in Vue DevTools
## When Direct Mutation Might Be Acceptable
In rare cases, direct mutation may be acceptable:
- Very simple, local state within a small component tree
- Form state that's isolated to a single form wizard
- Temporary state that doesn't affect app logic
Even then, consider using `readonly()` with update functions for consistency.
## Reference
- [Vue.js Provide/Inject - Working with Reactivity](https://vuejs.org/guide/components/provide-inject.html#working-with-reactivity)
- [The Complete Guide to Provide/Inject API in Vue 3](https://www.codemag.com/Article/2101091/The-Complete-Guide-to-Provide-Inject-API-in-Vue-3-Part-1)