Compare commits
15 Commits
5b638ca76f
...
feature/is
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0a020a6681 | ||
| ec6dd68e70 | |||
|
|
76c4a875ff | ||
|
|
2635483dbc | ||
| f6300c890b | |||
|
|
8a9f8f7fdd | ||
|
|
bd000649e3 | ||
|
|
1ed51c3667 | ||
|
|
76a229952f | ||
|
|
c5870f9e6f | ||
|
|
0ba695f159 | ||
| 7f9a92994c | |||
|
|
401d40fbe2 | ||
| 915b4fad5f | |||
|
|
2ca3c58f42 |
@@ -57,6 +57,21 @@
|
|||||||
/>
|
/>
|
||||||
</UFormGroup>
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Low Stock Threshold -->
|
||||||
|
<UFormGroup
|
||||||
|
label="Low Stock Alert"
|
||||||
|
hint="Optional - Alert when quantity falls below this"
|
||||||
|
>
|
||||||
|
<UInput
|
||||||
|
v-model.number="form.low_stock_threshold"
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
step="0.1"
|
||||||
|
placeholder="e.g. 2"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
<!-- Notes -->
|
<!-- Notes -->
|
||||||
<UFormGroup label="Notes" hint="Optional">
|
<UFormGroup label="Notes" hint="Optional">
|
||||||
<UTextarea
|
<UTextarea
|
||||||
@@ -121,6 +136,7 @@ const form = reactive({
|
|||||||
quantity: 1,
|
quantity: 1,
|
||||||
unit_id: '',
|
unit_id: '',
|
||||||
expiry_date: '',
|
expiry_date: '',
|
||||||
|
low_stock_threshold: null as number | null,
|
||||||
notes: ''
|
notes: ''
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -210,6 +226,7 @@ const handleSubmit = async () => {
|
|||||||
quantity: form.quantity,
|
quantity: form.quantity,
|
||||||
unit_id: form.unit_id,
|
unit_id: form.unit_id,
|
||||||
expiry_date: form.expiry_date || null,
|
expiry_date: form.expiry_date || null,
|
||||||
|
low_stock_threshold: form.low_stock_threshold,
|
||||||
notes: form.notes.trim() || null
|
notes: form.notes.trim() || null
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -55,6 +55,21 @@
|
|||||||
/>
|
/>
|
||||||
</UFormGroup>
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Low Stock Threshold -->
|
||||||
|
<UFormGroup
|
||||||
|
label="Low Stock Alert"
|
||||||
|
hint="Optional - Alert when quantity falls below this"
|
||||||
|
>
|
||||||
|
<UInput
|
||||||
|
v-model.number="form.low_stock_threshold"
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
step="0.1"
|
||||||
|
placeholder="e.g. 2"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
<!-- Notes -->
|
<!-- Notes -->
|
||||||
<UFormGroup label="Notes" hint="Optional">
|
<UFormGroup label="Notes" hint="Optional">
|
||||||
<UTextarea
|
<UTextarea
|
||||||
@@ -112,6 +127,7 @@ const form = reactive({
|
|||||||
quantity: 1,
|
quantity: 1,
|
||||||
unit_id: '',
|
unit_id: '',
|
||||||
expiry_date: '',
|
expiry_date: '',
|
||||||
|
low_stock_threshold: null as number | null,
|
||||||
notes: ''
|
notes: ''
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -136,6 +152,7 @@ watch(() => props.item, (newItem) => {
|
|||||||
form.quantity = Number(newItem.quantity)
|
form.quantity = Number(newItem.quantity)
|
||||||
form.unit_id = newItem.unit_id
|
form.unit_id = newItem.unit_id
|
||||||
form.expiry_date = newItem.expiry_date || ''
|
form.expiry_date = newItem.expiry_date || ''
|
||||||
|
form.low_stock_threshold = newItem.low_stock_threshold || null
|
||||||
form.notes = newItem.notes || ''
|
form.notes = newItem.notes || ''
|
||||||
isOpen.value = true
|
isOpen.value = true
|
||||||
}
|
}
|
||||||
@@ -168,6 +185,7 @@ const handleSubmit = async () => {
|
|||||||
quantity: form.quantity,
|
quantity: form.quantity,
|
||||||
unit_id: form.unit_id,
|
unit_id: form.unit_id,
|
||||||
expiry_date: form.expiry_date || null,
|
expiry_date: form.expiry_date || null,
|
||||||
|
low_stock_threshold: form.low_stock_threshold,
|
||||||
notes: form.notes.trim() || null
|
notes: form.notes.trim() || null
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -72,17 +72,53 @@
|
|||||||
{{ expiryText }}
|
{{ expiryText }}
|
||||||
</UBadge>
|
</UBadge>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- Low Stock Warning -->
|
||||||
|
<div v-if="isLowStock" class="text-xs">
|
||||||
|
<UBadge
|
||||||
|
color="orange"
|
||||||
|
variant="soft"
|
||||||
|
class="w-full justify-center"
|
||||||
|
>
|
||||||
|
<UIcon name="i-heroicons-exclamation-triangle" class="mr-1" />
|
||||||
|
Low stock ({{ item.quantity }}/{{ item.low_stock_threshold }})
|
||||||
|
</UBadge>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Action Buttons -->
|
<!-- Action Buttons -->
|
||||||
<template #footer>
|
<template #footer>
|
||||||
<div class="flex gap-2">
|
<div class="flex flex-col gap-2">
|
||||||
|
<!-- Quick Actions Row -->
|
||||||
|
<div class="grid grid-cols-2 gap-2">
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-arrow-trending-down"
|
||||||
|
size="sm"
|
||||||
|
color="orange"
|
||||||
|
variant="soft"
|
||||||
|
@click="handleConsume"
|
||||||
|
:disabled="item.quantity <= 0.01"
|
||||||
|
>
|
||||||
|
Consume
|
||||||
|
</UButton>
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-arrow-trending-up"
|
||||||
|
size="sm"
|
||||||
|
color="green"
|
||||||
|
variant="soft"
|
||||||
|
@click="showRestockModal = true"
|
||||||
|
>
|
||||||
|
Restock
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Management Actions Row -->
|
||||||
|
<div class="grid grid-cols-2 gap-2">
|
||||||
<UButton
|
<UButton
|
||||||
icon="i-heroicons-pencil"
|
icon="i-heroicons-pencil"
|
||||||
size="sm"
|
size="sm"
|
||||||
color="gray"
|
color="gray"
|
||||||
variant="soft"
|
variant="soft"
|
||||||
class="flex-1"
|
|
||||||
@click="$emit('edit', item)"
|
@click="$emit('edit', item)"
|
||||||
>
|
>
|
||||||
Edit
|
Edit
|
||||||
@@ -97,7 +133,61 @@
|
|||||||
Delete
|
Delete
|
||||||
</UButton>
|
</UButton>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<!-- Restock Modal -->
|
||||||
|
<UModal v-model="showRestockModal">
|
||||||
|
<UCard>
|
||||||
|
<template #header>
|
||||||
|
<h3 class="text-lg font-semibold">Restock {{ item.name }}</h3>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<div class="space-y-4">
|
||||||
|
<div class="text-sm text-gray-600">
|
||||||
|
Current: <span class="font-semibold">{{ item.quantity }} {{ item.unit?.abbreviation }}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<UFormGroup label="Amount to add">
|
||||||
|
<UInput
|
||||||
|
v-model.number="restockAmount"
|
||||||
|
type="number"
|
||||||
|
min="0.01"
|
||||||
|
step="0.01"
|
||||||
|
size="lg"
|
||||||
|
autofocus
|
||||||
|
placeholder="e.g. 5"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<div v-if="restockAmount > 0" class="text-sm text-gray-600">
|
||||||
|
New total: <span class="font-semibold">{{ (Number(item.quantity) + Number(restockAmount)).toFixed(2) }} {{ item.unit?.abbreviation }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<template #footer>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<UButton
|
||||||
|
color="primary"
|
||||||
|
size="lg"
|
||||||
|
class="flex-1"
|
||||||
|
@click="handleRestock"
|
||||||
|
:disabled="!restockAmount || restockAmount <= 0"
|
||||||
|
>
|
||||||
|
Add {{ restockAmount || 0 }} {{ item.unit?.abbreviation }}
|
||||||
|
</UButton>
|
||||||
|
<UButton
|
||||||
|
color="gray"
|
||||||
|
size="lg"
|
||||||
|
variant="soft"
|
||||||
|
@click="showRestockModal = false"
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</UCard>
|
||||||
|
</UModal>
|
||||||
</UCard>
|
</UCard>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -106,12 +196,17 @@ const props = defineProps<{
|
|||||||
item: any
|
item: any
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
defineEmits<{
|
const emit = defineEmits<{
|
||||||
edit: [item: any]
|
edit: [item: any]
|
||||||
delete: [id: string]
|
delete: [id: string]
|
||||||
'update-quantity': [id: string, change: number]
|
'update-quantity': [id: string, change: number]
|
||||||
|
'consume': [id: string]
|
||||||
|
'restock': [id: string, amount: number]
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
|
const showRestockModal = ref(false)
|
||||||
|
const restockAmount = ref<number>(1)
|
||||||
|
|
||||||
// Calculate days until expiry
|
// Calculate days until expiry
|
||||||
const daysUntilExpiry = computed(() => {
|
const daysUntilExpiry = computed(() => {
|
||||||
if (!props.item.expiry_date) return null
|
if (!props.item.expiry_date) return null
|
||||||
@@ -145,4 +240,30 @@ const expiryText = computed(() => {
|
|||||||
if (daysUntilExpiry.value === 1) return 'Expires tomorrow'
|
if (daysUntilExpiry.value === 1) return 'Expires tomorrow'
|
||||||
return `Expires in ${daysUntilExpiry.value} days`
|
return `Expires in ${daysUntilExpiry.value} days`
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// Low stock detection
|
||||||
|
const isLowStock = computed(() => {
|
||||||
|
if (!props.item.low_stock_threshold) return false
|
||||||
|
return Number(props.item.quantity) <= Number(props.item.low_stock_threshold)
|
||||||
|
})
|
||||||
|
|
||||||
|
// Quick actions
|
||||||
|
const handleConsume = () => {
|
||||||
|
emit('update-quantity', props.item.id, -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleRestock = () => {
|
||||||
|
if (restockAmount.value && restockAmount.value > 0) {
|
||||||
|
emit('update-quantity', props.item.id, restockAmount.value)
|
||||||
|
showRestockModal.value = false
|
||||||
|
restockAmount.value = 1 // Reset for next time
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset restock amount when modal closes
|
||||||
|
watch(showRestockModal, (isOpen) => {
|
||||||
|
if (!isOpen) {
|
||||||
|
restockAmount.value = 1
|
||||||
|
}
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -60,6 +60,7 @@ const { getInventory, deleteInventoryItem, updateQuantity } = useInventory()
|
|||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
refresh?: boolean
|
refresh?: boolean
|
||||||
tagFilters?: string[]
|
tagFilters?: string[]
|
||||||
|
searchQuery?: string
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
@@ -89,17 +90,27 @@ const loadInventory = async () => {
|
|||||||
|
|
||||||
// Computed filtered items
|
// Computed filtered items
|
||||||
const filteredItems = computed(() => {
|
const filteredItems = computed(() => {
|
||||||
if (!props.tagFilters || props.tagFilters.length === 0) {
|
let result = items.value
|
||||||
return items.value
|
|
||||||
|
// Filter by search query (case-insensitive)
|
||||||
|
if (props.searchQuery && props.searchQuery.trim()) {
|
||||||
|
const query = props.searchQuery.trim().toLowerCase()
|
||||||
|
result = result.filter(item =>
|
||||||
|
item.name.toLowerCase().includes(query)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Filter items that have at least one of the selected tags
|
// Filter by tags
|
||||||
return items.value.filter(item => {
|
if (props.tagFilters && props.tagFilters.length > 0) {
|
||||||
|
result = result.filter(item => {
|
||||||
if (!item.tags || item.tags.length === 0) return false
|
if (!item.tags || item.tags.length === 0) return false
|
||||||
|
|
||||||
const itemTagIds = item.tags.map((t: any) => t.tag.id)
|
const itemTagIds = item.tags.map((t: any) => t.tag.id)
|
||||||
return props.tagFilters!.some(filterId => itemTagIds.includes(filterId))
|
return props.tagFilters!.some(filterId => itemTagIds.includes(filterId))
|
||||||
})
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
})
|
})
|
||||||
|
|
||||||
const handleDelete = async (id: string) => {
|
const handleDelete = async (id: string) => {
|
||||||
|
|||||||
@@ -33,9 +33,36 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- Search & Filters -->
|
||||||
|
<UCard v-if="showFilters" class="mb-6 space-y-4">
|
||||||
|
<!-- Search Bar -->
|
||||||
|
<div>
|
||||||
|
<UFormGroup label="Search Items">
|
||||||
|
<UInput
|
||||||
|
v-model="searchQuery"
|
||||||
|
placeholder="Search by item name..."
|
||||||
|
icon="i-heroicons-magnifying-glass"
|
||||||
|
size="lg"
|
||||||
|
:ui="{ icon: { trailing: { pointer: '' } } }"
|
||||||
|
>
|
||||||
|
<template #trailing>
|
||||||
|
<UButton
|
||||||
|
v-if="searchQuery"
|
||||||
|
color="gray"
|
||||||
|
variant="link"
|
||||||
|
icon="i-heroicons-x-mark"
|
||||||
|
:padded="false"
|
||||||
|
@click="searchQuery = ''"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</UInput>
|
||||||
|
</UFormGroup>
|
||||||
|
</div>
|
||||||
|
|
||||||
<!-- Tag Filters -->
|
<!-- Tag Filters -->
|
||||||
<UCard v-if="showFilters" class="mb-6">
|
<div>
|
||||||
<TagsTagFilter v-model="selectedTagFilters" />
|
<TagsTagFilter v-model="selectedTagFilters" />
|
||||||
|
</div>
|
||||||
</UCard>
|
</UCard>
|
||||||
|
|
||||||
<!-- Add Item Form (Overlay) -->
|
<!-- Add Item Form (Overlay) -->
|
||||||
@@ -61,6 +88,7 @@
|
|||||||
ref="inventoryListRef"
|
ref="inventoryListRef"
|
||||||
:refresh="refreshKey"
|
:refresh="refreshKey"
|
||||||
:tag-filters="selectedTagFilters"
|
:tag-filters="selectedTagFilters"
|
||||||
|
:search-query="searchQuery"
|
||||||
@add-item="showAddForm = true"
|
@add-item="showAddForm = true"
|
||||||
@edit-item="editingItem = $event"
|
@edit-item="editingItem = $event"
|
||||||
/>
|
/>
|
||||||
@@ -82,6 +110,7 @@ const refreshKey = ref(0)
|
|||||||
const inventoryListRef = ref()
|
const inventoryListRef = ref()
|
||||||
const prefilledData = ref<any>(null)
|
const prefilledData = ref<any>(null)
|
||||||
const selectedTagFilters = ref<string[]>([])
|
const selectedTagFilters = ref<string[]>([])
|
||||||
|
const searchQuery = ref('')
|
||||||
|
|
||||||
// Handle scan-to-add flow (Issue #25)
|
// Handle scan-to-add flow (Issue #25)
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
|
|||||||
@@ -26,6 +26,8 @@ export interface Database {
|
|||||||
quantity: number
|
quantity: number
|
||||||
unit_id: string
|
unit_id: string
|
||||||
expiry_date: string | null
|
expiry_date: string | null
|
||||||
|
expires_at: string | null
|
||||||
|
low_stock_threshold: number | null
|
||||||
notes: string | null
|
notes: string | null
|
||||||
added_by: string
|
added_by: string
|
||||||
created_at: string
|
created_at: string
|
||||||
@@ -38,6 +40,8 @@ export interface Database {
|
|||||||
quantity: number
|
quantity: number
|
||||||
unit_id: string
|
unit_id: string
|
||||||
expiry_date?: string | null
|
expiry_date?: string | null
|
||||||
|
expires_at?: string | null
|
||||||
|
low_stock_threshold?: number | null
|
||||||
notes?: string | null
|
notes?: string | null
|
||||||
added_by: string
|
added_by: string
|
||||||
created_at?: string
|
created_at?: string
|
||||||
@@ -50,6 +54,8 @@ export interface Database {
|
|||||||
quantity?: number
|
quantity?: number
|
||||||
unit_id?: string
|
unit_id?: string
|
||||||
expiry_date?: string | null
|
expiry_date?: string | null
|
||||||
|
expires_at?: string | null
|
||||||
|
low_stock_threshold?: number | null
|
||||||
notes?: string | null
|
notes?: string | null
|
||||||
added_by?: string
|
added_by?: string
|
||||||
created_at?: string
|
created_at?: string
|
||||||
|
|||||||
376
docs/COOLIFY_DEPLOYMENT.md
Normal file
376
docs/COOLIFY_DEPLOYMENT.md
Normal file
@@ -0,0 +1,376 @@
|
|||||||
|
# Coolify Deployment Guide
|
||||||
|
|
||||||
|
## Prerequisites
|
||||||
|
|
||||||
|
- Coolify instance running (self-hosted or cloud)
|
||||||
|
- Gitea repository accessible to Coolify
|
||||||
|
- Supabase project (cloud or self-hosted)
|
||||||
|
- Domain name (optional, for production)
|
||||||
|
|
||||||
|
## Deployment Steps
|
||||||
|
|
||||||
|
### 1. Prepare Supabase
|
||||||
|
|
||||||
|
#### Option A: Supabase Cloud
|
||||||
|
|
||||||
|
1. Sign in to [supabase.com](https://supabase.com)
|
||||||
|
2. Create new project (or use existing)
|
||||||
|
3. Run migrations:
|
||||||
|
- Go to SQL Editor
|
||||||
|
- Copy/paste each file from `supabase/migrations/`
|
||||||
|
- Run in order: 001_, 002_, 003_, etc.
|
||||||
|
4. Get credentials:
|
||||||
|
- Project Settings → API
|
||||||
|
- Copy Project URL
|
||||||
|
- Copy `anon` / `public` key
|
||||||
|
|
||||||
|
#### Option B: Self-Hosted Supabase
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd supabase
|
||||||
|
docker-compose up -d
|
||||||
|
# Wait for services to start
|
||||||
|
docker-compose ps
|
||||||
|
```
|
||||||
|
|
||||||
|
Migrations run automatically from `supabase/migrations/` directory.
|
||||||
|
|
||||||
|
### 2. Add Resource in Coolify
|
||||||
|
|
||||||
|
1. Log in to Coolify
|
||||||
|
2. Click "New Resource"
|
||||||
|
3. Select "Docker Compose"
|
||||||
|
4. Choose deployment source:
|
||||||
|
- **Git Repository** (recommended)
|
||||||
|
- Public Git Repository
|
||||||
|
- Docker Image
|
||||||
|
|
||||||
|
### 3. Configure Git Repository
|
||||||
|
|
||||||
|
If using Git source:
|
||||||
|
|
||||||
|
1. **Repository URL:**
|
||||||
|
```
|
||||||
|
https://gitea.jeanlucmakiola.de/pantry-app/pantry.git
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Branch:** `main` (or `develop` for staging)
|
||||||
|
|
||||||
|
3. **Docker Compose File:** `docker-compose.prod.yml`
|
||||||
|
|
||||||
|
4. **Build Path:** Leave empty (uses root)
|
||||||
|
|
||||||
|
### 4. Set Environment Variables
|
||||||
|
|
||||||
|
In Coolify → Environment Variables, add:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Required
|
||||||
|
NUXT_PUBLIC_SUPABASE_URL=https://your-project.supabase.co
|
||||||
|
NUXT_PUBLIC_SUPABASE_ANON_KEY=your-anon-key-here
|
||||||
|
|
||||||
|
# Optional
|
||||||
|
NODE_ENV=production
|
||||||
|
HOST=0.0.0.0
|
||||||
|
PORT=3000
|
||||||
|
```
|
||||||
|
|
||||||
|
**Security Note:** Never commit `.env` files with real credentials!
|
||||||
|
|
||||||
|
### 5. Configure Domain (Optional)
|
||||||
|
|
||||||
|
1. In Coolify → Domains
|
||||||
|
2. Add your domain: `pantry.yourdomain.com`
|
||||||
|
3. Coolify auto-provisions SSL with Let's Encrypt
|
||||||
|
4. DNS: Point A record to Coolify server IP
|
||||||
|
|
||||||
|
### 6. Deploy
|
||||||
|
|
||||||
|
1. Click "Deploy" button
|
||||||
|
2. Watch build logs
|
||||||
|
3. Wait for "Deployed successfully" message
|
||||||
|
|
||||||
|
Expected deploy time: 2-5 minutes
|
||||||
|
|
||||||
|
### 7. Verify Deployment
|
||||||
|
|
||||||
|
#### Health Check
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl https://pantry.yourdomain.com/api/health
|
||||||
|
```
|
||||||
|
|
||||||
|
Expected response:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"timestamp": "2026-02-25T00:00:00.000Z",
|
||||||
|
"uptime": 123.456
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### PWA Check
|
||||||
|
|
||||||
|
1. Visit app in browser
|
||||||
|
2. Open DevTools → Application → Manifest
|
||||||
|
3. Verify manifest loads
|
||||||
|
4. Check Service Worker is registered
|
||||||
|
|
||||||
|
#### Functionality Test
|
||||||
|
|
||||||
|
- [ ] Homepage loads
|
||||||
|
- [ ] Can sign up / sign in
|
||||||
|
- [ ] Can view inventory
|
||||||
|
- [ ] Can add item
|
||||||
|
- [ ] PWA install prompt appears
|
||||||
|
- [ ] Offline mode works
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
### Build Fails
|
||||||
|
|
||||||
|
**Check build logs in Coolify:**
|
||||||
|
|
||||||
|
Common issues:
|
||||||
|
- Missing dependencies → Check package.json
|
||||||
|
- npm/node version mismatch → Use Node 20 Alpine
|
||||||
|
- Out of memory → Increase Coolify resource limits
|
||||||
|
|
||||||
|
### Container Won't Start
|
||||||
|
|
||||||
|
**Check runtime logs in Coolify:**
|
||||||
|
|
||||||
|
Common issues:
|
||||||
|
- Missing environment variables
|
||||||
|
- Port conflict (use 3000)
|
||||||
|
- Supabase connection timeout
|
||||||
|
|
||||||
|
### Supabase Connection Error
|
||||||
|
|
||||||
|
1. Verify Supabase URL is correct (no trailing slash)
|
||||||
|
2. Check anon key is valid
|
||||||
|
3. Test Supabase API directly:
|
||||||
|
```bash
|
||||||
|
curl https://your-project.supabase.co/rest/v1/
|
||||||
|
```
|
||||||
|
4. Check Supabase RLS policies allow access
|
||||||
|
|
||||||
|
### SSL Certificate Issues
|
||||||
|
|
||||||
|
Coolify should auto-provision Let's Encrypt cert:
|
||||||
|
- Ensure domain points to correct IP
|
||||||
|
- Check DNS propagation (can take 48h)
|
||||||
|
- Verify port 80/443 open on firewall
|
||||||
|
|
||||||
|
### App Loads but No Data
|
||||||
|
|
||||||
|
1. Check browser console for errors
|
||||||
|
2. Verify Supabase connection in Network tab
|
||||||
|
3. Check RLS policies in Supabase
|
||||||
|
4. Verify migrations ran successfully
|
||||||
|
|
||||||
|
## Continuous Deployment
|
||||||
|
|
||||||
|
### Auto-Deploy on Push
|
||||||
|
|
||||||
|
1. In Coolify → Settings → Webhooks
|
||||||
|
2. Copy webhook URL
|
||||||
|
3. In Gitea → Repo → Settings → Webhooks
|
||||||
|
4. Add webhook:
|
||||||
|
- URL: [Coolify webhook URL]
|
||||||
|
- Events: Push events
|
||||||
|
- Branch: main (or develop)
|
||||||
|
|
||||||
|
Now every git push triggers auto-deployment!
|
||||||
|
|
||||||
|
### Manual Deploy
|
||||||
|
|
||||||
|
In Coolify interface:
|
||||||
|
1. Click "Deploy Latest" button
|
||||||
|
2. Or use Coolify API:
|
||||||
|
```bash
|
||||||
|
curl -X POST https://coolify.example.com/api/deploy/[resource-id] \
|
||||||
|
-H "Authorization: Bearer [your-token]"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Monitoring
|
||||||
|
|
||||||
|
### View Logs
|
||||||
|
|
||||||
|
Coolify Dashboard → Logs tab
|
||||||
|
|
||||||
|
Real-time logs:
|
||||||
|
```bash
|
||||||
|
# If SSH access to Coolify host
|
||||||
|
docker logs -f [container-name]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Resource Usage
|
||||||
|
|
||||||
|
Coolify shows:
|
||||||
|
- CPU usage
|
||||||
|
- Memory usage
|
||||||
|
- Network traffic
|
||||||
|
- Storage
|
||||||
|
|
||||||
|
### Uptime Monitoring
|
||||||
|
|
||||||
|
Recommended external services:
|
||||||
|
- UptimeRobot (free tier)
|
||||||
|
- BetterStack
|
||||||
|
- Freshping
|
||||||
|
|
||||||
|
Monitor: `https://pantry.yourdomain.com/api/health`
|
||||||
|
|
||||||
|
## Rollback
|
||||||
|
|
||||||
|
### To Previous Version
|
||||||
|
|
||||||
|
1. In Coolify → Deployments
|
||||||
|
2. Click on previous successful deployment
|
||||||
|
3. Click "Redeploy"
|
||||||
|
|
||||||
|
### Using Git Tags
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Tag current release
|
||||||
|
git tag -a v0.1.0 -m "MVP Release"
|
||||||
|
git push origin v0.1.0
|
||||||
|
|
||||||
|
# Rollback by changing branch in Coolify
|
||||||
|
# Or deploy specific tag
|
||||||
|
```
|
||||||
|
|
||||||
|
## Performance Optimization
|
||||||
|
|
||||||
|
### Resource Limits
|
||||||
|
|
||||||
|
In docker-compose.prod.yml:
|
||||||
|
```yaml
|
||||||
|
deploy:
|
||||||
|
resources:
|
||||||
|
limits:
|
||||||
|
memory: 512M
|
||||||
|
cpus: '1.0'
|
||||||
|
```
|
||||||
|
|
||||||
|
Adjust based on traffic.
|
||||||
|
|
||||||
|
### CDN (Recommended)
|
||||||
|
|
||||||
|
1. Add domain to Cloudflare
|
||||||
|
2. Set DNS proxy (orange cloud)
|
||||||
|
3. Enable caching rules
|
||||||
|
4. Set SSL to "Full (strict)"
|
||||||
|
|
||||||
|
**Result:** ~50% faster load times globally
|
||||||
|
|
||||||
|
### Database Connection Pooling
|
||||||
|
|
||||||
|
If self-hosting Supabase:
|
||||||
|
- Use PgBouncer (included in Supabase)
|
||||||
|
- Set max connections: 20-50
|
||||||
|
|
||||||
|
## Security Checklist
|
||||||
|
|
||||||
|
Before going live:
|
||||||
|
|
||||||
|
- [ ] HTTPS enabled (automatic with Coolify)
|
||||||
|
- [ ] Environment variables set (not in repo)
|
||||||
|
- [ ] Supabase RLS policies enabled
|
||||||
|
- [ ] Strong database passwords
|
||||||
|
- [ ] Firewall configured (only 80/443 open)
|
||||||
|
- [ ] Supabase auth configured
|
||||||
|
- [ ] Regular backups enabled
|
||||||
|
- [ ] Monitoring alerts set up
|
||||||
|
|
||||||
|
## Backup Strategy
|
||||||
|
|
||||||
|
### Database Backups
|
||||||
|
|
||||||
|
**Supabase Cloud:** Automatic daily backups
|
||||||
|
|
||||||
|
**Self-hosted:**
|
||||||
|
```bash
|
||||||
|
# Manual backup
|
||||||
|
docker exec supabase-db pg_dump -U postgres > backup.sql
|
||||||
|
|
||||||
|
# Automated (cron)
|
||||||
|
0 2 * * * /path/to/backup-script.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
### Volume Backups
|
||||||
|
|
||||||
|
Coolify persistent volumes:
|
||||||
|
```bash
|
||||||
|
docker run --rm \
|
||||||
|
-v coolify_pantry_data:/data \
|
||||||
|
-v $(pwd):/backup \
|
||||||
|
ubuntu tar czf /backup/pantry-backup.tar.gz /data
|
||||||
|
```
|
||||||
|
|
||||||
|
## Staging Environment
|
||||||
|
|
||||||
|
Recommended setup:
|
||||||
|
|
||||||
|
**Production:**
|
||||||
|
- Branch: `main`
|
||||||
|
- Domain: `pantry.yourdomain.com`
|
||||||
|
- Supabase: Production project
|
||||||
|
|
||||||
|
**Staging:**
|
||||||
|
- Branch: `develop`
|
||||||
|
- Domain: `staging.pantry.yourdomain.com`
|
||||||
|
- Supabase: Separate test project
|
||||||
|
|
||||||
|
In Coolify, create two resources pointing to different branches.
|
||||||
|
|
||||||
|
## Cost Estimate
|
||||||
|
|
||||||
|
**Coolify (self-hosted):**
|
||||||
|
- VPS: $5-10/month (Hetzner, Digital Ocean)
|
||||||
|
- Domain: $10-15/year
|
||||||
|
- **Total:** ~$8/month
|
||||||
|
|
||||||
|
**Supabase Cloud:**
|
||||||
|
- Free tier: 500MB database, 1GB file storage
|
||||||
|
- Pro tier: $25/month (more resources)
|
||||||
|
|
||||||
|
**Total Cost:** $8-33/month for full stack
|
||||||
|
|
||||||
|
## Support
|
||||||
|
|
||||||
|
**Coolify:**
|
||||||
|
- Docs: https://coolify.io/docs
|
||||||
|
- Discord: https://discord.gg/coolify
|
||||||
|
|
||||||
|
**Pantry:**
|
||||||
|
- Issues: https://gitea.jeanlucmakiola.de/pantry-app/pantry/issues
|
||||||
|
- Discussions: TBD
|
||||||
|
|
||||||
|
## Post-Deployment Checklist
|
||||||
|
|
||||||
|
After first deployment:
|
||||||
|
|
||||||
|
- [ ] Health check passes
|
||||||
|
- [ ] Can access homepage
|
||||||
|
- [ ] Can sign up / sign in
|
||||||
|
- [ ] Can add inventory item
|
||||||
|
- [ ] PWA installs correctly
|
||||||
|
- [ ] Offline mode works
|
||||||
|
- [ ] SSL certificate valid
|
||||||
|
- [ ] Monitoring alerts configured
|
||||||
|
- [ ] Backup strategy in place
|
||||||
|
- [ ] Team notified of URL
|
||||||
|
|
||||||
|
## Next Steps
|
||||||
|
|
||||||
|
1. Run E2E tests (see E2E_TESTING.md)
|
||||||
|
2. Monitor for 24-48 hours
|
||||||
|
3. Announce to users
|
||||||
|
4. Set up analytics (optional)
|
||||||
|
5. Plan first iteration
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Congratulations! Your app is live! 🎉**
|
||||||
210
docs/DEPLOYMENT_CHECKLIST.md
Normal file
210
docs/DEPLOYMENT_CHECKLIST.md
Normal file
@@ -0,0 +1,210 @@
|
|||||||
|
# Deployment Checklist
|
||||||
|
|
||||||
|
Use this checklist to ensure a smooth deployment to production or staging.
|
||||||
|
|
||||||
|
## Pre-Deployment
|
||||||
|
|
||||||
|
### Code Quality
|
||||||
|
- [ ] All tests passing (E2E_TESTING.md)
|
||||||
|
- [ ] No critical bugs open
|
||||||
|
- [ ] Code reviewed and approved
|
||||||
|
- [ ] CHANGELOG.md updated
|
||||||
|
- [ ] Version tagged in git
|
||||||
|
|
||||||
|
### Database
|
||||||
|
- [ ] Migrations tested locally
|
||||||
|
- [ ] Migrations run on staging
|
||||||
|
- [ ] Seed data prepared (if needed)
|
||||||
|
- [ ] Backup created
|
||||||
|
- [ ] RLS policies verified
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
- [ ] Environment variables documented
|
||||||
|
- [ ] .env.production.example updated
|
||||||
|
- [ ] Secrets not in repository
|
||||||
|
- [ ] Docker files tested locally
|
||||||
|
- [ ] Health check endpoint working
|
||||||
|
|
||||||
|
### Documentation
|
||||||
|
- [ ] DEPLOYMENT.md reviewed
|
||||||
|
- [ ] README.md up to date
|
||||||
|
- [ ] API changes documented
|
||||||
|
- [ ] Known issues documented
|
||||||
|
|
||||||
|
## Deployment
|
||||||
|
|
||||||
|
### Coolify Setup
|
||||||
|
- [ ] Resource created in Coolify
|
||||||
|
- [ ] Git repository connected
|
||||||
|
- [ ] Branch configured (main/develop)
|
||||||
|
- [ ] docker-compose.prod.yml path set
|
||||||
|
- [ ] Environment variables added
|
||||||
|
- [ ] Domain configured (optional)
|
||||||
|
- [ ] SSL enabled
|
||||||
|
|
||||||
|
### Supabase Setup
|
||||||
|
- [ ] Project created
|
||||||
|
- [ ] Migrations run
|
||||||
|
- [ ] RLS policies enabled
|
||||||
|
- [ ] Auth providers configured
|
||||||
|
- [ ] Storage buckets created
|
||||||
|
- [ ] API keys copied
|
||||||
|
|
||||||
|
### Build & Deploy
|
||||||
|
- [ ] Triggered deployment
|
||||||
|
- [ ] Build logs checked (no errors)
|
||||||
|
- [ ] Container started successfully
|
||||||
|
- [ ] Health check passes
|
||||||
|
|
||||||
|
## Post-Deployment Verification
|
||||||
|
|
||||||
|
### Health Checks
|
||||||
|
- [ ] `/api/health` returns 200 OK
|
||||||
|
- [ ] Homepage loads
|
||||||
|
- [ ] Auth works (sign up/in)
|
||||||
|
- [ ] Database connection works
|
||||||
|
|
||||||
|
### PWA
|
||||||
|
- [ ] Manifest loads correctly
|
||||||
|
- [ ] Service worker registers
|
||||||
|
- [ ] Install prompt appears
|
||||||
|
- [ ] Offline mode works
|
||||||
|
- [ ] Icons display correctly
|
||||||
|
|
||||||
|
### Functionality
|
||||||
|
- [ ] Can create account
|
||||||
|
- [ ] Can sign in
|
||||||
|
- [ ] Can add inventory item
|
||||||
|
- [ ] Can edit item
|
||||||
|
- [ ] Can delete item
|
||||||
|
- [ ] Can add tags
|
||||||
|
- [ ] Can scan barcode (if implemented)
|
||||||
|
|
||||||
|
### Performance
|
||||||
|
- [ ] Page load < 3s
|
||||||
|
- [ ] Lighthouse score > 90 (PWA)
|
||||||
|
- [ ] No console errors
|
||||||
|
- [ ] No network errors
|
||||||
|
|
||||||
|
### Cross-Browser
|
||||||
|
- [ ] Chrome (desktop)
|
||||||
|
- [ ] Firefox (desktop)
|
||||||
|
- [ ] Safari (desktop)
|
||||||
|
- [ ] Chrome (mobile)
|
||||||
|
- [ ] Safari (iOS)
|
||||||
|
|
||||||
|
## Monitoring
|
||||||
|
|
||||||
|
### Setup
|
||||||
|
- [ ] Uptime monitoring configured
|
||||||
|
- [ ] Error tracking enabled (optional)
|
||||||
|
- [ ] Log aggregation set up
|
||||||
|
- [ ] Alerts configured
|
||||||
|
- [ ] Metrics dashboard created (optional)
|
||||||
|
|
||||||
|
### Checks
|
||||||
|
- [ ] CPU usage normal
|
||||||
|
- [ ] Memory usage normal
|
||||||
|
- [ ] Disk space sufficient
|
||||||
|
- [ ] No error spikes
|
||||||
|
|
||||||
|
## Security
|
||||||
|
|
||||||
|
### Verification
|
||||||
|
- [ ] HTTPS enabled
|
||||||
|
- [ ] SSL certificate valid
|
||||||
|
- [ ] No credentials exposed
|
||||||
|
- [ ] Firewall configured
|
||||||
|
- [ ] Supabase RLS enabled
|
||||||
|
- [ ] Strong admin passwords
|
||||||
|
|
||||||
|
### Compliance
|
||||||
|
- [ ] Privacy policy added (if required)
|
||||||
|
- [ ] Terms of service added (if required)
|
||||||
|
- [ ] Cookie notice (if applicable)
|
||||||
|
- [ ] GDPR compliance (if EU users)
|
||||||
|
|
||||||
|
## Communication
|
||||||
|
|
||||||
|
### Team
|
||||||
|
- [ ] Deployment announced
|
||||||
|
- [ ] Access details shared
|
||||||
|
- [ ] Rollback plan communicated
|
||||||
|
- [ ] Support plan established
|
||||||
|
|
||||||
|
### Users
|
||||||
|
- [ ] Announcement prepared
|
||||||
|
- [ ] Migration guide ready (if needed)
|
||||||
|
- [ ] Support channels available
|
||||||
|
- [ ] Feedback mechanism in place
|
||||||
|
|
||||||
|
## Backup & Recovery
|
||||||
|
|
||||||
|
### Backups
|
||||||
|
- [ ] Database backup verified
|
||||||
|
- [ ] Volume backup taken
|
||||||
|
- [ ] Backup restore tested
|
||||||
|
- [ ] Backup schedule configured
|
||||||
|
|
||||||
|
### Disaster Recovery
|
||||||
|
- [ ] Rollback plan documented
|
||||||
|
- [ ] Emergency contacts listed
|
||||||
|
- [ ] Recovery time objective (RTO) defined
|
||||||
|
- [ ] Recovery point objective (RPO) defined
|
||||||
|
|
||||||
|
## Post-Launch (24-48h)
|
||||||
|
|
||||||
|
### Monitoring
|
||||||
|
- [ ] Check logs for errors
|
||||||
|
- [ ] Review uptime metrics
|
||||||
|
- [ ] Analyze user behavior
|
||||||
|
- [ ] Check resource usage
|
||||||
|
|
||||||
|
### Optimization
|
||||||
|
- [ ] Identify slow queries
|
||||||
|
- [ ] Optimize heavy assets
|
||||||
|
- [ ] Review caching strategy
|
||||||
|
- [ ] Tune resource limits
|
||||||
|
|
||||||
|
### Feedback
|
||||||
|
- [ ] Collect user feedback
|
||||||
|
- [ ] Log issues found
|
||||||
|
- [ ] Prioritize fixes
|
||||||
|
- [ ] Plan next iteration
|
||||||
|
|
||||||
|
## Sign-Off
|
||||||
|
|
||||||
|
**Deployed by:** ___________________
|
||||||
|
**Date:** ___________________
|
||||||
|
**Environment:** Production / Staging
|
||||||
|
**Version:** ___________________
|
||||||
|
**Status:** ✅ Success / ⚠️ Issues / ❌ Failed
|
||||||
|
|
||||||
|
**Notes:**
|
||||||
|
_____________________________________________
|
||||||
|
_____________________________________________
|
||||||
|
_____________________________________________
|
||||||
|
|
||||||
|
**Next Review:** ___________________
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Rollback Criteria
|
||||||
|
|
||||||
|
Trigger rollback if:
|
||||||
|
- [ ] Critical bug discovered
|
||||||
|
- [ ] Data loss occurring
|
||||||
|
- [ ] Service unavailable > 15 min
|
||||||
|
- [ ] Security vulnerability found
|
||||||
|
- [ ] Performance degraded > 50%
|
||||||
|
|
||||||
|
**Rollback Procedure:**
|
||||||
|
1. In Coolify → Previous deployment → Redeploy
|
||||||
|
2. Or: `git revert` and redeploy
|
||||||
|
3. Notify team and users
|
||||||
|
4. Investigate root cause
|
||||||
|
5. Fix and redeploy
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Remember:** It's better to delay than to deploy broken code. Take your time with this checklist!
|
||||||
526
docs/E2E_TESTING.md
Normal file
526
docs/E2E_TESTING.md
Normal file
@@ -0,0 +1,526 @@
|
|||||||
|
# End-to-End Testing Guide
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
This guide covers manual end-to-end testing of the Pantry MVP. These tests ensure all critical user flows work correctly before release.
|
||||||
|
|
||||||
|
## Test Environment
|
||||||
|
|
||||||
|
**Requirements:**
|
||||||
|
- Browser: Chrome/Edge, Firefox, or Safari
|
||||||
|
- Resolution: 1920x1080 (desktop) and 390x844 (mobile)
|
||||||
|
- Network: Both online and offline modes
|
||||||
|
- Data: Fresh database with seed data
|
||||||
|
|
||||||
|
## Critical User Flows
|
||||||
|
|
||||||
|
### 1. Authentication Flow
|
||||||
|
|
||||||
|
#### Test 1.1: Sign Up
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Navigate to app home page
|
||||||
|
2. Click "Sign Up" or similar
|
||||||
|
3. Enter email: test@example.com
|
||||||
|
4. Enter password: SecurePass123!
|
||||||
|
5. Confirm password
|
||||||
|
6. Submit form
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Validation errors if password too weak
|
||||||
|
- ✅ Account created successfully
|
||||||
|
- ✅ Redirected to home/dashboard
|
||||||
|
- ✅ Welcome message shown (optional)
|
||||||
|
|
||||||
|
#### Test 1.2: Sign In
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Sign out if signed in
|
||||||
|
2. Navigate to home page
|
||||||
|
3. Click "Sign In"
|
||||||
|
4. Enter correct credentials
|
||||||
|
5. Submit form
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Successfully signed in
|
||||||
|
- ✅ Redirected to dashboard
|
||||||
|
- ✅ User menu shows email/name
|
||||||
|
|
||||||
|
#### Test 1.3: Sign In (Wrong Credentials)
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Try signing in with wrong password
|
||||||
|
2. Try signing in with non-existent email
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Error message shown
|
||||||
|
- ✅ Form not cleared (email retained)
|
||||||
|
- ✅ No redirect
|
||||||
|
- ✅ Try again allowed
|
||||||
|
|
||||||
|
#### Test 1.4: Sign Out
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. While signed in, click user menu
|
||||||
|
2. Click "Sign Out"
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Signed out successfully
|
||||||
|
- ✅ Redirected to sign-in page
|
||||||
|
- ✅ Cannot access protected pages
|
||||||
|
|
||||||
|
### 2. Inventory Management
|
||||||
|
|
||||||
|
#### Test 2.1: View Inventory List
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Sign in
|
||||||
|
2. Navigate to home/inventory page
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ List of pantry items displayed
|
||||||
|
- ✅ Items show: name, quantity, unit, location tags
|
||||||
|
- ✅ Empty state if no items
|
||||||
|
- ✅ Loading state while fetching
|
||||||
|
|
||||||
|
#### Test 2.2: Add Item Manually
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Navigate to inventory
|
||||||
|
2. Click "Add Item" button
|
||||||
|
3. Enter item details:
|
||||||
|
- Name: "Olive Oil"
|
||||||
|
- Quantity: 1
|
||||||
|
- Unit: L (liter)
|
||||||
|
- Tags: pantry, cooking-oil
|
||||||
|
4. Submit
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Form validation works
|
||||||
|
- ✅ Item appears in list immediately
|
||||||
|
- ✅ Success message shown
|
||||||
|
- ✅ Form cleared for next entry
|
||||||
|
|
||||||
|
#### Test 2.3: Edit Item
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Click on an existing item
|
||||||
|
2. Click "Edit" button
|
||||||
|
3. Change quantity from 1 to 0.5
|
||||||
|
4. Save changes
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Modal/form opens with current values
|
||||||
|
- ✅ Changes saved successfully
|
||||||
|
- ✅ List updated immediately
|
||||||
|
- ✅ No page reload needed
|
||||||
|
|
||||||
|
#### Test 2.4: Delete Item
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Click on an item
|
||||||
|
2. Click "Delete" button
|
||||||
|
3. Confirm deletion
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Confirmation dialog shown
|
||||||
|
- ✅ Item removed from list
|
||||||
|
- ✅ Success message shown
|
||||||
|
- ✅ Can undo (optional)
|
||||||
|
|
||||||
|
#### Test 2.5: Filter by Tag
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. View inventory list
|
||||||
|
2. Click on a tag filter (e.g., "fridge")
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Only items with that tag shown
|
||||||
|
- ✅ Filter state persists
|
||||||
|
- ✅ Clear filter button available
|
||||||
|
- ✅ Item count updated
|
||||||
|
|
||||||
|
### 3. Barcode Scanning
|
||||||
|
|
||||||
|
#### Test 3.1: Scan Known Product
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Navigate to Scan page
|
||||||
|
2. Allow camera permissions
|
||||||
|
3. Scan a barcode (e.g., Coca-Cola)
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Camera opens
|
||||||
|
- ✅ Barcode detected
|
||||||
|
- ✅ Product info fetched from Open Food Facts
|
||||||
|
- ✅ Pre-filled add form shown
|
||||||
|
- ✅ Can edit before adding
|
||||||
|
|
||||||
|
#### Test 3.2: Scan Unknown Product
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Scan a barcode not in database
|
||||||
|
2. Manual entry form shown
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ "Product not found" message
|
||||||
|
- ✅ Option to add manually
|
||||||
|
- ✅ Barcode pre-filled
|
||||||
|
- ✅ Can still save item
|
||||||
|
|
||||||
|
#### Test 3.3: Camera Permission Denied
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Block camera permission
|
||||||
|
2. Try to scan
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Permission prompt shown
|
||||||
|
- ✅ Helpful error message
|
||||||
|
- ✅ Instructions to enable camera
|
||||||
|
- ✅ Option to add manually
|
||||||
|
|
||||||
|
### 4. Tag Management
|
||||||
|
|
||||||
|
#### Test 4.1: Create Tag
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Go to Settings → Tags
|
||||||
|
2. Click "Add Tag"
|
||||||
|
3. Enter name: "freezer"
|
||||||
|
4. Pick color: blue
|
||||||
|
5. Save
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Tag created successfully
|
||||||
|
- ✅ Appears in tag list
|
||||||
|
- ✅ Available in item forms
|
||||||
|
- ✅ Color applied correctly
|
||||||
|
|
||||||
|
#### Test 4.2: Edit Tag
|
||||||
|
**Priority:** LOW
|
||||||
|
|
||||||
|
1. Click on existing tag
|
||||||
|
2. Change name or color
|
||||||
|
3. Save
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Tag updated
|
||||||
|
- ✅ All items with tag reflect changes
|
||||||
|
- ✅ No broken references
|
||||||
|
|
||||||
|
#### Test 4.3: Delete Tag
|
||||||
|
**Priority:** LOW
|
||||||
|
|
||||||
|
1. Click delete on a tag
|
||||||
|
2. Confirm
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Confirmation shown if tag in use
|
||||||
|
- ✅ Tag removed
|
||||||
|
- ✅ Items keep other tags
|
||||||
|
- ✅ No app crashes
|
||||||
|
|
||||||
|
### 5. PWA Installation
|
||||||
|
|
||||||
|
#### Test 5.1: Install Prompt (Desktop)
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Visit app on Chrome (desktop)
|
||||||
|
2. Wait 3 seconds
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Install banner appears
|
||||||
|
- ✅ Shows app icon and name
|
||||||
|
- ✅ Install button works
|
||||||
|
- ✅ Can dismiss and won't show for 7 days
|
||||||
|
|
||||||
|
#### Test 5.2: Install from Settings
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Go to Settings → App
|
||||||
|
2. Click "Install App" button
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Browser install dialog opens
|
||||||
|
- ✅ App installs to desktop/home screen
|
||||||
|
- ✅ Launches in standalone mode
|
||||||
|
- ✅ Icon and name correct
|
||||||
|
|
||||||
|
#### Test 5.3: iOS Safari Add to Home Screen
|
||||||
|
**Priority:** HIGH (iOS users)
|
||||||
|
|
||||||
|
1. Open app in Safari (iOS)
|
||||||
|
2. Tap Share button
|
||||||
|
3. Tap "Add to Home Screen"
|
||||||
|
4. Confirm
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Icon added to home screen
|
||||||
|
- ✅ Opens in standalone mode
|
||||||
|
- ✅ Splash screen shown
|
||||||
|
- ✅ No Safari UI
|
||||||
|
|
||||||
|
### 6. Offline Functionality
|
||||||
|
|
||||||
|
#### Test 6.1: Work Offline
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Load app while online
|
||||||
|
2. Navigate to all pages
|
||||||
|
3. Disconnect internet
|
||||||
|
4. Try navigating again
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Offline banner appears
|
||||||
|
- ✅ Previously visited pages load
|
||||||
|
- ✅ Cached data visible
|
||||||
|
- ✅ No white screens or errors
|
||||||
|
|
||||||
|
#### Test 6.2: Offline Indicator
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Go offline
|
||||||
|
2. Check for visual feedback
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ "You're offline" banner at top
|
||||||
|
- ✅ Banner is amber/warning color
|
||||||
|
- ✅ Icon indicates offline status
|
||||||
|
|
||||||
|
#### Test 6.3: Return Online
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Go offline
|
||||||
|
2. Wait a few seconds
|
||||||
|
3. Go online
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ "Back online!" banner (green)
|
||||||
|
- ✅ Auto-hides after 3 seconds
|
||||||
|
- ✅ Data syncs (if pending changes)
|
||||||
|
|
||||||
|
### 7. Responsive Design
|
||||||
|
|
||||||
|
#### Test 7.1: Mobile View
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Resize browser to 390px wide
|
||||||
|
2. Navigate all pages
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Navigation adapts (hamburger menu)
|
||||||
|
- ✅ Lists stack vertically
|
||||||
|
- ✅ Buttons are touch-friendly
|
||||||
|
- ✅ No horizontal scroll
|
||||||
|
- ✅ Text readable without zoom
|
||||||
|
|
||||||
|
#### Test 7.2: Tablet View
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Resize to 768px wide
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Layout adapts
|
||||||
|
- ✅ 2-column grids where appropriate
|
||||||
|
- ✅ Navigation hybrid or drawer
|
||||||
|
|
||||||
|
#### Test 7.3: Desktop View
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. View at 1920px wide
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Full navigation visible
|
||||||
|
- ✅ Multi-column layouts
|
||||||
|
- ✅ Content not stretched too wide
|
||||||
|
- ✅ Whitespace used effectively
|
||||||
|
|
||||||
|
### 8. Performance
|
||||||
|
|
||||||
|
#### Test 8.1: Page Load Time
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Clear cache
|
||||||
|
2. Load home page
|
||||||
|
3. Measure time to interactive
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ < 3 seconds on 4G
|
||||||
|
- ✅ < 1 second on repeat visit
|
||||||
|
- ✅ Loading indicators shown
|
||||||
|
|
||||||
|
#### Test 8.2: Lighthouse Score
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Open DevTools → Lighthouse
|
||||||
|
2. Run "Progressive Web App" audit
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ PWA score: 90+
|
||||||
|
- ✅ Performance: 80+
|
||||||
|
- ✅ Accessibility: 90+
|
||||||
|
- ✅ Best Practices: 90+
|
||||||
|
|
||||||
|
### 9. Accessibility
|
||||||
|
|
||||||
|
#### Test 9.1: Keyboard Navigation
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Use only keyboard (Tab, Enter, Esc)
|
||||||
|
2. Navigate entire app
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ All interactive elements reachable
|
||||||
|
- ✅ Focus indicators visible
|
||||||
|
- ✅ Modals can be closed with Esc
|
||||||
|
- ✅ Logical tab order
|
||||||
|
|
||||||
|
#### Test 9.2: Screen Reader
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Enable VoiceOver (Mac) or NVDA (Windows)
|
||||||
|
2. Navigate app
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ All text announced
|
||||||
|
- ✅ Images have alt text
|
||||||
|
- ✅ Form labels read correctly
|
||||||
|
- ✅ Buttons have meaningful labels
|
||||||
|
|
||||||
|
### 10. Error Handling
|
||||||
|
|
||||||
|
#### Test 10.1: Network Error
|
||||||
|
**Priority:** HIGH
|
||||||
|
|
||||||
|
1. Start action (add item)
|
||||||
|
2. Disable network mid-request
|
||||||
|
3. Check behavior
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ Error message shown
|
||||||
|
- ✅ Action can be retried
|
||||||
|
- ✅ Data not lost
|
||||||
|
- ✅ No crash
|
||||||
|
|
||||||
|
#### Test 10.2: Server Error (500)
|
||||||
|
**Priority:** MEDIUM
|
||||||
|
|
||||||
|
1. Trigger server error (if possible)
|
||||||
|
|
||||||
|
**Expected Results:**
|
||||||
|
- ✅ User-friendly error message
|
||||||
|
- ✅ No stack traces visible
|
||||||
|
- ✅ Option to try again
|
||||||
|
- ✅ Logs sent to monitoring (optional)
|
||||||
|
|
||||||
|
## Test Data Setup
|
||||||
|
|
||||||
|
### Seed Data Script
|
||||||
|
|
||||||
|
```sql
|
||||||
|
-- Run in Supabase SQL Editor
|
||||||
|
-- Insert test household
|
||||||
|
INSERT INTO households (id, name) VALUES
|
||||||
|
('test-household-1', 'Test Household');
|
||||||
|
|
||||||
|
-- Insert test tags
|
||||||
|
INSERT INTO tags (id, household_id, name, color) VALUES
|
||||||
|
('tag-1', 'test-household-1', 'fridge', '#3b82f6'),
|
||||||
|
('tag-2', 'test-household-1', 'pantry', '#10b981'),
|
||||||
|
('tag-3', 'test-household-1', 'freezer', '#6366f1');
|
||||||
|
|
||||||
|
-- Insert test units
|
||||||
|
INSERT INTO units (id, household_id, name, abbreviation, base_unit, conversion_factor) VALUES
|
||||||
|
('unit-1', 'test-household-1', 'liter', 'L', 'L', 1),
|
||||||
|
('unit-2', 'test-household-1', 'milliliter', 'mL', 'L', 0.001),
|
||||||
|
('unit-3', 'test-household-1', 'kilogram', 'kg', 'kg', 1);
|
||||||
|
|
||||||
|
-- Insert test items
|
||||||
|
INSERT INTO pantry_items (household_id, name, quantity, unit_id) VALUES
|
||||||
|
('test-household-1', 'Milk', 1, 'unit-1'),
|
||||||
|
('test-household-1', 'Rice', 2, 'unit-3'),
|
||||||
|
('test-household-1', 'Olive Oil', 0.5, 'unit-1');
|
||||||
|
```
|
||||||
|
|
||||||
|
## Test Execution Checklist
|
||||||
|
|
||||||
|
**Before Testing:**
|
||||||
|
- [ ] Fresh database with seed data
|
||||||
|
- [ ] Clear browser cache
|
||||||
|
- [ ] Clear localStorage
|
||||||
|
- [ ] Unregister service workers
|
||||||
|
- [ ] Sign out all accounts
|
||||||
|
|
||||||
|
**Test Environments:**
|
||||||
|
- [ ] Chrome (Windows/Mac/Linux)
|
||||||
|
- [ ] Firefox (Windows/Mac/Linux)
|
||||||
|
- [ ] Safari (Mac/iOS)
|
||||||
|
- [ ] Mobile Chrome (Android)
|
||||||
|
- [ ] Mobile Safari (iOS)
|
||||||
|
|
||||||
|
**Test Scenarios:**
|
||||||
|
- [ ] Authentication (4 tests)
|
||||||
|
- [ ] Inventory Management (5 tests)
|
||||||
|
- [ ] Barcode Scanning (3 tests)
|
||||||
|
- [ ] Tag Management (3 tests)
|
||||||
|
- [ ] PWA Installation (3 tests)
|
||||||
|
- [ ] Offline Functionality (3 tests)
|
||||||
|
- [ ] Responsive Design (3 tests)
|
||||||
|
- [ ] Performance (2 tests)
|
||||||
|
- [ ] Accessibility (2 tests)
|
||||||
|
- [ ] Error Handling (2 tests)
|
||||||
|
|
||||||
|
## Bug Report Template
|
||||||
|
|
||||||
|
```
|
||||||
|
**Title:** Short description
|
||||||
|
|
||||||
|
**Priority:** High / Medium / Low
|
||||||
|
|
||||||
|
**Environment:**
|
||||||
|
- Browser: Chrome 120.0
|
||||||
|
- OS: Windows 11
|
||||||
|
- Device: Desktop 1920x1080
|
||||||
|
|
||||||
|
**Steps to Reproduce:**
|
||||||
|
1. Navigate to...
|
||||||
|
2. Click on...
|
||||||
|
3. Enter...
|
||||||
|
|
||||||
|
**Expected Result:**
|
||||||
|
What should happen
|
||||||
|
|
||||||
|
**Actual Result:**
|
||||||
|
What actually happened
|
||||||
|
|
||||||
|
**Screenshots/Video:**
|
||||||
|
[Attach if applicable]
|
||||||
|
|
||||||
|
**Console Errors:**
|
||||||
|
[Copy any errors from DevTools console]
|
||||||
|
|
||||||
|
**Workaround:**
|
||||||
|
[If known]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Sign-off
|
||||||
|
|
||||||
|
**Tested by:** [Name]
|
||||||
|
**Date:** [Date]
|
||||||
|
**Environment:** [Browser, OS]
|
||||||
|
**Total Tests:** [X]
|
||||||
|
**Passed:** [X]
|
||||||
|
**Failed:** [X]
|
||||||
|
**Blocked:** [X]
|
||||||
|
|
||||||
|
**Critical Issues Found:** [List or "None"]
|
||||||
|
**Recommendation:** ✅ Ready for Release / ❌ Needs Fixes
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Next Steps:**
|
||||||
|
- Fix critical issues
|
||||||
|
- Retest failed scenarios
|
||||||
|
- Document known limitations
|
||||||
|
- Prepare release notes
|
||||||
26
supabase/migrations/006_add_expiry_lowstock.sql
Normal file
26
supabase/migrations/006_add_expiry_lowstock.sql
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
-- Migration: Add expiry date and low-stock threshold tracking
|
||||||
|
-- Issues: #63 (expiry tracking), #67 (low-stock threshold)
|
||||||
|
-- Created: 2026-02-25
|
||||||
|
|
||||||
|
-- Note: expiry_date already exists as DATE type. Adding expires_at as TIMESTAMPTZ for consistency
|
||||||
|
-- and low_stock_threshold for threshold tracking.
|
||||||
|
|
||||||
|
-- Add expires_at column for precise expiry date/time tracking (complementing existing expiry_date)
|
||||||
|
-- We'll keep both: expiry_date (DATE) for simple day-based expiry, expires_at (TIMESTAMPTZ) for precise tracking
|
||||||
|
ALTER TABLE inventory_items
|
||||||
|
ADD COLUMN expires_at TIMESTAMP WITH TIME ZONE DEFAULT NULL;
|
||||||
|
|
||||||
|
-- Add low_stock_threshold column for low-stock alerts
|
||||||
|
ALTER TABLE inventory_items
|
||||||
|
ADD COLUMN low_stock_threshold NUMERIC(10,2) DEFAULT NULL;
|
||||||
|
|
||||||
|
-- Add comments for documentation
|
||||||
|
COMMENT ON COLUMN inventory_items.expires_at IS 'Optional precise expiration timestamp. Complements expiry_date for items needing time-specific expiry.';
|
||||||
|
COMMENT ON COLUMN inventory_items.low_stock_threshold IS 'Minimum quantity threshold. Item is considered low-stock when quantity <= threshold. Null means no threshold set.';
|
||||||
|
|
||||||
|
-- Create index for efficient expiry queries (finding items expiring soon)
|
||||||
|
CREATE INDEX idx_inventory_items_expires_at ON inventory_items(expires_at) WHERE expires_at IS NOT NULL;
|
||||||
|
|
||||||
|
-- Create index for efficient low-stock queries
|
||||||
|
CREATE INDEX idx_inventory_items_low_stock ON inventory_items(quantity, low_stock_threshold)
|
||||||
|
WHERE low_stock_threshold IS NOT NULL;
|
||||||
Reference in New Issue
Block a user