Pulled ALL skills from 15 source repositories: - anthropics/skills: 16 (docs, design, MCP, testing) - obra/superpowers: 14 (TDD, debugging, agents, planning) - coreyhaines31/marketingskills: 25 (marketing, CRO, SEO, growth) - better-auth/skills: 5 (auth patterns) - vercel-labs/agent-skills: 5 (React, design, Vercel) - antfu/skills: 16 (Vue, Vite, Vitest, pnpm, Turborepo) - Plus 13 individual skills from various repos Mosaic Stack is not limited to coding — the Orchestrator and subagents serve coding, business, design, marketing, writing, logistics, analysis, and more. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
5.8 KiB
5.8 KiB
title, impact, impactDescription, type, tags
| title | impact | impactDescription | type | tags | |||||
|---|---|---|---|---|---|---|---|---|---|
| Async Navigation Guards Require Proper Promise Handling | MEDIUM | Unawaited promises in guards cause navigation to complete before async checks finish, allowing unauthorized access or missing data | gotcha |
|
Async Navigation Guards Require Proper Promise Handling
Impact: MEDIUM - Navigation guards that perform async operations (API calls, auth checks) must properly handle promises. If you don't await async operations or return the promise, navigation completes before your check finishes, potentially allowing unauthorized access or navigating with incomplete data.
Task Checklist
- Use async/await in navigation guards
- Return the promise if not using async/await
- Add loading states for long async operations
- Implement timeouts for slow API calls
- Handle errors to prevent navigation hanging
The Problem
// WRONG: Not awaiting - navigation proceeds immediately
router.beforeEach((to, from) => {
if (to.meta.requiresAuth) {
checkAuth() // This returns a Promise but we're not waiting!
// Navigation continues before checkAuth completes
}
})
// WRONG: Async function but forgot return
router.beforeEach(async (to, from) => {
if (to.meta.requiresAuth) {
const isValid = await checkAuth()
if (!isValid) {
// This redirect might happen after navigation already completed!
return '/login'
}
}
// Missing return - implicitly returns undefined, allowing navigation
})
Solution: Proper Async/Await Pattern
// CORRECT: Async function with proper returns
router.beforeEach(async (to, from) => {
if (to.meta.requiresAuth) {
try {
const isAuthenticated = await checkAuth()
if (!isAuthenticated) {
return { name: 'Login', query: { redirect: to.fullPath } }
}
} catch (error) {
console.error('Auth check failed:', error)
return { name: 'Error', params: { message: 'Authentication failed' } }
}
}
// Explicitly return nothing to proceed
return true
})
Solution: Promise-Based Pattern (Alternative)
// CORRECT: Return promise explicitly
router.beforeEach((to, from) => {
if (to.meta.requiresAuth) {
return checkAuth()
.then(isAuthenticated => {
if (!isAuthenticated) {
return { name: 'Login' }
}
})
.catch(error => {
console.error('Auth check failed:', error)
return { name: 'Error' }
})
}
})
Loading State During Async Guards
// app/composables/useNavigationLoading.js
import { ref } from 'vue'
const isNavigating = ref(false)
export function useNavigationLoading() {
return { isNavigating }
}
export function setupNavigationLoading(router) {
router.beforeEach(() => {
isNavigating.value = true
})
router.afterEach(() => {
isNavigating.value = false
})
router.onError(() => {
isNavigating.value = false
})
}
<!-- App.vue -->
<script setup>
import { useNavigationLoading } from '@/composables/useNavigationLoading'
const { isNavigating } = useNavigationLoading()
</script>
<template>
<LoadingBar v-if="isNavigating" />
<router-view />
</template>
Timeout Pattern for Slow APIs
// CORRECT: Add timeout to prevent indefinite waiting
function withTimeout(promise, ms = 5000) {
return Promise.race([
promise,
new Promise((_, reject) =>
setTimeout(() => reject(new Error('Request timeout')), ms)
)
])
}
router.beforeEach(async (to, from) => {
if (to.meta.requiresAuth) {
try {
const isValid = await withTimeout(checkAuth(), 5000)
if (!isValid) {
return '/login'
}
} catch (error) {
if (error.message === 'Request timeout') {
// Let user through but show warning
console.warn('Auth check timed out')
} else {
return '/login'
}
}
}
})
Multiple Async Checks
// CORRECT: Run independent checks in parallel
router.beforeEach(async (to, from) => {
if (to.meta.requiresAuth && to.meta.requiresSubscription) {
try {
const [isAuthenticated, hasSubscription] = await Promise.all([
checkAuth(),
checkSubscription()
])
if (!isAuthenticated) {
return '/login'
}
if (!hasSubscription) {
return '/subscribe'
}
} catch (error) {
return '/error'
}
}
})
Error Handling Best Practices
router.beforeEach(async (to, from) => {
try {
// Your async logic here
await performChecks(to)
} catch (error) {
// Always handle errors to prevent navigation from hanging
if (error.response?.status === 401) {
return '/login'
}
if (error.response?.status === 403) {
return '/forbidden'
}
if (error.code === 'NETWORK_ERROR') {
// Offline - maybe allow navigation but show warning
return true
}
// Unknown error - redirect to error page
console.error('Navigation guard error:', error)
return { name: 'Error', state: { error: error.message } }
}
})
Key Points
- Always await async operations - Otherwise navigation proceeds immediately
- Return values matter - Return route to redirect, false to cancel, true/undefined to proceed
- Handle all error cases - Uncaught errors can hang navigation
- Add timeouts - Slow APIs shouldn't block navigation indefinitely
- Show loading state - Users need feedback during async checks
- Parallelize independent checks - Use Promise.all for better performance