Initial commit: YesChef family shopping list and recipe app
Backend (.NET 10 minimal API): - Vertical slice architecture with feature folders - Postgres via EF Core with initial migration - JWT auth with family invite code registration - REST endpoints for stores, shopping lists, items, recipes - SignalR hub for real-time list collaboration (per-list groups and lists-overview group for live list creation/archival/progress) - Multi-stage Dockerfile Frontend (SvelteKit + Svelte 5 runes, Tailwind v4): - Mobile-first PWA with web manifest and service worker - Bottom-nav layout, login/register, lists overview, list detail, stores management, recipes (list/create/detail with add-to-list) - SignalR client with reference-counted connection - Real-time updates on both lists overview and list detail pages Infrastructure: - docker-compose.yml with postgres, backend, frontend services and Traefik labels for path-based routing (/api, /hubs to backend) - .env.example with required config End-to-end tests (Playwright): - test-e2e.mjs: single-user flow (auth, stores, lists, items, recipes) - test-e2e-multiuser.mjs: two-user real-time sync coverage Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,4 @@
|
||||
POSTGRES_PASSWORD=change-me-strong-password
|
||||
JWT_SECRET=change-me-generate-a-random-64-char-string
|
||||
FAMILY_CODE=your-family-invite-phrase
|
||||
DOMAIN=yeschef.yourdomain.com
|
||||
+23
@@ -0,0 +1,23 @@
|
||||
## .NET
|
||||
bin/
|
||||
obj/
|
||||
*.user
|
||||
*.suo
|
||||
*.cache
|
||||
|
||||
## Node
|
||||
node_modules/
|
||||
build/
|
||||
.svelte-kit/
|
||||
|
||||
## Environment
|
||||
.env
|
||||
|
||||
## IDE
|
||||
.vs/
|
||||
.vscode/
|
||||
.idea/
|
||||
|
||||
## OS
|
||||
Thumbs.db
|
||||
.DS_Store
|
||||
@@ -0,0 +1,60 @@
|
||||
services:
|
||||
postgres:
|
||||
image: postgres:17
|
||||
environment:
|
||||
POSTGRES_DB: yeschef
|
||||
POSTGRES_USER: yeschef
|
||||
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
|
||||
volumes:
|
||||
- yeschef-pgdata:/var/lib/postgresql/data
|
||||
expose:
|
||||
- "5432"
|
||||
restart: unless-stopped
|
||||
healthcheck:
|
||||
test: ["CMD-SHELL", "pg_isready -U yeschef"]
|
||||
interval: 10s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
|
||||
backend:
|
||||
build:
|
||||
context: ./src/backend/YesChef.Api
|
||||
dockerfile: Dockerfile
|
||||
environment:
|
||||
ConnectionStrings__DefaultConnection: "Host=postgres;Database=yeschef;Username=yeschef;Password=${POSTGRES_PASSWORD}"
|
||||
Jwt__Secret: ${JWT_SECRET}
|
||||
FamilyCode: ${FAMILY_CODE}
|
||||
expose:
|
||||
- "5000"
|
||||
depends_on:
|
||||
postgres:
|
||||
condition: service_healthy
|
||||
restart: unless-stopped
|
||||
labels:
|
||||
- "traefik.enable=true"
|
||||
- "traefik.http.routers.yeschef-api.rule=Host(`${DOMAIN}`) && (PathPrefix(`/api`) || PathPrefix(`/hubs`) || Path(`/health`))"
|
||||
- "traefik.http.routers.yeschef-api.entrypoints=websecure"
|
||||
- "traefik.http.routers.yeschef-api.tls.certresolver=letsencrypt"
|
||||
- "traefik.http.services.yeschef-api.loadbalancer.server.port=5000"
|
||||
|
||||
frontend:
|
||||
build:
|
||||
context: ./src/frontend
|
||||
dockerfile: Dockerfile
|
||||
environment:
|
||||
ORIGIN: https://${DOMAIN}
|
||||
expose:
|
||||
- "3000"
|
||||
depends_on:
|
||||
- backend
|
||||
restart: unless-stopped
|
||||
labels:
|
||||
- "traefik.enable=true"
|
||||
- "traefik.http.routers.yeschef-web.rule=Host(`${DOMAIN}`)"
|
||||
- "traefik.http.routers.yeschef-web.entrypoints=websecure"
|
||||
- "traefik.http.routers.yeschef-web.tls.certresolver=letsencrypt"
|
||||
- "traefik.http.routers.yeschef-web.priority=1"
|
||||
- "traefik.http.services.yeschef-web.loadbalancer.server.port=3000"
|
||||
|
||||
volumes:
|
||||
yeschef-pgdata:
|
||||
Generated
+60
@@ -0,0 +1,60 @@
|
||||
{
|
||||
"name": "yeschef",
|
||||
"version": "1.0.0",
|
||||
"lockfileVersion": 3,
|
||||
"requires": true,
|
||||
"packages": {
|
||||
"": {
|
||||
"name": "yeschef",
|
||||
"version": "1.0.0",
|
||||
"license": "ISC",
|
||||
"dependencies": {
|
||||
"playwright": "^1.59.1"
|
||||
}
|
||||
},
|
||||
"node_modules/fsevents": {
|
||||
"version": "2.3.2",
|
||||
"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
|
||||
"integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
|
||||
"hasInstallScript": true,
|
||||
"license": "MIT",
|
||||
"optional": true,
|
||||
"os": [
|
||||
"darwin"
|
||||
],
|
||||
"engines": {
|
||||
"node": "^8.16.0 || ^10.6.0 || >=11.0.0"
|
||||
}
|
||||
},
|
||||
"node_modules/playwright": {
|
||||
"version": "1.59.1",
|
||||
"resolved": "https://registry.npmjs.org/playwright/-/playwright-1.59.1.tgz",
|
||||
"integrity": "sha512-C8oWjPR3F81yljW9o5OxcWzfh6avkVwDD2VYdwIGqTkl+OGFISgypqzfu7dOe4QNLL2aqcWBmI3PMtLIK233lw==",
|
||||
"license": "Apache-2.0",
|
||||
"dependencies": {
|
||||
"playwright-core": "1.59.1"
|
||||
},
|
||||
"bin": {
|
||||
"playwright": "cli.js"
|
||||
},
|
||||
"engines": {
|
||||
"node": ">=18"
|
||||
},
|
||||
"optionalDependencies": {
|
||||
"fsevents": "2.3.2"
|
||||
}
|
||||
},
|
||||
"node_modules/playwright-core": {
|
||||
"version": "1.59.1",
|
||||
"resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.59.1.tgz",
|
||||
"integrity": "sha512-HBV/RJg81z5BiiZ9yPzIiClYV/QMsDCKUyogwH9p3MCP6IYjUFu/MActgYAvK0oWyV9NlwM3GLBjADyWgydVyg==",
|
||||
"license": "Apache-2.0",
|
||||
"bin": {
|
||||
"playwright-core": "cli.js"
|
||||
},
|
||||
"engines": {
|
||||
"node": ">=18"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,16 @@
|
||||
{
|
||||
"name": "yeschef",
|
||||
"version": "1.0.0",
|
||||
"description": "",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
"test": "echo \"Error: no test specified\" && exit 1"
|
||||
},
|
||||
"keywords": [],
|
||||
"author": "",
|
||||
"license": "ISC",
|
||||
"type": "commonjs",
|
||||
"dependencies": {
|
||||
"playwright": "^1.59.1"
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,60 @@
|
||||
using Microsoft.AspNetCore.Identity;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using YesChef.Api.Data;
|
||||
using YesChef.Api.Entities;
|
||||
|
||||
namespace YesChef.Api.Auth;
|
||||
|
||||
public static class AuthEndpoints
|
||||
{
|
||||
public record RegisterRequest(string Name, string Password, string FamilyCode);
|
||||
public record LoginRequest(string Name, string Password);
|
||||
public record AuthResponse(string Token, string Name);
|
||||
|
||||
public static RouteGroupBuilder MapAuthEndpoints(this RouteGroupBuilder group)
|
||||
{
|
||||
var hasher = new PasswordHasher<User>();
|
||||
|
||||
group.MapPost("/register", async (RegisterRequest request, YesChefDb db, JwtTokenService jwt, IConfiguration config) =>
|
||||
{
|
||||
var familyCode = config["FamilyCode"];
|
||||
if (string.IsNullOrEmpty(familyCode) || request.FamilyCode != familyCode)
|
||||
return Results.BadRequest(new { error = "Invalid family code." });
|
||||
|
||||
if (await db.Users.AnyAsync(u => u.Name == request.Name))
|
||||
return Results.Conflict(new { error = "Name already taken." });
|
||||
|
||||
var user = new User { Name = request.Name, PasswordHash = "" };
|
||||
user.PasswordHash = hasher.HashPassword(user, request.Password);
|
||||
|
||||
db.Users.Add(user);
|
||||
await db.SaveChangesAsync();
|
||||
|
||||
var token = jwt.GenerateToken(user);
|
||||
return Results.Ok(new AuthResponse(token, user.Name));
|
||||
});
|
||||
|
||||
group.MapPost("/login", async (LoginRequest request, YesChefDb db, JwtTokenService jwt) =>
|
||||
{
|
||||
var user = await db.Users.FirstOrDefaultAsync(u => u.Name == request.Name);
|
||||
if (user is null)
|
||||
return Results.Unauthorized();
|
||||
|
||||
var result = hasher.VerifyHashedPassword(user, user.PasswordHash, request.Password);
|
||||
if (result == PasswordVerificationResult.Failed)
|
||||
return Results.Unauthorized();
|
||||
|
||||
var token = jwt.GenerateToken(user);
|
||||
return Results.Ok(new AuthResponse(token, user.Name));
|
||||
});
|
||||
|
||||
group.MapGet("/me", (HttpContext http) =>
|
||||
{
|
||||
var userId = http.User.GetUserId();
|
||||
var name = http.User.GetUserName();
|
||||
return Results.Ok(new { id = userId, name });
|
||||
}).RequireAuthorization();
|
||||
|
||||
return group;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,12 @@
|
||||
using System.Security.Claims;
|
||||
|
||||
namespace YesChef.Api.Auth;
|
||||
|
||||
public static class ClaimsPrincipalExtensions
|
||||
{
|
||||
public static int GetUserId(this ClaimsPrincipal principal) =>
|
||||
int.Parse(principal.FindFirstValue(ClaimTypes.NameIdentifier)!);
|
||||
|
||||
public static string GetUserName(this ClaimsPrincipal principal) =>
|
||||
principal.FindFirstValue(ClaimTypes.Name)!;
|
||||
}
|
||||
@@ -0,0 +1,29 @@
|
||||
using System.IdentityModel.Tokens.Jwt;
|
||||
using System.Security.Claims;
|
||||
using System.Text;
|
||||
using Microsoft.IdentityModel.Tokens;
|
||||
using YesChef.Api.Entities;
|
||||
|
||||
namespace YesChef.Api.Auth;
|
||||
|
||||
public class JwtTokenService(IConfiguration config)
|
||||
{
|
||||
public string GenerateToken(User user)
|
||||
{
|
||||
var key = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(config["Jwt:Secret"]!));
|
||||
var credentials = new SigningCredentials(key, SecurityAlgorithms.HmacSha256);
|
||||
|
||||
var claims = new[]
|
||||
{
|
||||
new Claim(ClaimTypes.NameIdentifier, user.Id.ToString()),
|
||||
new Claim(ClaimTypes.Name, user.Name)
|
||||
};
|
||||
|
||||
var token = new JwtSecurityToken(
|
||||
claims: claims,
|
||||
expires: DateTime.UtcNow.AddDays(30),
|
||||
signingCredentials: credentials);
|
||||
|
||||
return new JwtSecurityTokenHandler().WriteToken(token);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,57 @@
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using YesChef.Api.Entities;
|
||||
|
||||
namespace YesChef.Api.Data;
|
||||
|
||||
public class YesChefDb(DbContextOptions<YesChefDb> options) : DbContext(options)
|
||||
{
|
||||
public DbSet<User> Users => Set<User>();
|
||||
public DbSet<Store> Stores => Set<Store>();
|
||||
public DbSet<ShoppingList> ShoppingLists => Set<ShoppingList>();
|
||||
public DbSet<ShoppingListItem> ShoppingListItems => Set<ShoppingListItem>();
|
||||
public DbSet<Recipe> Recipes => Set<Recipe>();
|
||||
public DbSet<RecipeIngredient> RecipeIngredients => Set<RecipeIngredient>();
|
||||
|
||||
protected override void OnModelCreating(ModelBuilder modelBuilder)
|
||||
{
|
||||
modelBuilder.Entity<User>(e =>
|
||||
{
|
||||
e.HasIndex(u => u.Name).IsUnique();
|
||||
e.Property(u => u.Name).HasMaxLength(100);
|
||||
});
|
||||
|
||||
modelBuilder.Entity<Store>(e =>
|
||||
{
|
||||
e.HasIndex(s => s.Name).IsUnique();
|
||||
e.Property(s => s.Name).HasMaxLength(100);
|
||||
});
|
||||
|
||||
modelBuilder.Entity<ShoppingList>(e =>
|
||||
{
|
||||
e.Property(l => l.Name).HasMaxLength(200);
|
||||
e.HasOne(l => l.Store).WithMany().HasForeignKey(l => l.StoreId);
|
||||
e.HasOne(l => l.CreatedByUser).WithMany().HasForeignKey(l => l.CreatedByUserId);
|
||||
e.HasMany(l => l.Items).WithOne(i => i.ShoppingList).HasForeignKey(i => i.ShoppingListId).OnDelete(DeleteBehavior.Cascade);
|
||||
});
|
||||
|
||||
modelBuilder.Entity<ShoppingListItem>(e =>
|
||||
{
|
||||
e.Property(i => i.Name).HasMaxLength(300);
|
||||
e.HasOne(i => i.CheckedByUser).WithMany().HasForeignKey(i => i.CheckedByUserId).OnDelete(DeleteBehavior.SetNull);
|
||||
e.HasOne(i => i.Recipe).WithMany().HasForeignKey(i => i.RecipeId).OnDelete(DeleteBehavior.SetNull);
|
||||
});
|
||||
|
||||
modelBuilder.Entity<Recipe>(e =>
|
||||
{
|
||||
e.Property(r => r.Title).HasMaxLength(300);
|
||||
e.HasOne(r => r.CreatedByUser).WithMany().HasForeignKey(r => r.CreatedByUserId);
|
||||
e.HasMany(r => r.Ingredients).WithOne(i => i.Recipe).HasForeignKey(i => i.RecipeId).OnDelete(DeleteBehavior.Cascade);
|
||||
});
|
||||
|
||||
modelBuilder.Entity<RecipeIngredient>(e =>
|
||||
{
|
||||
e.Property(i => i.Name).HasMaxLength(200);
|
||||
e.Property(i => i.Quantity).HasMaxLength(50);
|
||||
});
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,13 @@
|
||||
FROM mcr.microsoft.com/dotnet/sdk:10.0 AS build
|
||||
WORKDIR /src
|
||||
COPY YesChef.Api.csproj .
|
||||
RUN dotnet restore
|
||||
COPY . .
|
||||
RUN dotnet publish -c Release -o /app
|
||||
|
||||
FROM mcr.microsoft.com/dotnet/aspnet:10.0
|
||||
WORKDIR /app
|
||||
COPY --from=build /app .
|
||||
EXPOSE 5000
|
||||
ENV ASPNETCORE_URLS=http://+:5000
|
||||
ENTRYPOINT ["dotnet", "YesChef.Api.dll"]
|
||||
@@ -0,0 +1,16 @@
|
||||
namespace YesChef.Api.Entities;
|
||||
|
||||
public class Recipe
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public required string Title { get; set; }
|
||||
public string? Description { get; set; }
|
||||
public string? Instructions { get; set; }
|
||||
public int? Servings { get; set; }
|
||||
public string? SourceUrl { get; set; }
|
||||
public int CreatedByUserId { get; set; }
|
||||
public User CreatedByUser { get; set; } = null!;
|
||||
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
||||
public DateTime UpdatedAt { get; set; } = DateTime.UtcNow;
|
||||
public List<RecipeIngredient> Ingredients { get; set; } = [];
|
||||
}
|
||||
@@ -0,0 +1,11 @@
|
||||
namespace YesChef.Api.Entities;
|
||||
|
||||
public class RecipeIngredient
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public int RecipeId { get; set; }
|
||||
public Recipe Recipe { get; set; } = null!;
|
||||
public required string Name { get; set; }
|
||||
public string? Quantity { get; set; }
|
||||
public int SortOrder { get; set; }
|
||||
}
|
||||
@@ -0,0 +1,15 @@
|
||||
namespace YesChef.Api.Entities;
|
||||
|
||||
public class ShoppingList
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public required string Name { get; set; }
|
||||
public int StoreId { get; set; }
|
||||
public Store Store { get; set; } = null!;
|
||||
public bool IsArchived { get; set; }
|
||||
public int CreatedByUserId { get; set; }
|
||||
public User CreatedByUser { get; set; } = null!;
|
||||
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
||||
public DateTime UpdatedAt { get; set; } = DateTime.UtcNow;
|
||||
public List<ShoppingListItem> Items { get; set; } = [];
|
||||
}
|
||||
@@ -0,0 +1,16 @@
|
||||
namespace YesChef.Api.Entities;
|
||||
|
||||
public class ShoppingListItem
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public int ShoppingListId { get; set; }
|
||||
public ShoppingList ShoppingList { get; set; } = null!;
|
||||
public required string Name { get; set; }
|
||||
public bool IsChecked { get; set; }
|
||||
public int? CheckedByUserId { get; set; }
|
||||
public User? CheckedByUser { get; set; }
|
||||
public int SortOrder { get; set; }
|
||||
public int? RecipeId { get; set; }
|
||||
public Recipe? Recipe { get; set; }
|
||||
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
||||
}
|
||||
@@ -0,0 +1,9 @@
|
||||
namespace YesChef.Api.Entities;
|
||||
|
||||
public class Store
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public required string Name { get; set; }
|
||||
public int SortOrder { get; set; }
|
||||
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
||||
}
|
||||
@@ -0,0 +1,9 @@
|
||||
namespace YesChef.Api.Entities;
|
||||
|
||||
public class User
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public required string Name { get; set; }
|
||||
public required string PasswordHash { get; set; }
|
||||
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
||||
}
|
||||
@@ -0,0 +1,117 @@
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using YesChef.Api.Auth;
|
||||
using YesChef.Api.Data;
|
||||
using YesChef.Api.Entities;
|
||||
|
||||
namespace YesChef.Api.Features.Recipes;
|
||||
|
||||
public static class RecipeEndpoints
|
||||
{
|
||||
public record IngredientRequest(string Name, string? Quantity, int SortOrder);
|
||||
public record CreateRecipeRequest(string Title, string? Description, string? Instructions, int? Servings, string? SourceUrl, List<IngredientRequest> Ingredients);
|
||||
public record UpdateRecipeRequest(string Title, string? Description, string? Instructions, int? Servings, string? SourceUrl, List<IngredientRequest> Ingredients);
|
||||
|
||||
public static RouteGroupBuilder MapRecipeEndpoints(this RouteGroupBuilder group)
|
||||
{
|
||||
group.MapGet("/", async (YesChefDb db, string? q) =>
|
||||
{
|
||||
var query = db.Recipes.AsQueryable();
|
||||
if (!string.IsNullOrWhiteSpace(q))
|
||||
query = query.Where(r => r.Title.Contains(q));
|
||||
|
||||
return await query.OrderByDescending(r => r.UpdatedAt)
|
||||
.Select(r => new
|
||||
{
|
||||
r.Id,
|
||||
r.Title,
|
||||
r.Description,
|
||||
r.Servings,
|
||||
IngredientCount = r.Ingredients.Count,
|
||||
r.UpdatedAt
|
||||
})
|
||||
.ToListAsync();
|
||||
});
|
||||
|
||||
group.MapPost("/", async (CreateRecipeRequest request, YesChefDb db, HttpContext http) =>
|
||||
{
|
||||
var recipe = new Recipe
|
||||
{
|
||||
Title = request.Title,
|
||||
Description = request.Description,
|
||||
Instructions = request.Instructions,
|
||||
Servings = request.Servings,
|
||||
SourceUrl = request.SourceUrl,
|
||||
CreatedByUserId = http.User.GetUserId(),
|
||||
Ingredients = request.Ingredients.Select(i => new RecipeIngredient
|
||||
{
|
||||
Name = i.Name,
|
||||
Quantity = i.Quantity,
|
||||
SortOrder = i.SortOrder
|
||||
}).ToList()
|
||||
};
|
||||
|
||||
db.Recipes.Add(recipe);
|
||||
await db.SaveChangesAsync();
|
||||
return Results.Created($"/api/recipes/{recipe.Id}", new { recipe.Id, recipe.Title });
|
||||
});
|
||||
|
||||
group.MapGet("/{id:int}", async (int id, YesChefDb db) =>
|
||||
{
|
||||
var recipe = await db.Recipes
|
||||
.Include(r => r.Ingredients.OrderBy(i => i.SortOrder))
|
||||
.Include(r => r.CreatedByUser)
|
||||
.FirstOrDefaultAsync(r => r.Id == id);
|
||||
|
||||
if (recipe is null) return Results.NotFound();
|
||||
|
||||
return Results.Ok(new
|
||||
{
|
||||
recipe.Id,
|
||||
recipe.Title,
|
||||
recipe.Description,
|
||||
recipe.Instructions,
|
||||
recipe.Servings,
|
||||
recipe.SourceUrl,
|
||||
CreatedBy = recipe.CreatedByUser.Name,
|
||||
recipe.UpdatedAt,
|
||||
Ingredients = recipe.Ingredients.Select(i => new { i.Id, i.Name, i.Quantity, i.SortOrder })
|
||||
});
|
||||
});
|
||||
|
||||
group.MapPut("/{id:int}", async (int id, UpdateRecipeRequest request, YesChefDb db, HttpContext http) =>
|
||||
{
|
||||
var recipe = await db.Recipes.Include(r => r.Ingredients).FirstOrDefaultAsync(r => r.Id == id);
|
||||
if (recipe is null) return Results.NotFound();
|
||||
|
||||
recipe.Title = request.Title;
|
||||
recipe.Description = request.Description;
|
||||
recipe.Instructions = request.Instructions;
|
||||
recipe.Servings = request.Servings;
|
||||
recipe.SourceUrl = request.SourceUrl;
|
||||
recipe.UpdatedAt = DateTime.UtcNow;
|
||||
|
||||
db.RecipeIngredients.RemoveRange(recipe.Ingredients);
|
||||
recipe.Ingredients = request.Ingredients.Select(i => new RecipeIngredient
|
||||
{
|
||||
Name = i.Name,
|
||||
Quantity = i.Quantity,
|
||||
SortOrder = i.SortOrder
|
||||
}).ToList();
|
||||
|
||||
await db.SaveChangesAsync();
|
||||
return Results.Ok(new { recipe.Id, recipe.Title });
|
||||
});
|
||||
|
||||
group.MapDelete("/{id:int}", async (int id, YesChefDb db) =>
|
||||
{
|
||||
var recipe = await db.Recipes.FindAsync(id);
|
||||
if (recipe is null) return Results.NotFound();
|
||||
|
||||
db.Recipes.Remove(recipe);
|
||||
await db.SaveChangesAsync();
|
||||
return Results.NoContent();
|
||||
});
|
||||
|
||||
return group;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,230 @@
|
||||
using Microsoft.AspNetCore.SignalR;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using YesChef.Api.Auth;
|
||||
using YesChef.Api.Data;
|
||||
using YesChef.Api.Entities;
|
||||
|
||||
namespace YesChef.Api.Features.ShoppingLists;
|
||||
|
||||
public static class ShoppingListEndpoints
|
||||
{
|
||||
public record CreateListRequest(string Name, int StoreId);
|
||||
public record UpdateListRequest(string Name, int StoreId);
|
||||
public record AddItemRequest(string Name, int SortOrder = 0);
|
||||
|
||||
private static async Task BroadcastListSummary(IHubContext<ShoppingListHub> hub, YesChefDb db, int listId)
|
||||
{
|
||||
var summary = await db.ShoppingLists
|
||||
.Where(l => l.Id == listId)
|
||||
.Select(l => new
|
||||
{
|
||||
l.Id,
|
||||
l.Name,
|
||||
Store = new { l.Store.Id, l.Store.Name },
|
||||
ItemCount = l.Items.Count,
|
||||
CheckedCount = l.Items.Count(i => i.IsChecked),
|
||||
l.UpdatedAt
|
||||
})
|
||||
.FirstAsync();
|
||||
|
||||
await hub.Clients.Group("lists-overview").SendAsync("ListSummaryUpdated", summary);
|
||||
}
|
||||
|
||||
public static RouteGroupBuilder MapShoppingListEndpoints(this RouteGroupBuilder group)
|
||||
{
|
||||
group.MapGet("/", async (YesChefDb db, int? storeId) =>
|
||||
{
|
||||
var query = db.ShoppingLists
|
||||
.Where(l => !l.IsArchived)
|
||||
.Include(l => l.Store)
|
||||
.Include(l => l.Items)
|
||||
.AsQueryable();
|
||||
|
||||
if (storeId.HasValue)
|
||||
query = query.Where(l => l.StoreId == storeId);
|
||||
|
||||
return await query.OrderByDescending(l => l.UpdatedAt)
|
||||
.Select(l => new
|
||||
{
|
||||
l.Id,
|
||||
l.Name,
|
||||
Store = new { l.Store.Id, l.Store.Name },
|
||||
ItemCount = l.Items.Count,
|
||||
CheckedCount = l.Items.Count(i => i.IsChecked),
|
||||
l.UpdatedAt
|
||||
})
|
||||
.ToListAsync();
|
||||
});
|
||||
|
||||
group.MapPost("/", async (CreateListRequest request, YesChefDb db, HttpContext http, IHubContext<ShoppingListHub> hub) =>
|
||||
{
|
||||
var list = new ShoppingList
|
||||
{
|
||||
Name = request.Name,
|
||||
StoreId = request.StoreId,
|
||||
CreatedByUserId = http.User.GetUserId()
|
||||
};
|
||||
db.ShoppingLists.Add(list);
|
||||
await db.SaveChangesAsync();
|
||||
|
||||
var store = await db.Stores.FindAsync(list.StoreId);
|
||||
await hub.Clients.Group("lists-overview").SendAsync("ListCreated", new
|
||||
{
|
||||
list.Id,
|
||||
list.Name,
|
||||
Store = new { store!.Id, store.Name },
|
||||
ItemCount = 0,
|
||||
CheckedCount = 0,
|
||||
list.UpdatedAt
|
||||
});
|
||||
|
||||
return Results.Created($"/api/lists/{list.Id}", new { list.Id, list.Name, list.StoreId });
|
||||
});
|
||||
|
||||
group.MapGet("/{id:int}", async (int id, YesChefDb db) =>
|
||||
{
|
||||
var list = await db.ShoppingLists
|
||||
.Include(l => l.Store)
|
||||
.Include(l => l.Items.OrderBy(i => i.SortOrder))
|
||||
.ThenInclude(i => i.CheckedByUser)
|
||||
.Include(l => l.Items)
|
||||
.ThenInclude(i => i.Recipe)
|
||||
.FirstOrDefaultAsync(l => l.Id == id);
|
||||
|
||||
if (list is null) return Results.NotFound();
|
||||
|
||||
return Results.Ok(new
|
||||
{
|
||||
list.Id,
|
||||
list.Name,
|
||||
Store = new { list.Store.Id, list.Store.Name },
|
||||
list.IsArchived,
|
||||
list.UpdatedAt,
|
||||
Items = list.Items.Select(i => new
|
||||
{
|
||||
i.Id,
|
||||
i.Name,
|
||||
i.IsChecked,
|
||||
CheckedByUserName = i.CheckedByUser?.Name,
|
||||
i.SortOrder,
|
||||
RecipeTitle = i.Recipe?.Title
|
||||
})
|
||||
});
|
||||
});
|
||||
|
||||
group.MapPut("/{id:int}", async (int id, UpdateListRequest request, YesChefDb db, IHubContext<ShoppingListHub> hub) =>
|
||||
{
|
||||
var list = await db.ShoppingLists.FindAsync(id);
|
||||
if (list is null) return Results.NotFound();
|
||||
|
||||
list.Name = request.Name;
|
||||
list.StoreId = request.StoreId;
|
||||
list.UpdatedAt = DateTime.UtcNow;
|
||||
await db.SaveChangesAsync();
|
||||
|
||||
await hub.Clients.Group($"list-{id}").SendAsync("ListUpdated", new { list.Id, list.Name, list.StoreId });
|
||||
return Results.Ok();
|
||||
});
|
||||
|
||||
group.MapDelete("/{id:int}", async (int id, YesChefDb db, IHubContext<ShoppingListHub> hub) =>
|
||||
{
|
||||
var list = await db.ShoppingLists.FindAsync(id);
|
||||
if (list is null) return Results.NotFound();
|
||||
|
||||
list.IsArchived = true;
|
||||
list.UpdatedAt = DateTime.UtcNow;
|
||||
await db.SaveChangesAsync();
|
||||
|
||||
await hub.Clients.Group("lists-overview").SendAsync("ListArchived", new { list.Id });
|
||||
|
||||
return Results.NoContent();
|
||||
});
|
||||
|
||||
group.MapPost("/{listId:int}/items", async (int listId, AddItemRequest request, YesChefDb db, IHubContext<ShoppingListHub> hub) =>
|
||||
{
|
||||
if (!await db.ShoppingLists.AnyAsync(l => l.Id == listId)) return Results.NotFound();
|
||||
|
||||
var item = new ShoppingListItem
|
||||
{
|
||||
ShoppingListId = listId,
|
||||
Name = request.Name,
|
||||
SortOrder = request.SortOrder
|
||||
};
|
||||
db.ShoppingListItems.Add(item);
|
||||
|
||||
var list = await db.ShoppingLists.FindAsync(listId);
|
||||
list!.UpdatedAt = DateTime.UtcNow;
|
||||
|
||||
await db.SaveChangesAsync();
|
||||
|
||||
await hub.Clients.Group($"list-{listId}").SendAsync("ItemAdded", new { item.Id, item.Name, item.SortOrder });
|
||||
await BroadcastListSummary(hub, db, listId);
|
||||
return Results.Created($"/api/lists/{listId}/items/{item.Id}", new { item.Id, item.Name, item.SortOrder });
|
||||
});
|
||||
|
||||
group.MapPatch("/{listId:int}/items/{itemId:int}/check", async (int listId, int itemId, YesChefDb db, IHubContext<ShoppingListHub> hub, HttpContext http) =>
|
||||
{
|
||||
var item = await db.ShoppingListItems.FirstOrDefaultAsync(i => i.Id == itemId && i.ShoppingListId == listId);
|
||||
if (item is null) return Results.NotFound();
|
||||
|
||||
var userId = http.User.GetUserId();
|
||||
item.IsChecked = !item.IsChecked;
|
||||
item.CheckedByUserId = item.IsChecked ? userId : null;
|
||||
await db.SaveChangesAsync();
|
||||
|
||||
string? checkedByName = null;
|
||||
if (item.IsChecked)
|
||||
checkedByName = await db.Users.Where(u => u.Id == userId).Select(u => u.Name).FirstAsync();
|
||||
|
||||
await hub.Clients.Group($"list-{listId}").SendAsync("ItemChecked", new { item.Id, item.IsChecked, CheckedByUserName = checkedByName });
|
||||
await BroadcastListSummary(hub, db, listId);
|
||||
return Results.Ok();
|
||||
});
|
||||
|
||||
group.MapDelete("/{listId:int}/items/{itemId:int}", async (int listId, int itemId, YesChefDb db, IHubContext<ShoppingListHub> hub) =>
|
||||
{
|
||||
var item = await db.ShoppingListItems.FirstOrDefaultAsync(i => i.Id == itemId && i.ShoppingListId == listId);
|
||||
if (item is null) return Results.NotFound();
|
||||
|
||||
db.ShoppingListItems.Remove(item);
|
||||
await db.SaveChangesAsync();
|
||||
|
||||
await hub.Clients.Group($"list-{listId}").SendAsync("ItemRemoved", new { item.Id });
|
||||
await BroadcastListSummary(hub, db, listId);
|
||||
return Results.NoContent();
|
||||
});
|
||||
|
||||
group.MapPost("/{listId:int}/add-recipe/{recipeId:int}", async (int listId, int recipeId, YesChefDb db, IHubContext<ShoppingListHub> hub) =>
|
||||
{
|
||||
var list = await db.ShoppingLists.FindAsync(listId);
|
||||
if (list is null) return Results.NotFound();
|
||||
|
||||
var recipe = await db.Recipes.Include(r => r.Ingredients).FirstOrDefaultAsync(r => r.Id == recipeId);
|
||||
if (recipe is null) return Results.NotFound();
|
||||
|
||||
var maxSort = await db.ShoppingListItems.Where(i => i.ShoppingListId == listId).MaxAsync(i => (int?)i.SortOrder) ?? 0;
|
||||
|
||||
var newItems = recipe.Ingredients.Select((ing, idx) => new ShoppingListItem
|
||||
{
|
||||
ShoppingListId = listId,
|
||||
Name = string.IsNullOrEmpty(ing.Quantity) ? ing.Name : $"{ing.Quantity} {ing.Name}",
|
||||
SortOrder = maxSort + idx + 1,
|
||||
RecipeId = recipeId
|
||||
}).ToList();
|
||||
|
||||
db.ShoppingListItems.AddRange(newItems);
|
||||
list.UpdatedAt = DateTime.UtcNow;
|
||||
await db.SaveChangesAsync();
|
||||
|
||||
foreach (var item in newItems)
|
||||
{
|
||||
await hub.Clients.Group($"list-{listId}").SendAsync("ItemAdded", new { item.Id, item.Name, item.SortOrder, RecipeTitle = recipe.Title });
|
||||
}
|
||||
|
||||
await BroadcastListSummary(hub, db, listId);
|
||||
return Results.Ok(new { added = newItems.Count });
|
||||
});
|
||||
|
||||
return group;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,20 @@
|
||||
using Microsoft.AspNetCore.Authorization;
|
||||
using Microsoft.AspNetCore.SignalR;
|
||||
|
||||
namespace YesChef.Api.Features.ShoppingLists;
|
||||
|
||||
[Authorize]
|
||||
public class ShoppingListHub : Hub
|
||||
{
|
||||
public async Task JoinList(int listId) =>
|
||||
await Groups.AddToGroupAsync(Context.ConnectionId, $"list-{listId}");
|
||||
|
||||
public async Task LeaveList(int listId) =>
|
||||
await Groups.RemoveFromGroupAsync(Context.ConnectionId, $"list-{listId}");
|
||||
|
||||
public async Task JoinListsOverview() =>
|
||||
await Groups.AddToGroupAsync(Context.ConnectionId, "lists-overview");
|
||||
|
||||
public async Task LeaveListsOverview() =>
|
||||
await Groups.RemoveFromGroupAsync(Context.ConnectionId, "lists-overview");
|
||||
}
|
||||
@@ -0,0 +1,51 @@
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using YesChef.Api.Data;
|
||||
using YesChef.Api.Entities;
|
||||
|
||||
namespace YesChef.Api.Features.Stores;
|
||||
|
||||
public static class StoreEndpoints
|
||||
{
|
||||
public record CreateStoreRequest(string Name, int SortOrder = 0);
|
||||
public record UpdateStoreRequest(string Name, int SortOrder);
|
||||
|
||||
public static RouteGroupBuilder MapStoreEndpoints(this RouteGroupBuilder group)
|
||||
{
|
||||
group.MapGet("/", async (YesChefDb db) =>
|
||||
await db.Stores.OrderBy(s => s.SortOrder).ThenBy(s => s.Name).ToListAsync());
|
||||
|
||||
group.MapPost("/", async (CreateStoreRequest request, YesChefDb db) =>
|
||||
{
|
||||
var store = new Store { Name = request.Name, SortOrder = request.SortOrder };
|
||||
db.Stores.Add(store);
|
||||
await db.SaveChangesAsync();
|
||||
return Results.Created($"/api/stores/{store.Id}", store);
|
||||
});
|
||||
|
||||
group.MapPut("/{id:int}", async (int id, UpdateStoreRequest request, YesChefDb db) =>
|
||||
{
|
||||
var store = await db.Stores.FindAsync(id);
|
||||
if (store is null) return Results.NotFound();
|
||||
|
||||
store.Name = request.Name;
|
||||
store.SortOrder = request.SortOrder;
|
||||
await db.SaveChangesAsync();
|
||||
return Results.Ok(store);
|
||||
});
|
||||
|
||||
group.MapDelete("/{id:int}", async (int id, YesChefDb db) =>
|
||||
{
|
||||
var hasLists = await db.ShoppingLists.AnyAsync(l => l.StoreId == id);
|
||||
if (hasLists) return Results.BadRequest(new { error = "Store has shopping lists. Remove them first." });
|
||||
|
||||
var store = await db.Stores.FindAsync(id);
|
||||
if (store is null) return Results.NotFound();
|
||||
|
||||
db.Stores.Remove(store);
|
||||
await db.SaveChangesAsync();
|
||||
return Results.NoContent();
|
||||
});
|
||||
|
||||
return group;
|
||||
}
|
||||
}
|
||||
+311
@@ -0,0 +1,311 @@
|
||||
// <auto-generated />
|
||||
using System;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||
using Microsoft.EntityFrameworkCore.Migrations;
|
||||
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
|
||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
||||
using YesChef.Api.Data;
|
||||
|
||||
#nullable disable
|
||||
|
||||
namespace YesChef.Api.Migrations
|
||||
{
|
||||
[DbContext(typeof(YesChefDb))]
|
||||
[Migration("20260506041045_InitialCreate")]
|
||||
partial class InitialCreate
|
||||
{
|
||||
/// <inheritdoc />
|
||||
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||
{
|
||||
#pragma warning disable 612, 618
|
||||
modelBuilder
|
||||
.HasAnnotation("ProductVersion", "10.0.7")
|
||||
.HasAnnotation("Relational:MaxIdentifierLength", 63);
|
||||
|
||||
NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.Recipe", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<int>("CreatedByUserId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<string>("Description")
|
||||
.HasColumnType("text");
|
||||
|
||||
b.Property<string>("Instructions")
|
||||
.HasColumnType("text");
|
||||
|
||||
b.Property<int?>("Servings")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<string>("SourceUrl")
|
||||
.HasColumnType("text");
|
||||
|
||||
b.Property<string>("Title")
|
||||
.IsRequired()
|
||||
.HasMaxLength(300)
|
||||
.HasColumnType("character varying(300)");
|
||||
|
||||
b.Property<DateTime>("UpdatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("CreatedByUserId");
|
||||
|
||||
b.ToTable("Recipes");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.RecipeIngredient", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(200)
|
||||
.HasColumnType("character varying(200)");
|
||||
|
||||
b.Property<string>("Quantity")
|
||||
.HasMaxLength(50)
|
||||
.HasColumnType("character varying(50)");
|
||||
|
||||
b.Property<int>("RecipeId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<int>("SortOrder")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("RecipeId");
|
||||
|
||||
b.ToTable("RecipeIngredients");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingList", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<int>("CreatedByUserId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<bool>("IsArchived")
|
||||
.HasColumnType("boolean");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(200)
|
||||
.HasColumnType("character varying(200)");
|
||||
|
||||
b.Property<int>("StoreId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<DateTime>("UpdatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("CreatedByUserId");
|
||||
|
||||
b.HasIndex("StoreId");
|
||||
|
||||
b.ToTable("ShoppingLists");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingListItem", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<int?>("CheckedByUserId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<bool>("IsChecked")
|
||||
.HasColumnType("boolean");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(300)
|
||||
.HasColumnType("character varying(300)");
|
||||
|
||||
b.Property<int?>("RecipeId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<int>("ShoppingListId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<int>("SortOrder")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("CheckedByUserId");
|
||||
|
||||
b.HasIndex("RecipeId");
|
||||
|
||||
b.HasIndex("ShoppingListId");
|
||||
|
||||
b.ToTable("ShoppingListItems");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.Store", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(100)
|
||||
.HasColumnType("character varying(100)");
|
||||
|
||||
b.Property<int>("SortOrder")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("Name")
|
||||
.IsUnique();
|
||||
|
||||
b.ToTable("Stores");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.User", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(100)
|
||||
.HasColumnType("character varying(100)");
|
||||
|
||||
b.Property<string>("PasswordHash")
|
||||
.IsRequired()
|
||||
.HasColumnType("text");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("Name")
|
||||
.IsUnique();
|
||||
|
||||
b.ToTable("Users");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.Recipe", b =>
|
||||
{
|
||||
b.HasOne("YesChef.Api.Entities.User", "CreatedByUser")
|
||||
.WithMany()
|
||||
.HasForeignKey("CreatedByUserId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.Navigation("CreatedByUser");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.RecipeIngredient", b =>
|
||||
{
|
||||
b.HasOne("YesChef.Api.Entities.Recipe", "Recipe")
|
||||
.WithMany("Ingredients")
|
||||
.HasForeignKey("RecipeId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.Navigation("Recipe");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingList", b =>
|
||||
{
|
||||
b.HasOne("YesChef.Api.Entities.User", "CreatedByUser")
|
||||
.WithMany()
|
||||
.HasForeignKey("CreatedByUserId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.HasOne("YesChef.Api.Entities.Store", "Store")
|
||||
.WithMany()
|
||||
.HasForeignKey("StoreId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.Navigation("CreatedByUser");
|
||||
|
||||
b.Navigation("Store");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingListItem", b =>
|
||||
{
|
||||
b.HasOne("YesChef.Api.Entities.User", "CheckedByUser")
|
||||
.WithMany()
|
||||
.HasForeignKey("CheckedByUserId")
|
||||
.OnDelete(DeleteBehavior.SetNull);
|
||||
|
||||
b.HasOne("YesChef.Api.Entities.Recipe", "Recipe")
|
||||
.WithMany()
|
||||
.HasForeignKey("RecipeId")
|
||||
.OnDelete(DeleteBehavior.SetNull);
|
||||
|
||||
b.HasOne("YesChef.Api.Entities.ShoppingList", "ShoppingList")
|
||||
.WithMany("Items")
|
||||
.HasForeignKey("ShoppingListId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.Navigation("CheckedByUser");
|
||||
|
||||
b.Navigation("Recipe");
|
||||
|
||||
b.Navigation("ShoppingList");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.Recipe", b =>
|
||||
{
|
||||
b.Navigation("Ingredients");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingList", b =>
|
||||
{
|
||||
b.Navigation("Items");
|
||||
});
|
||||
#pragma warning restore 612, 618
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,230 @@
|
||||
using System;
|
||||
using Microsoft.EntityFrameworkCore.Migrations;
|
||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
||||
|
||||
#nullable disable
|
||||
|
||||
namespace YesChef.Api.Migrations
|
||||
{
|
||||
/// <inheritdoc />
|
||||
public partial class InitialCreate : Migration
|
||||
{
|
||||
/// <inheritdoc />
|
||||
protected override void Up(MigrationBuilder migrationBuilder)
|
||||
{
|
||||
migrationBuilder.CreateTable(
|
||||
name: "Stores",
|
||||
columns: table => new
|
||||
{
|
||||
Id = table.Column<int>(type: "integer", nullable: false)
|
||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
||||
Name = table.Column<string>(type: "character varying(100)", maxLength: 100, nullable: false),
|
||||
SortOrder = table.Column<int>(type: "integer", nullable: false),
|
||||
CreatedAt = table.Column<DateTime>(type: "timestamp with time zone", nullable: false)
|
||||
},
|
||||
constraints: table =>
|
||||
{
|
||||
table.PrimaryKey("PK_Stores", x => x.Id);
|
||||
});
|
||||
|
||||
migrationBuilder.CreateTable(
|
||||
name: "Users",
|
||||
columns: table => new
|
||||
{
|
||||
Id = table.Column<int>(type: "integer", nullable: false)
|
||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
||||
Name = table.Column<string>(type: "character varying(100)", maxLength: 100, nullable: false),
|
||||
PasswordHash = table.Column<string>(type: "text", nullable: false),
|
||||
CreatedAt = table.Column<DateTime>(type: "timestamp with time zone", nullable: false)
|
||||
},
|
||||
constraints: table =>
|
||||
{
|
||||
table.PrimaryKey("PK_Users", x => x.Id);
|
||||
});
|
||||
|
||||
migrationBuilder.CreateTable(
|
||||
name: "Recipes",
|
||||
columns: table => new
|
||||
{
|
||||
Id = table.Column<int>(type: "integer", nullable: false)
|
||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
||||
Title = table.Column<string>(type: "character varying(300)", maxLength: 300, nullable: false),
|
||||
Description = table.Column<string>(type: "text", nullable: true),
|
||||
Instructions = table.Column<string>(type: "text", nullable: true),
|
||||
Servings = table.Column<int>(type: "integer", nullable: true),
|
||||
SourceUrl = table.Column<string>(type: "text", nullable: true),
|
||||
CreatedByUserId = table.Column<int>(type: "integer", nullable: false),
|
||||
CreatedAt = table.Column<DateTime>(type: "timestamp with time zone", nullable: false),
|
||||
UpdatedAt = table.Column<DateTime>(type: "timestamp with time zone", nullable: false)
|
||||
},
|
||||
constraints: table =>
|
||||
{
|
||||
table.PrimaryKey("PK_Recipes", x => x.Id);
|
||||
table.ForeignKey(
|
||||
name: "FK_Recipes_Users_CreatedByUserId",
|
||||
column: x => x.CreatedByUserId,
|
||||
principalTable: "Users",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.Cascade);
|
||||
});
|
||||
|
||||
migrationBuilder.CreateTable(
|
||||
name: "ShoppingLists",
|
||||
columns: table => new
|
||||
{
|
||||
Id = table.Column<int>(type: "integer", nullable: false)
|
||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
||||
Name = table.Column<string>(type: "character varying(200)", maxLength: 200, nullable: false),
|
||||
StoreId = table.Column<int>(type: "integer", nullable: false),
|
||||
IsArchived = table.Column<bool>(type: "boolean", nullable: false),
|
||||
CreatedByUserId = table.Column<int>(type: "integer", nullable: false),
|
||||
CreatedAt = table.Column<DateTime>(type: "timestamp with time zone", nullable: false),
|
||||
UpdatedAt = table.Column<DateTime>(type: "timestamp with time zone", nullable: false)
|
||||
},
|
||||
constraints: table =>
|
||||
{
|
||||
table.PrimaryKey("PK_ShoppingLists", x => x.Id);
|
||||
table.ForeignKey(
|
||||
name: "FK_ShoppingLists_Stores_StoreId",
|
||||
column: x => x.StoreId,
|
||||
principalTable: "Stores",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.Cascade);
|
||||
table.ForeignKey(
|
||||
name: "FK_ShoppingLists_Users_CreatedByUserId",
|
||||
column: x => x.CreatedByUserId,
|
||||
principalTable: "Users",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.Cascade);
|
||||
});
|
||||
|
||||
migrationBuilder.CreateTable(
|
||||
name: "RecipeIngredients",
|
||||
columns: table => new
|
||||
{
|
||||
Id = table.Column<int>(type: "integer", nullable: false)
|
||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
||||
RecipeId = table.Column<int>(type: "integer", nullable: false),
|
||||
Name = table.Column<string>(type: "character varying(200)", maxLength: 200, nullable: false),
|
||||
Quantity = table.Column<string>(type: "character varying(50)", maxLength: 50, nullable: true),
|
||||
SortOrder = table.Column<int>(type: "integer", nullable: false)
|
||||
},
|
||||
constraints: table =>
|
||||
{
|
||||
table.PrimaryKey("PK_RecipeIngredients", x => x.Id);
|
||||
table.ForeignKey(
|
||||
name: "FK_RecipeIngredients_Recipes_RecipeId",
|
||||
column: x => x.RecipeId,
|
||||
principalTable: "Recipes",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.Cascade);
|
||||
});
|
||||
|
||||
migrationBuilder.CreateTable(
|
||||
name: "ShoppingListItems",
|
||||
columns: table => new
|
||||
{
|
||||
Id = table.Column<int>(type: "integer", nullable: false)
|
||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
||||
ShoppingListId = table.Column<int>(type: "integer", nullable: false),
|
||||
Name = table.Column<string>(type: "character varying(300)", maxLength: 300, nullable: false),
|
||||
IsChecked = table.Column<bool>(type: "boolean", nullable: false),
|
||||
CheckedByUserId = table.Column<int>(type: "integer", nullable: true),
|
||||
SortOrder = table.Column<int>(type: "integer", nullable: false),
|
||||
RecipeId = table.Column<int>(type: "integer", nullable: true),
|
||||
CreatedAt = table.Column<DateTime>(type: "timestamp with time zone", nullable: false)
|
||||
},
|
||||
constraints: table =>
|
||||
{
|
||||
table.PrimaryKey("PK_ShoppingListItems", x => x.Id);
|
||||
table.ForeignKey(
|
||||
name: "FK_ShoppingListItems_Recipes_RecipeId",
|
||||
column: x => x.RecipeId,
|
||||
principalTable: "Recipes",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.SetNull);
|
||||
table.ForeignKey(
|
||||
name: "FK_ShoppingListItems_ShoppingLists_ShoppingListId",
|
||||
column: x => x.ShoppingListId,
|
||||
principalTable: "ShoppingLists",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.Cascade);
|
||||
table.ForeignKey(
|
||||
name: "FK_ShoppingListItems_Users_CheckedByUserId",
|
||||
column: x => x.CheckedByUserId,
|
||||
principalTable: "Users",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.SetNull);
|
||||
});
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_RecipeIngredients_RecipeId",
|
||||
table: "RecipeIngredients",
|
||||
column: "RecipeId");
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_Recipes_CreatedByUserId",
|
||||
table: "Recipes",
|
||||
column: "CreatedByUserId");
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_ShoppingListItems_CheckedByUserId",
|
||||
table: "ShoppingListItems",
|
||||
column: "CheckedByUserId");
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_ShoppingListItems_RecipeId",
|
||||
table: "ShoppingListItems",
|
||||
column: "RecipeId");
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_ShoppingListItems_ShoppingListId",
|
||||
table: "ShoppingListItems",
|
||||
column: "ShoppingListId");
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_ShoppingLists_CreatedByUserId",
|
||||
table: "ShoppingLists",
|
||||
column: "CreatedByUserId");
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_ShoppingLists_StoreId",
|
||||
table: "ShoppingLists",
|
||||
column: "StoreId");
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_Stores_Name",
|
||||
table: "Stores",
|
||||
column: "Name",
|
||||
unique: true);
|
||||
|
||||
migrationBuilder.CreateIndex(
|
||||
name: "IX_Users_Name",
|
||||
table: "Users",
|
||||
column: "Name",
|
||||
unique: true);
|
||||
}
|
||||
|
||||
/// <inheritdoc />
|
||||
protected override void Down(MigrationBuilder migrationBuilder)
|
||||
{
|
||||
migrationBuilder.DropTable(
|
||||
name: "RecipeIngredients");
|
||||
|
||||
migrationBuilder.DropTable(
|
||||
name: "ShoppingListItems");
|
||||
|
||||
migrationBuilder.DropTable(
|
||||
name: "Recipes");
|
||||
|
||||
migrationBuilder.DropTable(
|
||||
name: "ShoppingLists");
|
||||
|
||||
migrationBuilder.DropTable(
|
||||
name: "Stores");
|
||||
|
||||
migrationBuilder.DropTable(
|
||||
name: "Users");
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,308 @@
|
||||
// <auto-generated />
|
||||
using System;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
|
||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
||||
using YesChef.Api.Data;
|
||||
|
||||
#nullable disable
|
||||
|
||||
namespace YesChef.Api.Migrations
|
||||
{
|
||||
[DbContext(typeof(YesChefDb))]
|
||||
partial class YesChefDbModelSnapshot : ModelSnapshot
|
||||
{
|
||||
protected override void BuildModel(ModelBuilder modelBuilder)
|
||||
{
|
||||
#pragma warning disable 612, 618
|
||||
modelBuilder
|
||||
.HasAnnotation("ProductVersion", "10.0.7")
|
||||
.HasAnnotation("Relational:MaxIdentifierLength", 63);
|
||||
|
||||
NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.Recipe", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<int>("CreatedByUserId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<string>("Description")
|
||||
.HasColumnType("text");
|
||||
|
||||
b.Property<string>("Instructions")
|
||||
.HasColumnType("text");
|
||||
|
||||
b.Property<int?>("Servings")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<string>("SourceUrl")
|
||||
.HasColumnType("text");
|
||||
|
||||
b.Property<string>("Title")
|
||||
.IsRequired()
|
||||
.HasMaxLength(300)
|
||||
.HasColumnType("character varying(300)");
|
||||
|
||||
b.Property<DateTime>("UpdatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("CreatedByUserId");
|
||||
|
||||
b.ToTable("Recipes");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.RecipeIngredient", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(200)
|
||||
.HasColumnType("character varying(200)");
|
||||
|
||||
b.Property<string>("Quantity")
|
||||
.HasMaxLength(50)
|
||||
.HasColumnType("character varying(50)");
|
||||
|
||||
b.Property<int>("RecipeId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<int>("SortOrder")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("RecipeId");
|
||||
|
||||
b.ToTable("RecipeIngredients");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingList", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<int>("CreatedByUserId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<bool>("IsArchived")
|
||||
.HasColumnType("boolean");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(200)
|
||||
.HasColumnType("character varying(200)");
|
||||
|
||||
b.Property<int>("StoreId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<DateTime>("UpdatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("CreatedByUserId");
|
||||
|
||||
b.HasIndex("StoreId");
|
||||
|
||||
b.ToTable("ShoppingLists");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingListItem", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<int?>("CheckedByUserId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<bool>("IsChecked")
|
||||
.HasColumnType("boolean");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(300)
|
||||
.HasColumnType("character varying(300)");
|
||||
|
||||
b.Property<int?>("RecipeId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<int>("ShoppingListId")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.Property<int>("SortOrder")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("CheckedByUserId");
|
||||
|
||||
b.HasIndex("RecipeId");
|
||||
|
||||
b.HasIndex("ShoppingListId");
|
||||
|
||||
b.ToTable("ShoppingListItems");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.Store", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(100)
|
||||
.HasColumnType("character varying(100)");
|
||||
|
||||
b.Property<int>("SortOrder")
|
||||
.HasColumnType("integer");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("Name")
|
||||
.IsUnique();
|
||||
|
||||
b.ToTable("Stores");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.User", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("integer");
|
||||
|
||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<DateTime>("CreatedAt")
|
||||
.HasColumnType("timestamp with time zone");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasMaxLength(100)
|
||||
.HasColumnType("character varying(100)");
|
||||
|
||||
b.Property<string>("PasswordHash")
|
||||
.IsRequired()
|
||||
.HasColumnType("text");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("Name")
|
||||
.IsUnique();
|
||||
|
||||
b.ToTable("Users");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.Recipe", b =>
|
||||
{
|
||||
b.HasOne("YesChef.Api.Entities.User", "CreatedByUser")
|
||||
.WithMany()
|
||||
.HasForeignKey("CreatedByUserId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.Navigation("CreatedByUser");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.RecipeIngredient", b =>
|
||||
{
|
||||
b.HasOne("YesChef.Api.Entities.Recipe", "Recipe")
|
||||
.WithMany("Ingredients")
|
||||
.HasForeignKey("RecipeId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.Navigation("Recipe");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingList", b =>
|
||||
{
|
||||
b.HasOne("YesChef.Api.Entities.User", "CreatedByUser")
|
||||
.WithMany()
|
||||
.HasForeignKey("CreatedByUserId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.HasOne("YesChef.Api.Entities.Store", "Store")
|
||||
.WithMany()
|
||||
.HasForeignKey("StoreId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.Navigation("CreatedByUser");
|
||||
|
||||
b.Navigation("Store");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingListItem", b =>
|
||||
{
|
||||
b.HasOne("YesChef.Api.Entities.User", "CheckedByUser")
|
||||
.WithMany()
|
||||
.HasForeignKey("CheckedByUserId")
|
||||
.OnDelete(DeleteBehavior.SetNull);
|
||||
|
||||
b.HasOne("YesChef.Api.Entities.Recipe", "Recipe")
|
||||
.WithMany()
|
||||
.HasForeignKey("RecipeId")
|
||||
.OnDelete(DeleteBehavior.SetNull);
|
||||
|
||||
b.HasOne("YesChef.Api.Entities.ShoppingList", "ShoppingList")
|
||||
.WithMany("Items")
|
||||
.HasForeignKey("ShoppingListId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.Navigation("CheckedByUser");
|
||||
|
||||
b.Navigation("Recipe");
|
||||
|
||||
b.Navigation("ShoppingList");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.Recipe", b =>
|
||||
{
|
||||
b.Navigation("Ingredients");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("YesChef.Api.Entities.ShoppingList", b =>
|
||||
{
|
||||
b.Navigation("Items");
|
||||
});
|
||||
#pragma warning restore 612, 618
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,69 @@
|
||||
using System.Text;
|
||||
using Microsoft.AspNetCore.Authentication.JwtBearer;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.IdentityModel.Tokens;
|
||||
using YesChef.Api.Auth;
|
||||
using YesChef.Api.Data;
|
||||
using YesChef.Api.Features.Recipes;
|
||||
using YesChef.Api.Features.ShoppingLists;
|
||||
using YesChef.Api.Features.Stores;
|
||||
|
||||
var builder = WebApplication.CreateBuilder(args);
|
||||
|
||||
builder.Services.AddDbContext<YesChefDb>(options =>
|
||||
options.UseNpgsql(builder.Configuration.GetConnectionString("DefaultConnection")));
|
||||
|
||||
builder.Services.AddSingleton<JwtTokenService>();
|
||||
|
||||
builder.Services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
|
||||
.AddJwtBearer(options =>
|
||||
{
|
||||
options.TokenValidationParameters = new TokenValidationParameters
|
||||
{
|
||||
ValidateIssuer = false,
|
||||
ValidateAudience = false,
|
||||
ValidateLifetime = true,
|
||||
ValidateIssuerSigningKey = true,
|
||||
IssuerSigningKey = new SymmetricSecurityKey(
|
||||
Encoding.UTF8.GetBytes(builder.Configuration["Jwt:Secret"]!))
|
||||
};
|
||||
|
||||
options.Events = new JwtBearerEvents
|
||||
{
|
||||
OnMessageReceived = context =>
|
||||
{
|
||||
var accessToken = context.Request.Query["access_token"];
|
||||
if (!string.IsNullOrEmpty(accessToken) && context.HttpContext.Request.Path.StartsWithSegments("/hubs"))
|
||||
context.Token = accessToken;
|
||||
return Task.CompletedTask;
|
||||
}
|
||||
};
|
||||
});
|
||||
|
||||
builder.Services.AddAuthorization();
|
||||
builder.Services.AddSignalR();
|
||||
|
||||
var app = builder.Build();
|
||||
|
||||
using (var scope = app.Services.CreateScope())
|
||||
{
|
||||
var db = scope.ServiceProvider.GetRequiredService<YesChefDb>();
|
||||
await db.Database.MigrateAsync();
|
||||
}
|
||||
|
||||
app.UseAuthentication();
|
||||
app.UseAuthorization();
|
||||
|
||||
app.MapGet("/health", async (YesChefDb db) =>
|
||||
{
|
||||
await db.Database.CanConnectAsync();
|
||||
return Results.Ok(new { status = "healthy" });
|
||||
});
|
||||
|
||||
app.MapGroup("/api/auth").MapAuthEndpoints();
|
||||
app.MapGroup("/api/stores").MapStoreEndpoints().RequireAuthorization();
|
||||
app.MapGroup("/api/lists").MapShoppingListEndpoints().RequireAuthorization();
|
||||
app.MapGroup("/api/recipes").MapRecipeEndpoints().RequireAuthorization();
|
||||
app.MapHub<ShoppingListHub>("/hubs/shopping-list");
|
||||
|
||||
app.Run();
|
||||
@@ -0,0 +1,23 @@
|
||||
{
|
||||
"$schema": "https://json.schemastore.org/launchsettings.json",
|
||||
"profiles": {
|
||||
"http": {
|
||||
"commandName": "Project",
|
||||
"dotnetRunMessages": true,
|
||||
"launchBrowser": false,
|
||||
"applicationUrl": "http://localhost:5291",
|
||||
"environmentVariables": {
|
||||
"ASPNETCORE_ENVIRONMENT": "Development"
|
||||
}
|
||||
},
|
||||
"https": {
|
||||
"commandName": "Project",
|
||||
"dotnetRunMessages": true,
|
||||
"launchBrowser": false,
|
||||
"applicationUrl": "https://localhost:7264;http://localhost:5291",
|
||||
"environmentVariables": {
|
||||
"ASPNETCORE_ENVIRONMENT": "Development"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,18 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk.Web">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net10.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="10.0.7" />
|
||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="10.0.7">
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" Version="10.0.1" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
@@ -0,0 +1,6 @@
|
||||
@YesChef.Api_HostAddress = http://localhost:5291
|
||||
|
||||
GET {{YesChef.Api_HostAddress}}/weatherforecast/
|
||||
Accept: application/json
|
||||
|
||||
###
|
||||
@@ -0,0 +1,8 @@
|
||||
{
|
||||
"Logging": {
|
||||
"LogLevel": {
|
||||
"Default": "Information",
|
||||
"Microsoft.AspNetCore": "Warning"
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,15 @@
|
||||
{
|
||||
"ConnectionStrings": {
|
||||
"DefaultConnection": "Host=localhost;Database=yeschef;Username=yeschef;Password=yeschef"
|
||||
},
|
||||
"Jwt": {
|
||||
"Secret": "dev-secret-key-change-in-production-must-be-at-least-32-chars!"
|
||||
},
|
||||
"FamilyCode": "dev-family-code",
|
||||
"Logging": {
|
||||
"LogLevel": {
|
||||
"Default": "Information",
|
||||
"Microsoft.AspNetCore": "Warning"
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,3 @@
|
||||
<Solution>
|
||||
<Project Path="YesChef.Api/YesChef.Api.csproj" />
|
||||
</Solution>
|
||||
@@ -0,0 +1,23 @@
|
||||
node_modules
|
||||
|
||||
# Output
|
||||
.output
|
||||
.vercel
|
||||
.netlify
|
||||
.wrangler
|
||||
/.svelte-kit
|
||||
/build
|
||||
|
||||
# OS
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
|
||||
# Env
|
||||
.env
|
||||
.env.*
|
||||
!.env.example
|
||||
!.env.test
|
||||
|
||||
# Vite
|
||||
vite.config.js.timestamp-*
|
||||
vite.config.ts.timestamp-*
|
||||
@@ -0,0 +1 @@
|
||||
engine-strict=true
|
||||
@@ -0,0 +1,15 @@
|
||||
FROM node:22-slim AS build
|
||||
WORKDIR /app
|
||||
COPY package*.json .
|
||||
RUN npm ci
|
||||
COPY . .
|
||||
RUN npm run build
|
||||
|
||||
FROM node:22-slim
|
||||
WORKDIR /app
|
||||
COPY --from=build /app/build .
|
||||
COPY --from=build /app/package.json .
|
||||
RUN npm ci --omit=dev
|
||||
EXPOSE 3000
|
||||
ENV PORT=3000
|
||||
CMD ["node", "index.js"]
|
||||
@@ -0,0 +1,42 @@
|
||||
# sv
|
||||
|
||||
Everything you need to build a Svelte project, powered by [`sv`](https://github.com/sveltejs/cli).
|
||||
|
||||
## Creating a project
|
||||
|
||||
If you're seeing this, you've probably already done this step. Congrats!
|
||||
|
||||
```sh
|
||||
# create a new project
|
||||
npx sv create my-app
|
||||
```
|
||||
|
||||
To recreate this project with the same configuration:
|
||||
|
||||
```sh
|
||||
# recreate this project
|
||||
npx sv@0.15.2 create --template minimal --types ts --no-install src/frontend
|
||||
```
|
||||
|
||||
## Developing
|
||||
|
||||
Once you've created a project and installed dependencies with `npm install` (or `pnpm install` or `yarn`), start a development server:
|
||||
|
||||
```sh
|
||||
npm run dev
|
||||
|
||||
# or start the server and open the app in a new browser tab
|
||||
npm run dev -- --open
|
||||
```
|
||||
|
||||
## Building
|
||||
|
||||
To create a production version of your app:
|
||||
|
||||
```sh
|
||||
npm run build
|
||||
```
|
||||
|
||||
You can preview the production build with `npm run preview`.
|
||||
|
||||
> To deploy your app, you may need to install an [adapter](https://svelte.dev/docs/kit/adapters) for your target environment.
|
||||
Generated
+2388
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,29 @@
|
||||
{
|
||||
"name": "frontend",
|
||||
"private": true,
|
||||
"version": "0.0.1",
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dev": "vite dev",
|
||||
"build": "vite build",
|
||||
"preview": "vite preview",
|
||||
"prepare": "svelte-kit sync || echo ''",
|
||||
"check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json",
|
||||
"check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@sveltejs/adapter-auto": "^7.0.1",
|
||||
"@sveltejs/kit": "^2.57.0",
|
||||
"@sveltejs/vite-plugin-svelte": "^7.0.0",
|
||||
"svelte": "^5.55.2",
|
||||
"svelte-check": "^4.4.6",
|
||||
"typescript": "^6.0.2",
|
||||
"vite": "^8.0.7"
|
||||
},
|
||||
"dependencies": {
|
||||
"@microsoft/signalr": "^10.0.0",
|
||||
"@sveltejs/adapter-node": "^5.5.4",
|
||||
"@tailwindcss/vite": "^4.2.4",
|
||||
"tailwindcss": "^4.2.4"
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,23 @@
|
||||
@import 'tailwindcss';
|
||||
|
||||
@theme {
|
||||
--color-primary: #16a34a;
|
||||
--color-primary-dark: #15803d;
|
||||
--color-primary-light: #22c55e;
|
||||
--color-danger: #dc2626;
|
||||
--color-danger-dark: #b91c1c;
|
||||
}
|
||||
|
||||
html {
|
||||
font-family:
|
||||
-apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, sans-serif;
|
||||
}
|
||||
|
||||
body {
|
||||
@apply bg-gray-50 text-gray-900;
|
||||
-webkit-tap-highlight-color: transparent;
|
||||
}
|
||||
|
||||
button {
|
||||
@apply cursor-pointer;
|
||||
}
|
||||
Vendored
+13
@@ -0,0 +1,13 @@
|
||||
// See https://svelte.dev/docs/kit/types#app.d.ts
|
||||
// for information about these interfaces
|
||||
declare global {
|
||||
namespace App {
|
||||
// interface Error {}
|
||||
// interface Locals {}
|
||||
// interface PageData {}
|
||||
// interface PageState {}
|
||||
// interface Platform {}
|
||||
}
|
||||
}
|
||||
|
||||
export {};
|
||||
@@ -0,0 +1,17 @@
|
||||
<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1, viewport-fit=cover" />
|
||||
<meta name="theme-color" content="#16a34a" />
|
||||
<meta name="apple-mobile-web-app-capable" content="yes" />
|
||||
<meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" />
|
||||
<link rel="manifest" href="/manifest.webmanifest" />
|
||||
<link rel="icon" href="/favicon.svg" type="image/svg+xml" />
|
||||
<link rel="apple-touch-icon" href="/icons/icon-192.png" />
|
||||
%sveltekit.head%
|
||||
</head>
|
||||
<body data-sveltekit-preload-data="hover">
|
||||
<div style="display: contents">%sveltekit.body%</div>
|
||||
</body>
|
||||
</html>
|
||||
@@ -0,0 +1,48 @@
|
||||
import { goto } from '$app/navigation';
|
||||
import { setLoggedIn } from '$lib/auth.svelte';
|
||||
|
||||
let token: string | null = null;
|
||||
|
||||
export function getToken(): string | null {
|
||||
if (token === null && typeof localStorage !== 'undefined') {
|
||||
token = localStorage.getItem('token');
|
||||
}
|
||||
return token;
|
||||
}
|
||||
|
||||
export function setToken(t: string) {
|
||||
token = t;
|
||||
localStorage.setItem('token', t);
|
||||
setLoggedIn(true);
|
||||
}
|
||||
|
||||
export function clearToken() {
|
||||
token = null;
|
||||
localStorage.removeItem('token');
|
||||
setLoggedIn(false);
|
||||
}
|
||||
|
||||
export async function api<T>(path: string, options: RequestInit = {}): Promise<T> {
|
||||
const t = getToken();
|
||||
const headers: Record<string, string> = {
|
||||
'Content-Type': 'application/json',
|
||||
...(options.headers as Record<string, string>)
|
||||
};
|
||||
if (t) headers['Authorization'] = `Bearer ${t}`;
|
||||
|
||||
const res = await fetch(path, { ...options, headers });
|
||||
|
||||
if (res.status === 401) {
|
||||
clearToken();
|
||||
goto('/login');
|
||||
throw new Error('Unauthorized');
|
||||
}
|
||||
|
||||
if (!res.ok) {
|
||||
const body = await res.json().catch(() => ({}));
|
||||
throw new Error(body.error || `Request failed: ${res.status}`);
|
||||
}
|
||||
|
||||
if (res.status === 204) return undefined as T;
|
||||
return res.json();
|
||||
}
|
||||
@@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" width="107" height="128" viewBox="0 0 107 128"><title>svelte-logo</title><path d="M94.157 22.819c-10.4-14.885-30.94-19.297-45.792-9.835L22.282 29.608A29.92 29.92 0 0 0 8.764 49.65a31.5 31.5 0 0 0 3.108 20.231 30 30 0 0 0-4.477 11.183 31.9 31.9 0 0 0 5.448 24.116c10.402 14.887 30.942 19.297 45.791 9.835l26.083-16.624A29.92 29.92 0 0 0 98.235 78.35a31.53 31.53 0 0 0-3.105-20.232 30 30 0 0 0 4.474-11.182 31.88 31.88 0 0 0-5.447-24.116" style="fill:#ff3e00"/><path d="M45.817 106.582a20.72 20.72 0 0 1-22.237-8.243 19.17 19.17 0 0 1-3.277-14.503 18 18 0 0 1 .624-2.435l.49-1.498 1.337.981a33.6 33.6 0 0 0 10.203 5.098l.97.294-.09.968a5.85 5.85 0 0 0 1.052 3.878 6.24 6.24 0 0 0 6.695 2.485 5.8 5.8 0 0 0 1.603-.704L69.27 76.28a5.43 5.43 0 0 0 2.45-3.631 5.8 5.8 0 0 0-.987-4.371 6.24 6.24 0 0 0-6.698-2.487 5.7 5.7 0 0 0-1.6.704l-9.953 6.345a19 19 0 0 1-5.296 2.326 20.72 20.72 0 0 1-22.237-8.243 19.17 19.17 0 0 1-3.277-14.502 17.99 17.99 0 0 1 8.13-12.052l26.081-16.623a19 19 0 0 1 5.3-2.329 20.72 20.72 0 0 1 22.237 8.243 19.17 19.17 0 0 1 3.277 14.503 18 18 0 0 1-.624 2.435l-.49 1.498-1.337-.98a33.6 33.6 0 0 0-10.203-5.1l-.97-.294.09-.968a5.86 5.86 0 0 0-1.052-3.878 6.24 6.24 0 0 0-6.696-2.485 5.8 5.8 0 0 0-1.602.704L37.73 51.72a5.42 5.42 0 0 0-2.449 3.63 5.79 5.79 0 0 0 .986 4.372 6.24 6.24 0 0 0 6.698 2.486 5.8 5.8 0 0 0 1.602-.704l9.952-6.342a19 19 0 0 1 5.295-2.328 20.72 20.72 0 0 1 22.237 8.242 19.17 19.17 0 0 1 3.277 14.503 18 18 0 0 1-8.13 12.053l-26.081 16.622a19 19 0 0 1-5.3 2.328" style="fill:#fff"/></svg>
|
||||
|
After Width: | Height: | Size: 1.5 KiB |
@@ -0,0 +1,15 @@
|
||||
let loggedIn = $state(false);
|
||||
|
||||
export function initAuth() {
|
||||
if (typeof localStorage !== 'undefined') {
|
||||
loggedIn = !!localStorage.getItem('token');
|
||||
}
|
||||
}
|
||||
|
||||
export function setLoggedIn(value: boolean) {
|
||||
loggedIn = value;
|
||||
}
|
||||
|
||||
export function isLoggedIn() {
|
||||
return loggedIn;
|
||||
}
|
||||
@@ -0,0 +1 @@
|
||||
// place files you want to import through the `$lib` alias in this folder.
|
||||
@@ -0,0 +1,37 @@
|
||||
import { HubConnectionBuilder, HubConnectionState, type HubConnection } from '@microsoft/signalr';
|
||||
import { getToken } from './api';
|
||||
|
||||
let connection: HubConnection | null = null;
|
||||
let refCount = 0;
|
||||
|
||||
function getConnection(): HubConnection {
|
||||
if (connection && connection.state !== HubConnectionState.Disconnected) {
|
||||
return connection;
|
||||
}
|
||||
|
||||
connection = new HubConnectionBuilder()
|
||||
.withUrl('/hubs/shopping-list', {
|
||||
accessTokenFactory: () => getToken() ?? ''
|
||||
})
|
||||
.withAutomaticReconnect([0, 1000, 5000, 10000, 30000])
|
||||
.build();
|
||||
|
||||
return connection;
|
||||
}
|
||||
|
||||
export async function startConnection(): Promise<HubConnection> {
|
||||
const conn = getConnection();
|
||||
if (conn.state === HubConnectionState.Disconnected) {
|
||||
await conn.start();
|
||||
}
|
||||
refCount++;
|
||||
return conn;
|
||||
}
|
||||
|
||||
export async function stopConnection() {
|
||||
refCount--;
|
||||
if (refCount <= 0 && connection && connection.state !== HubConnectionState.Disconnected) {
|
||||
refCount = 0;
|
||||
await connection.stop();
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,67 @@
|
||||
<script lang="ts">
|
||||
import '../app.css';
|
||||
import { page } from '$app/state';
|
||||
import { clearToken } from '$lib/api';
|
||||
import { initAuth, isLoggedIn } from '$lib/auth.svelte';
|
||||
import { goto } from '$app/navigation';
|
||||
import { onMount } from 'svelte';
|
||||
|
||||
let { children } = $props();
|
||||
|
||||
onMount(() => initAuth());
|
||||
|
||||
const loggedIn = $derived(isLoggedIn());
|
||||
const currentPath = $derived(page.url.pathname);
|
||||
|
||||
const navItems = [
|
||||
{ href: '/lists', label: 'Lists', icon: '📋' },
|
||||
{ href: '/recipes', label: 'Recipes', icon: '📖' },
|
||||
{ href: '/stores', label: 'Stores', icon: '🏪' }
|
||||
];
|
||||
|
||||
function logout() {
|
||||
clearToken();
|
||||
goto('/login');
|
||||
}
|
||||
</script>
|
||||
|
||||
{#if loggedIn && currentPath !== '/login'}
|
||||
<div class="flex min-h-dvh flex-col pb-16">
|
||||
<header class="sticky top-0 z-40 border-b border-gray-200 bg-white px-4 py-3">
|
||||
<div class="mx-auto flex max-w-lg items-center justify-between">
|
||||
<h1 class="text-xl font-bold text-primary">YesChef</h1>
|
||||
<button onclick={logout} class="text-sm text-gray-500">Sign out</button>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
<main class="mx-auto w-full max-w-lg flex-1 px-4 py-4">
|
||||
{@render children()}
|
||||
</main>
|
||||
|
||||
<nav class="fixed bottom-0 left-0 right-0 z-50 border-t border-gray-200 bg-white safe-bottom">
|
||||
<div class="mx-auto flex max-w-lg justify-around">
|
||||
{#each navItems as item}
|
||||
<a
|
||||
href={item.href}
|
||||
class="flex flex-1 flex-col items-center py-2 text-xs transition-colors {currentPath.startsWith(
|
||||
item.href
|
||||
)
|
||||
? 'text-primary font-semibold'
|
||||
: 'text-gray-500'}"
|
||||
>
|
||||
<span class="text-xl">{item.icon}</span>
|
||||
<span class="mt-0.5">{item.label}</span>
|
||||
</a>
|
||||
{/each}
|
||||
</div>
|
||||
</nav>
|
||||
</div>
|
||||
{:else}
|
||||
{@render children()}
|
||||
{/if}
|
||||
|
||||
<style>
|
||||
.safe-bottom {
|
||||
padding-bottom: env(safe-area-inset-bottom);
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,13 @@
|
||||
<script lang="ts">
|
||||
import { goto } from '$app/navigation';
|
||||
import { getToken } from '$lib/api';
|
||||
import { onMount } from 'svelte';
|
||||
|
||||
onMount(() => {
|
||||
if (getToken()) {
|
||||
goto('/lists');
|
||||
} else {
|
||||
goto('/login');
|
||||
}
|
||||
});
|
||||
</script>
|
||||
@@ -0,0 +1,151 @@
|
||||
<script lang="ts">
|
||||
import { onMount, onDestroy } from 'svelte';
|
||||
import { api } from '$lib/api';
|
||||
import { startConnection, stopConnection } from '$lib/signalr';
|
||||
import type { HubConnection } from '@microsoft/signalr';
|
||||
|
||||
interface ListSummary {
|
||||
id: number;
|
||||
name: string;
|
||||
store: { id: number; name: string };
|
||||
itemCount: number;
|
||||
checkedCount: number;
|
||||
updatedAt: string;
|
||||
}
|
||||
|
||||
interface Store {
|
||||
id: number;
|
||||
name: string;
|
||||
}
|
||||
|
||||
let lists = $state<ListSummary[]>([]);
|
||||
let stores = $state<Store[]>([]);
|
||||
let showCreate = $state(false);
|
||||
let newName = $state('');
|
||||
let newStoreId = $state<number | null>(null);
|
||||
let loading = $state(true);
|
||||
let connection: HubConnection | null = null;
|
||||
|
||||
onMount(async () => {
|
||||
[lists, stores] = await Promise.all([
|
||||
api<ListSummary[]>('/api/lists'),
|
||||
api<Store[]>('/api/stores')
|
||||
]);
|
||||
loading = false;
|
||||
|
||||
connection = await startConnection();
|
||||
await connection.invoke('JoinListsOverview');
|
||||
|
||||
connection.on('ListCreated', (data: ListSummary) => {
|
||||
if (!lists.find((l) => l.id === data.id)) {
|
||||
lists = [data, ...lists];
|
||||
}
|
||||
});
|
||||
|
||||
connection.on('ListArchived', (data: { id: number }) => {
|
||||
lists = lists.filter((l) => l.id !== data.id);
|
||||
});
|
||||
|
||||
connection.on('ListSummaryUpdated', (data: ListSummary) => {
|
||||
lists = lists.map((l) =>
|
||||
l.id === data.id ? { ...l, itemCount: data.itemCount, checkedCount: data.checkedCount, updatedAt: data.updatedAt } : l
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
onDestroy(async () => {
|
||||
if (connection) {
|
||||
try { await connection.invoke('LeaveListsOverview'); } catch {}
|
||||
}
|
||||
await stopConnection();
|
||||
});
|
||||
|
||||
async function createList() {
|
||||
if (!newName.trim() || !newStoreId) return;
|
||||
await api<{ id: number }>('/api/lists', {
|
||||
method: 'POST',
|
||||
body: JSON.stringify({ name: newName, storeId: newStoreId })
|
||||
});
|
||||
newName = '';
|
||||
showCreate = false;
|
||||
}
|
||||
</script>
|
||||
|
||||
<div>
|
||||
<div class="mb-4 flex items-center justify-between">
|
||||
<h2 class="text-2xl font-bold">Shopping Lists</h2>
|
||||
<button
|
||||
onclick={() => (showCreate = !showCreate)}
|
||||
class="rounded-full bg-primary px-4 py-2 text-sm font-semibold text-white"
|
||||
>
|
||||
{showCreate ? 'Cancel' : '+ New list'}
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{#if showCreate}
|
||||
<form onsubmit={e => { e.preventDefault(); createList(); }} class="mb-4 rounded-lg border border-gray-200 bg-white p-4 shadow-sm">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={newName}
|
||||
placeholder="List name"
|
||||
required
|
||||
class="mb-3 w-full rounded-lg border border-gray-300 px-3 py-2 focus:border-primary focus:outline-none"
|
||||
/>
|
||||
<select
|
||||
bind:value={newStoreId}
|
||||
required
|
||||
class="mb-3 w-full rounded-lg border border-gray-300 px-3 py-2 focus:border-primary focus:outline-none"
|
||||
>
|
||||
<option value={null} disabled>Select store</option>
|
||||
{#each stores as store}
|
||||
<option value={store.id}>{store.name}</option>
|
||||
{/each}
|
||||
</select>
|
||||
<button
|
||||
type="submit"
|
||||
class="w-full rounded-lg bg-primary py-2 font-semibold text-white"
|
||||
>
|
||||
Create
|
||||
</button>
|
||||
</form>
|
||||
{/if}
|
||||
|
||||
{#if loading}
|
||||
<p class="py-8 text-center text-gray-400">Loading...</p>
|
||||
{:else if lists.length === 0}
|
||||
<div class="py-12 text-center">
|
||||
<p class="text-lg text-gray-400">No shopping lists yet</p>
|
||||
<p class="mt-1 text-sm text-gray-400">Create one to get started</p>
|
||||
</div>
|
||||
{:else}
|
||||
<div class="space-y-3">
|
||||
{#each lists as list (list.id)}
|
||||
<a
|
||||
href="/lists/{list.id}"
|
||||
class="block rounded-lg border border-gray-200 bg-white p-4 shadow-sm transition-shadow active:shadow-md"
|
||||
>
|
||||
<div class="flex items-center justify-between">
|
||||
<div>
|
||||
<h3 class="font-semibold">{list.name}</h3>
|
||||
<p class="mt-0.5 text-sm text-gray-500">{list.store.name}</p>
|
||||
</div>
|
||||
<div class="text-right">
|
||||
<p class="text-sm font-medium">
|
||||
{list.checkedCount}/{list.itemCount}
|
||||
</p>
|
||||
<p class="text-xs text-gray-400">items</p>
|
||||
</div>
|
||||
</div>
|
||||
{#if list.itemCount > 0}
|
||||
<div class="mt-2 h-1.5 overflow-hidden rounded-full bg-gray-100">
|
||||
<div
|
||||
class="h-full rounded-full bg-primary transition-all"
|
||||
style="width: {(list.checkedCount / list.itemCount) * 100}%"
|
||||
></div>
|
||||
</div>
|
||||
{/if}
|
||||
</a>
|
||||
{/each}
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
@@ -0,0 +1,207 @@
|
||||
<script lang="ts">
|
||||
import { onMount, onDestroy } from 'svelte';
|
||||
import { page } from '$app/state';
|
||||
import { goto } from '$app/navigation';
|
||||
import { api } from '$lib/api';
|
||||
import { startConnection, stopConnection } from '$lib/signalr';
|
||||
import type { HubConnection } from '@microsoft/signalr';
|
||||
|
||||
interface ListItem {
|
||||
id: number;
|
||||
name: string;
|
||||
isChecked: boolean;
|
||||
checkedByUserName: string | null;
|
||||
sortOrder: number;
|
||||
recipeTitle: string | null;
|
||||
}
|
||||
|
||||
interface ShoppingListDetail {
|
||||
id: number;
|
||||
name: string;
|
||||
store: { id: number; name: string };
|
||||
isArchived: boolean;
|
||||
items: ListItem[];
|
||||
}
|
||||
|
||||
let list = $state<ShoppingListDetail | null>(null);
|
||||
let items = $state<ListItem[]>([]);
|
||||
let newItemName = $state('');
|
||||
let loading = $state(true);
|
||||
let connection: HubConnection | null = null;
|
||||
|
||||
const listId = $derived(Number(page.params.id));
|
||||
const uncheckedItems = $derived(items.filter((i) => !i.isChecked));
|
||||
const checkedItems = $derived(items.filter((i) => i.isChecked));
|
||||
|
||||
onMount(async () => {
|
||||
const data = await api<ShoppingListDetail>(`/api/lists/${listId}`);
|
||||
list = data;
|
||||
items = data.items;
|
||||
loading = false;
|
||||
|
||||
connection = await startConnection();
|
||||
await connection.invoke('JoinList', listId);
|
||||
|
||||
connection.on('ItemAdded', (data: { id: number; name: string; sortOrder: number; recipeTitle?: string }) => {
|
||||
if (!items.find((i) => i.id === data.id)) {
|
||||
items = [
|
||||
...items,
|
||||
{
|
||||
id: data.id,
|
||||
name: data.name,
|
||||
isChecked: false,
|
||||
checkedByUserName: null,
|
||||
sortOrder: data.sortOrder,
|
||||
recipeTitle: data.recipeTitle ?? null
|
||||
}
|
||||
];
|
||||
}
|
||||
});
|
||||
|
||||
connection.on('ItemChecked', (data: { id: number; isChecked: boolean; checkedByUserName: string | null }) => {
|
||||
items = items.map((i) =>
|
||||
i.id === data.id
|
||||
? { ...i, isChecked: data.isChecked, checkedByUserName: data.checkedByUserName }
|
||||
: i
|
||||
);
|
||||
});
|
||||
|
||||
connection.on('ItemRemoved', (data: { id: number }) => {
|
||||
items = items.filter((i) => i.id !== data.id);
|
||||
});
|
||||
|
||||
connection.on('ListUpdated', (data: { id: number; name: string; storeId: number }) => {
|
||||
if (list) list = { ...list, name: data.name };
|
||||
});
|
||||
});
|
||||
|
||||
onDestroy(async () => {
|
||||
if (connection) {
|
||||
try {
|
||||
await connection.invoke('LeaveList', listId);
|
||||
} catch {}
|
||||
}
|
||||
await stopConnection();
|
||||
});
|
||||
|
||||
async function addItem() {
|
||||
if (!newItemName.trim()) return;
|
||||
const maxSort = items.length > 0 ? Math.max(...items.map((i) => i.sortOrder)) : 0;
|
||||
await api('/api/lists/' + listId + '/items', {
|
||||
method: 'POST',
|
||||
body: JSON.stringify({ name: newItemName, sortOrder: maxSort + 1 })
|
||||
});
|
||||
newItemName = '';
|
||||
}
|
||||
|
||||
async function toggleItem(itemId: number) {
|
||||
await api(`/api/lists/${listId}/items/${itemId}/check`, { method: 'PATCH' });
|
||||
}
|
||||
|
||||
async function removeItem(itemId: number) {
|
||||
await api(`/api/lists/${listId}/items/${itemId}`, { method: 'DELETE' });
|
||||
}
|
||||
|
||||
async function archiveList() {
|
||||
await api(`/api/lists/${listId}`, { method: 'DELETE' });
|
||||
goto('/lists');
|
||||
}
|
||||
</script>
|
||||
|
||||
{#if loading}
|
||||
<p class="py-8 text-center text-gray-400">Loading...</p>
|
||||
{:else if list}
|
||||
<div>
|
||||
<div class="mb-4 flex items-center justify-between">
|
||||
<div>
|
||||
<button onclick={() => goto('/lists')} class="text-sm text-gray-500">← Back</button>
|
||||
<h2 class="text-2xl font-bold">{list.name}</h2>
|
||||
<p class="text-sm text-gray-500">{list.store.name}</p>
|
||||
</div>
|
||||
<button
|
||||
onclick={archiveList}
|
||||
class="rounded-lg border border-gray-300 px-3 py-1.5 text-sm text-gray-500"
|
||||
>
|
||||
Archive
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<form onsubmit={e => { e.preventDefault(); addItem(); }} class="mb-4 flex gap-2">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={newItemName}
|
||||
placeholder="Add an item..."
|
||||
class="flex-1 rounded-lg border border-gray-300 px-3 py-2.5 text-base focus:border-primary focus:outline-none"
|
||||
/>
|
||||
<button
|
||||
type="submit"
|
||||
class="rounded-lg bg-primary px-4 py-2.5 font-semibold text-white"
|
||||
>
|
||||
Add
|
||||
</button>
|
||||
</form>
|
||||
|
||||
{#if uncheckedItems.length > 0}
|
||||
<ul class="space-y-1">
|
||||
{#each uncheckedItems as item (item.id)}
|
||||
<li class="flex items-center gap-3 rounded-lg bg-white px-3 py-3 shadow-sm">
|
||||
<button
|
||||
onclick={() => toggleItem(item.id)}
|
||||
class="flex h-6 w-6 shrink-0 items-center justify-center rounded-full border-2 border-gray-300"
|
||||
aria-label="Check {item.name}"
|
||||
></button>
|
||||
<div class="min-w-0 flex-1">
|
||||
<span class="text-base">{item.name}</span>
|
||||
{#if item.recipeTitle}
|
||||
<span class="ml-1 text-xs text-gray-400">from {item.recipeTitle}</span>
|
||||
{/if}
|
||||
</div>
|
||||
<button
|
||||
onclick={() => removeItem(item.id)}
|
||||
class="shrink-0 p-1 text-gray-300 active:text-danger"
|
||||
aria-label="Remove {item.name}"
|
||||
>
|
||||
✕
|
||||
</button>
|
||||
</li>
|
||||
{/each}
|
||||
</ul>
|
||||
{:else if checkedItems.length === 0}
|
||||
<p class="py-8 text-center text-gray-400">No items yet — add some above</p>
|
||||
{/if}
|
||||
|
||||
{#if checkedItems.length > 0}
|
||||
<div class="mt-6">
|
||||
<h3 class="mb-2 text-sm font-medium text-gray-400">
|
||||
Checked ({checkedItems.length})
|
||||
</h3>
|
||||
<ul class="space-y-1">
|
||||
{#each checkedItems as item (item.id)}
|
||||
<li class="flex items-center gap-3 rounded-lg bg-white/60 px-3 py-3 shadow-sm">
|
||||
<button
|
||||
onclick={() => toggleItem(item.id)}
|
||||
class="flex h-6 w-6 shrink-0 items-center justify-center rounded-full border-2 border-primary bg-primary text-white"
|
||||
aria-label="Uncheck {item.name}"
|
||||
>
|
||||
✓
|
||||
</button>
|
||||
<div class="min-w-0 flex-1">
|
||||
<span class="text-base text-gray-400 line-through">{item.name}</span>
|
||||
{#if item.checkedByUserName}
|
||||
<span class="ml-1 text-xs text-gray-300">{item.checkedByUserName}</span>
|
||||
{/if}
|
||||
</div>
|
||||
<button
|
||||
onclick={() => removeItem(item.id)}
|
||||
class="shrink-0 p-1 text-gray-300 active:text-danger"
|
||||
aria-label="Remove {item.name}"
|
||||
>
|
||||
✕
|
||||
</button>
|
||||
</li>
|
||||
{/each}
|
||||
</ul>
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
{/if}
|
||||
@@ -0,0 +1,104 @@
|
||||
<script lang="ts">
|
||||
import { goto } from '$app/navigation';
|
||||
import { api, setToken } from '$lib/api';
|
||||
|
||||
let mode = $state<'login' | 'register'>('login');
|
||||
let name = $state('');
|
||||
let password = $state('');
|
||||
let familyCode = $state('');
|
||||
let error = $state('');
|
||||
let loading = $state(false);
|
||||
|
||||
async function handleSubmit() {
|
||||
error = '';
|
||||
loading = true;
|
||||
try {
|
||||
const endpoint = mode === 'login' ? '/api/auth/login' : '/api/auth/register';
|
||||
const body =
|
||||
mode === 'login'
|
||||
? { name, password }
|
||||
: { name, password, familyCode };
|
||||
|
||||
const res = await api<{ token: string }>(endpoint, {
|
||||
method: 'POST',
|
||||
body: JSON.stringify(body)
|
||||
});
|
||||
|
||||
setToken(res.token);
|
||||
goto('/lists');
|
||||
} catch (e: any) {
|
||||
error = e.message || 'Something went wrong';
|
||||
} finally {
|
||||
loading = false;
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<div class="flex min-h-dvh items-center justify-center bg-gray-50 px-4">
|
||||
<div class="w-full max-w-sm">
|
||||
<div class="mb-8 text-center">
|
||||
<h1 class="text-4xl font-bold text-primary">YesChef</h1>
|
||||
<p class="mt-2 text-gray-500">Family shopping & recipes</p>
|
||||
</div>
|
||||
|
||||
<form onsubmit={e => { e.preventDefault(); handleSubmit(); }} class="space-y-4">
|
||||
<div>
|
||||
<input
|
||||
type="text"
|
||||
bind:value={name}
|
||||
placeholder="Name"
|
||||
required
|
||||
class="w-full rounded-lg border border-gray-300 px-4 py-3 text-lg focus:border-primary focus:ring-2 focus:ring-primary/20 focus:outline-none"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<input
|
||||
type="password"
|
||||
bind:value={password}
|
||||
placeholder="Password"
|
||||
required
|
||||
class="w-full rounded-lg border border-gray-300 px-4 py-3 text-lg focus:border-primary focus:ring-2 focus:ring-primary/20 focus:outline-none"
|
||||
/>
|
||||
</div>
|
||||
|
||||
{#if mode === 'register'}
|
||||
<div>
|
||||
<input
|
||||
type="text"
|
||||
bind:value={familyCode}
|
||||
placeholder="Family code"
|
||||
required
|
||||
class="w-full rounded-lg border border-gray-300 px-4 py-3 text-lg focus:border-primary focus:ring-2 focus:ring-primary/20 focus:outline-none"
|
||||
/>
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
{#if error}
|
||||
<p class="text-sm text-danger">{error}</p>
|
||||
{/if}
|
||||
|
||||
<button
|
||||
type="submit"
|
||||
disabled={loading}
|
||||
class="w-full rounded-lg bg-primary py-3 text-lg font-semibold text-white transition-colors hover:bg-primary-dark disabled:opacity-50"
|
||||
>
|
||||
{loading ? '...' : mode === 'login' ? 'Sign in' : 'Create account'}
|
||||
</button>
|
||||
</form>
|
||||
|
||||
<p class="mt-6 text-center text-sm text-gray-500">
|
||||
{#if mode === 'login'}
|
||||
New here?
|
||||
<button onclick={() => (mode = 'register')} class="font-medium text-primary">
|
||||
Create account
|
||||
</button>
|
||||
{:else}
|
||||
Already have an account?
|
||||
<button onclick={() => (mode = 'login')} class="font-medium text-primary">
|
||||
Sign in
|
||||
</button>
|
||||
{/if}
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
@@ -0,0 +1,72 @@
|
||||
<script lang="ts">
|
||||
import { onMount } from 'svelte';
|
||||
import { api } from '$lib/api';
|
||||
|
||||
interface RecipeSummary {
|
||||
id: number;
|
||||
title: string;
|
||||
description: string | null;
|
||||
servings: number | null;
|
||||
ingredientCount: number;
|
||||
updatedAt: string;
|
||||
}
|
||||
|
||||
let recipes = $state<RecipeSummary[]>([]);
|
||||
let search = $state('');
|
||||
let loading = $state(true);
|
||||
|
||||
const filtered = $derived(
|
||||
search.trim()
|
||||
? recipes.filter((r) => r.title.toLowerCase().includes(search.toLowerCase()))
|
||||
: recipes
|
||||
);
|
||||
|
||||
onMount(async () => {
|
||||
recipes = await api<RecipeSummary[]>('/api/recipes');
|
||||
loading = false;
|
||||
});
|
||||
</script>
|
||||
|
||||
<div>
|
||||
<div class="mb-4 flex items-center justify-between">
|
||||
<h2 class="text-2xl font-bold">Recipes</h2>
|
||||
<a href="/recipes/new" class="rounded-full bg-primary px-4 py-2 text-sm font-semibold text-white">
|
||||
+ New recipe
|
||||
</a>
|
||||
</div>
|
||||
|
||||
<input
|
||||
type="search"
|
||||
bind:value={search}
|
||||
placeholder="Search recipes..."
|
||||
class="mb-4 w-full rounded-lg border border-gray-300 px-3 py-2.5 focus:border-primary focus:outline-none"
|
||||
/>
|
||||
|
||||
{#if loading}
|
||||
<p class="py-8 text-center text-gray-400">Loading...</p>
|
||||
{:else if filtered.length === 0}
|
||||
<p class="py-12 text-center text-gray-400">
|
||||
{search ? 'No recipes match your search' : 'No recipes yet — add one above'}
|
||||
</p>
|
||||
{:else}
|
||||
<div class="space-y-3">
|
||||
{#each filtered as recipe (recipe.id)}
|
||||
<a
|
||||
href="/recipes/{recipe.id}"
|
||||
class="block rounded-lg border border-gray-200 bg-white p-4 shadow-sm active:shadow-md"
|
||||
>
|
||||
<h3 class="font-semibold">{recipe.title}</h3>
|
||||
{#if recipe.description}
|
||||
<p class="mt-0.5 line-clamp-2 text-sm text-gray-500">{recipe.description}</p>
|
||||
{/if}
|
||||
<div class="mt-2 flex gap-3 text-xs text-gray-400">
|
||||
<span>{recipe.ingredientCount} ingredients</span>
|
||||
{#if recipe.servings}
|
||||
<span>Serves {recipe.servings}</span>
|
||||
{/if}
|
||||
</div>
|
||||
</a>
|
||||
{/each}
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
@@ -0,0 +1,148 @@
|
||||
<script lang="ts">
|
||||
import { onMount } from 'svelte';
|
||||
import { page } from '$app/state';
|
||||
import { goto } from '$app/navigation';
|
||||
import { api } from '$lib/api';
|
||||
|
||||
interface Ingredient {
|
||||
id: number;
|
||||
name: string;
|
||||
quantity: string | null;
|
||||
sortOrder: number;
|
||||
}
|
||||
|
||||
interface Recipe {
|
||||
id: number;
|
||||
title: string;
|
||||
description: string | null;
|
||||
instructions: string | null;
|
||||
servings: number | null;
|
||||
sourceUrl: string | null;
|
||||
createdBy: string;
|
||||
ingredients: Ingredient[];
|
||||
}
|
||||
|
||||
interface ListSummary {
|
||||
id: number;
|
||||
name: string;
|
||||
store: { id: number; name: string };
|
||||
}
|
||||
|
||||
let recipe = $state<Recipe | null>(null);
|
||||
let lists = $state<ListSummary[]>([]);
|
||||
let loading = $state(true);
|
||||
let showAddToList = $state(false);
|
||||
let addingToList = $state<number | null>(null);
|
||||
|
||||
const recipeId = $derived(Number(page.params.id));
|
||||
|
||||
onMount(async () => {
|
||||
[recipe, lists] = await Promise.all([
|
||||
api<Recipe>(`/api/recipes/${recipeId}`),
|
||||
api<ListSummary[]>('/api/lists')
|
||||
]);
|
||||
loading = false;
|
||||
});
|
||||
|
||||
async function addToList(listId: number) {
|
||||
addingToList = listId;
|
||||
try {
|
||||
await api(`/api/lists/${listId}/add-recipe/${recipeId}`, { method: 'POST' });
|
||||
showAddToList = false;
|
||||
goto(`/lists/${listId}`);
|
||||
} finally {
|
||||
addingToList = null;
|
||||
}
|
||||
}
|
||||
|
||||
async function deleteRecipe() {
|
||||
if (!confirm('Delete this recipe?')) return;
|
||||
await api(`/api/recipes/${recipeId}`, { method: 'DELETE' });
|
||||
goto('/recipes');
|
||||
}
|
||||
</script>
|
||||
|
||||
{#if loading}
|
||||
<p class="py-8 text-center text-gray-400">Loading...</p>
|
||||
{:else if recipe}
|
||||
<div>
|
||||
<button onclick={() => goto('/recipes')} class="mb-2 text-sm text-gray-500">← Back</button>
|
||||
|
||||
<div class="mb-4 flex items-start justify-between">
|
||||
<div>
|
||||
<h2 class="text-2xl font-bold">{recipe.title}</h2>
|
||||
{#if recipe.description}
|
||||
<p class="mt-1 text-gray-500">{recipe.description}</p>
|
||||
{/if}
|
||||
<div class="mt-2 flex gap-3 text-sm text-gray-400">
|
||||
{#if recipe.servings}
|
||||
<span>Serves {recipe.servings}</span>
|
||||
{/if}
|
||||
<span>By {recipe.createdBy}</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="mb-4 flex gap-2">
|
||||
<button
|
||||
onclick={() => (showAddToList = !showAddToList)}
|
||||
class="flex-1 rounded-lg bg-primary py-2.5 font-semibold text-white"
|
||||
>
|
||||
Add to list
|
||||
</button>
|
||||
<button
|
||||
onclick={deleteRecipe}
|
||||
class="rounded-lg border border-danger px-4 py-2.5 text-sm text-danger"
|
||||
>
|
||||
Delete
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{#if showAddToList}
|
||||
<div class="mb-4 rounded-lg border border-gray-200 bg-white p-3 shadow-sm">
|
||||
<p class="mb-2 text-sm font-medium text-gray-600">Choose a list:</p>
|
||||
{#if lists.length === 0}
|
||||
<p class="text-sm text-gray-400">No active lists. Create one first.</p>
|
||||
{:else}
|
||||
<div class="space-y-1">
|
||||
{#each lists as list}
|
||||
<button
|
||||
onclick={() => addToList(list.id)}
|
||||
disabled={addingToList === list.id}
|
||||
class="w-full rounded-lg px-3 py-2 text-left transition-colors hover:bg-gray-50 active:bg-gray-100"
|
||||
>
|
||||
<span class="font-medium">{list.name}</span>
|
||||
<span class="ml-1 text-sm text-gray-400">{list.store.name}</span>
|
||||
</button>
|
||||
{/each}
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
{#if recipe.ingredients.length > 0}
|
||||
<div class="mb-6">
|
||||
<h3 class="mb-2 text-lg font-semibold">Ingredients</h3>
|
||||
<ul class="space-y-1.5">
|
||||
{#each recipe.ingredients as ingredient}
|
||||
<li class="flex gap-2 rounded-lg bg-white px-3 py-2 shadow-sm">
|
||||
{#if ingredient.quantity}
|
||||
<span class="font-medium text-primary">{ingredient.quantity}</span>
|
||||
{/if}
|
||||
<span>{ingredient.name}</span>
|
||||
</li>
|
||||
{/each}
|
||||
</ul>
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
{#if recipe.instructions}
|
||||
<div>
|
||||
<h3 class="mb-2 text-lg font-semibold">Instructions</h3>
|
||||
<div class="whitespace-pre-wrap rounded-lg bg-white p-4 text-gray-700 shadow-sm">
|
||||
{recipe.instructions}
|
||||
</div>
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
{/if}
|
||||
@@ -0,0 +1,128 @@
|
||||
<script lang="ts">
|
||||
import { goto } from '$app/navigation';
|
||||
import { api } from '$lib/api';
|
||||
|
||||
let title = $state('');
|
||||
let description = $state('');
|
||||
let instructions = $state('');
|
||||
let servings = $state<number | undefined>();
|
||||
let ingredients = $state<{ name: string; quantity: string }[]>([{ name: '', quantity: '' }]);
|
||||
let saving = $state(false);
|
||||
|
||||
function addIngredient() {
|
||||
ingredients = [...ingredients, { name: '', quantity: '' }];
|
||||
}
|
||||
|
||||
function removeIngredient(idx: number) {
|
||||
ingredients = ingredients.filter((_, i) => i !== idx);
|
||||
}
|
||||
|
||||
async function save() {
|
||||
if (!title.trim()) return;
|
||||
saving = true;
|
||||
try {
|
||||
const res = await api<{ id: number }>('/api/recipes', {
|
||||
method: 'POST',
|
||||
body: JSON.stringify({
|
||||
title,
|
||||
description: description || null,
|
||||
instructions: instructions || null,
|
||||
servings: servings || null,
|
||||
sourceUrl: null,
|
||||
ingredients: ingredients
|
||||
.filter((i) => i.name.trim())
|
||||
.map((i, idx) => ({ name: i.name, quantity: i.quantity || null, sortOrder: idx }))
|
||||
})
|
||||
});
|
||||
goto(`/recipes/${res.id}`);
|
||||
} finally {
|
||||
saving = false;
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<div>
|
||||
<button onclick={() => goto('/recipes')} class="mb-2 text-sm text-gray-500">← Back</button>
|
||||
<h2 class="mb-4 text-2xl font-bold">New Recipe</h2>
|
||||
|
||||
<form onsubmit={e => { e.preventDefault(); save(); }} class="space-y-4">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={title}
|
||||
placeholder="Recipe title"
|
||||
required
|
||||
class="w-full rounded-lg border border-gray-300 px-3 py-2.5 text-lg focus:border-primary focus:outline-none"
|
||||
/>
|
||||
|
||||
<textarea
|
||||
bind:value={description}
|
||||
placeholder="Short description (optional)"
|
||||
rows={2}
|
||||
class="w-full rounded-lg border border-gray-300 px-3 py-2.5 focus:border-primary focus:outline-none"
|
||||
></textarea>
|
||||
|
||||
<div>
|
||||
<label class="mb-1 block text-sm font-medium text-gray-600">
|
||||
Servings
|
||||
<input
|
||||
type="number"
|
||||
bind:value={servings}
|
||||
min={1}
|
||||
placeholder="e.g. 4"
|
||||
class="mt-1 block w-24 rounded-lg border border-gray-300 px-3 py-2 focus:border-primary focus:outline-none"
|
||||
/>
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<span class="mb-2 block text-sm font-medium text-gray-600">Ingredients</span>
|
||||
{#each ingredients as ingredient, idx}
|
||||
<div class="mb-2 flex gap-2">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={ingredient.quantity}
|
||||
placeholder="Qty"
|
||||
class="w-20 rounded-lg border border-gray-300 px-2 py-2 text-sm focus:border-primary focus:outline-none"
|
||||
/>
|
||||
<input
|
||||
type="text"
|
||||
bind:value={ingredient.name}
|
||||
placeholder="Ingredient name"
|
||||
class="flex-1 rounded-lg border border-gray-300 px-3 py-2 text-sm focus:border-primary focus:outline-none"
|
||||
/>
|
||||
{#if ingredients.length > 1}
|
||||
<button
|
||||
type="button"
|
||||
onclick={() => removeIngredient(idx)}
|
||||
class="px-2 text-gray-300 active:text-danger"
|
||||
>
|
||||
✕
|
||||
</button>
|
||||
{/if}
|
||||
</div>
|
||||
{/each}
|
||||
<button
|
||||
type="button"
|
||||
onclick={addIngredient}
|
||||
class="mt-1 text-sm font-medium text-primary"
|
||||
>
|
||||
+ Add ingredient
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<textarea
|
||||
bind:value={instructions}
|
||||
placeholder="Instructions (optional)"
|
||||
rows={6}
|
||||
class="w-full rounded-lg border border-gray-300 px-3 py-2.5 focus:border-primary focus:outline-none"
|
||||
></textarea>
|
||||
|
||||
<button
|
||||
type="submit"
|
||||
disabled={saving}
|
||||
class="w-full rounded-lg bg-primary py-3 font-semibold text-white disabled:opacity-50"
|
||||
>
|
||||
{saving ? 'Saving...' : 'Save Recipe'}
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
@@ -0,0 +1,101 @@
|
||||
<script lang="ts">
|
||||
import { onMount } from 'svelte';
|
||||
import { api } from '$lib/api';
|
||||
|
||||
interface Store {
|
||||
id: number;
|
||||
name: string;
|
||||
sortOrder: number;
|
||||
}
|
||||
|
||||
let stores = $state<Store[]>([]);
|
||||
let newName = $state('');
|
||||
let editingId = $state<number | null>(null);
|
||||
let editName = $state('');
|
||||
let loading = $state(true);
|
||||
|
||||
onMount(async () => {
|
||||
stores = await api<Store[]>('/api/stores');
|
||||
loading = false;
|
||||
});
|
||||
|
||||
async function addStore() {
|
||||
if (!newName.trim()) return;
|
||||
await api('/api/stores', {
|
||||
method: 'POST',
|
||||
body: JSON.stringify({ name: newName, sortOrder: stores.length })
|
||||
});
|
||||
newName = '';
|
||||
stores = await api<Store[]>('/api/stores');
|
||||
}
|
||||
|
||||
function startEdit(store: Store) {
|
||||
editingId = store.id;
|
||||
editName = store.name;
|
||||
}
|
||||
|
||||
async function saveEdit() {
|
||||
if (!editName.trim() || !editingId) return;
|
||||
const store = stores.find((s) => s.id === editingId)!;
|
||||
await api(`/api/stores/${editingId}`, {
|
||||
method: 'PUT',
|
||||
body: JSON.stringify({ name: editName, sortOrder: store.sortOrder })
|
||||
});
|
||||
editingId = null;
|
||||
stores = await api<Store[]>('/api/stores');
|
||||
}
|
||||
|
||||
async function deleteStore(id: number) {
|
||||
try {
|
||||
await api(`/api/stores/${id}`, { method: 'DELETE' });
|
||||
stores = stores.filter((s) => s.id !== id);
|
||||
} catch (e: any) {
|
||||
alert(e.message);
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<div>
|
||||
<h2 class="mb-4 text-2xl font-bold">Stores</h2>
|
||||
|
||||
<form onsubmit={e => { e.preventDefault(); addStore(); }} class="mb-6 flex gap-2">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={newName}
|
||||
placeholder="New store name"
|
||||
required
|
||||
class="flex-1 rounded-lg border border-gray-300 px-3 py-2.5 focus:border-primary focus:outline-none"
|
||||
/>
|
||||
<button type="submit" class="rounded-lg bg-primary px-4 py-2.5 font-semibold text-white">
|
||||
Add
|
||||
</button>
|
||||
</form>
|
||||
|
||||
{#if loading}
|
||||
<p class="py-8 text-center text-gray-400">Loading...</p>
|
||||
{:else if stores.length === 0}
|
||||
<p class="py-12 text-center text-gray-400">No stores yet — add one above</p>
|
||||
{:else}
|
||||
<ul class="space-y-2">
|
||||
{#each stores as store (store.id)}
|
||||
<li class="flex items-center gap-3 rounded-lg bg-white px-4 py-3 shadow-sm">
|
||||
{#if editingId === store.id}
|
||||
<form onsubmit={e => { e.preventDefault(); saveEdit(); }} class="flex flex-1 gap-2">
|
||||
<input
|
||||
type="text"
|
||||
bind:value={editName}
|
||||
class="flex-1 rounded border border-gray-300 px-2 py-1 focus:border-primary focus:outline-none"
|
||||
/>
|
||||
<button type="submit" class="text-sm font-medium text-primary">Save</button>
|
||||
<button type="button" onclick={() => (editingId = null)} class="text-sm text-gray-400">Cancel</button>
|
||||
</form>
|
||||
{:else}
|
||||
<span class="flex-1 font-medium">{store.name}</span>
|
||||
<button onclick={() => startEdit(store)} class="text-sm text-gray-400">Edit</button>
|
||||
<button onclick={() => deleteStore(store.id)} class="text-sm text-danger">Delete</button>
|
||||
{/if}
|
||||
</li>
|
||||
{/each}
|
||||
</ul>
|
||||
{/if}
|
||||
</div>
|
||||
@@ -0,0 +1,45 @@
|
||||
/// <reference types="@sveltejs/kit" />
|
||||
/// <reference no-default-lib="true"/>
|
||||
/// <reference lib="esnext" />
|
||||
/// <reference lib="webworker" />
|
||||
|
||||
import { build, files, version } from '$service-worker';
|
||||
|
||||
const sw = self as unknown as ServiceWorkerGlobalScope;
|
||||
const CACHE = `cache-${version}`;
|
||||
const ASSETS = [...build, ...files];
|
||||
|
||||
sw.addEventListener('install', (event) => {
|
||||
event.waitUntil(
|
||||
caches
|
||||
.open(CACHE)
|
||||
.then((cache) => cache.addAll(ASSETS))
|
||||
.then(() => sw.skipWaiting())
|
||||
);
|
||||
});
|
||||
|
||||
sw.addEventListener('activate', (event) => {
|
||||
event.waitUntil(
|
||||
caches.keys().then(async (keys) => {
|
||||
for (const key of keys) {
|
||||
if (key !== CACHE) await caches.delete(key);
|
||||
}
|
||||
await sw.clients.claim();
|
||||
})
|
||||
);
|
||||
});
|
||||
|
||||
sw.addEventListener('fetch', (event) => {
|
||||
if (event.request.method !== 'GET') return;
|
||||
|
||||
const url = new URL(event.request.url);
|
||||
|
||||
// Don't cache API or SignalR requests
|
||||
if (url.pathname.startsWith('/api/') || url.pathname.startsWith('/hubs/')) return;
|
||||
|
||||
event.respondWith(
|
||||
caches.match(event.request).then((cached) => {
|
||||
return cached || fetch(event.request);
|
||||
})
|
||||
);
|
||||
});
|
||||
@@ -0,0 +1,4 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 64 64">
|
||||
<rect width="64" height="64" rx="12" fill="#16a34a"/>
|
||||
<text x="32" y="44" font-size="36" text-anchor="middle" fill="white" font-family="sans-serif" font-weight="bold">Y</text>
|
||||
</svg>
|
||||
|
After Width: | Height: | Size: 248 B |
Binary file not shown.
|
After Width: | Height: | Size: 2.5 KiB |
Binary file not shown.
|
After Width: | Height: | Size: 7.1 KiB |
@@ -0,0 +1,12 @@
|
||||
{
|
||||
"name": "YesChef",
|
||||
"short_name": "YesChef",
|
||||
"start_url": "/lists",
|
||||
"display": "standalone",
|
||||
"background_color": "#ffffff",
|
||||
"theme_color": "#16a34a",
|
||||
"icons": [
|
||||
{ "src": "/icons/icon-192.png", "sizes": "192x192", "type": "image/png" },
|
||||
{ "src": "/icons/icon-512.png", "sizes": "512x512", "type": "image/png" }
|
||||
]
|
||||
}
|
||||
@@ -0,0 +1,3 @@
|
||||
# allow crawling everything by default
|
||||
User-agent: *
|
||||
Disallow:
|
||||
@@ -0,0 +1,16 @@
|
||||
import adapter from '@sveltejs/adapter-node';
|
||||
|
||||
/** @type {import('@sveltejs/kit').Config} */
|
||||
const config = {
|
||||
compilerOptions: {
|
||||
runes: ({ filename }) => (filename.split(/[/\\]/).includes('node_modules') ? undefined : true)
|
||||
},
|
||||
kit: {
|
||||
adapter: adapter({ out: 'build' }),
|
||||
alias: {
|
||||
$lib: 'src/lib'
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
export default config;
|
||||
@@ -0,0 +1,20 @@
|
||||
{
|
||||
"extends": "./.svelte-kit/tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"rewriteRelativeImportExtensions": true,
|
||||
"allowJs": true,
|
||||
"checkJs": true,
|
||||
"esModuleInterop": true,
|
||||
"forceConsistentCasingInFileNames": true,
|
||||
"resolveJsonModule": true,
|
||||
"skipLibCheck": true,
|
||||
"sourceMap": true,
|
||||
"strict": true,
|
||||
"moduleResolution": "bundler"
|
||||
}
|
||||
// Path aliases are handled by https://svelte.dev/docs/kit/configuration#alias
|
||||
// except $lib which is handled by https://svelte.dev/docs/kit/configuration#files
|
||||
//
|
||||
// To make changes to top-level options such as include and exclude, we recommend extending
|
||||
// the generated config; see https://svelte.dev/docs/kit/configuration#typescript
|
||||
}
|
||||
@@ -0,0 +1,16 @@
|
||||
import { sveltekit } from '@sveltejs/kit/vite';
|
||||
import tailwindcss from '@tailwindcss/vite';
|
||||
import { defineConfig } from 'vite';
|
||||
|
||||
export default defineConfig({
|
||||
plugins: [tailwindcss(), sveltekit()],
|
||||
server: {
|
||||
proxy: {
|
||||
'/api': 'http://localhost:5291',
|
||||
'/hubs': {
|
||||
target: 'http://localhost:5291',
|
||||
ws: true
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
@@ -0,0 +1,147 @@
|
||||
import { chromium } from 'playwright';
|
||||
|
||||
const browser = await chromium.launch({ channel: 'chrome', headless: false, slowMo: 200 });
|
||||
|
||||
const contextA = await browser.newContext({ viewport: { width: 390, height: 844 } });
|
||||
const contextB = await browser.newContext({ viewport: { width: 390, height: 844 } });
|
||||
const pageA = await contextA.newPage();
|
||||
const pageB = await contextB.newPage();
|
||||
|
||||
const BASE = 'http://localhost:5173';
|
||||
|
||||
console.log('⏳ Arrange the two browser windows side by side. Starting in 8 seconds...');
|
||||
await new Promise(r => setTimeout(r, 8000));
|
||||
|
||||
async function step(name, fn) {
|
||||
process.stdout.write(`▶ ${name}... `);
|
||||
try {
|
||||
await fn();
|
||||
console.log('✅');
|
||||
} catch (e) {
|
||||
console.log(`❌ ${e.message}`);
|
||||
throw e;
|
||||
}
|
||||
}
|
||||
|
||||
async function register(page, name) {
|
||||
await page.goto(BASE);
|
||||
await page.waitForURL('**/login', { timeout: 5000 });
|
||||
await page.click('text=Create account');
|
||||
await page.fill('input[placeholder="Name"]', name);
|
||||
await page.fill('input[placeholder="Password"]', 'password123');
|
||||
await page.fill('input[placeholder="Family code"]', 'dev-family-code');
|
||||
await page.click('button[type="submit"]');
|
||||
await page.waitForURL('**/lists', { timeout: 5000 });
|
||||
}
|
||||
|
||||
try {
|
||||
await step('Register User A (Mom)', async () => {
|
||||
await register(pageA, 'Mom');
|
||||
});
|
||||
|
||||
await step('Register User B (Dad)', async () => {
|
||||
await register(pageB, 'Dad');
|
||||
});
|
||||
|
||||
// Mom creates a store
|
||||
await step('Mom creates store "Kroger"', async () => {
|
||||
await pageA.click('a:has-text("Stores")');
|
||||
await pageA.waitForURL('**/stores');
|
||||
await pageA.fill('input[placeholder="New store name"]', 'Kroger');
|
||||
await pageA.click('button:has-text("Add")');
|
||||
await pageA.waitForSelector('text=Kroger');
|
||||
});
|
||||
|
||||
// Both on lists overview — Mom creates a list, Dad should see it appear via SignalR
|
||||
await step('Mom navigates to Lists', async () => {
|
||||
await pageA.click('a:has-text("Lists")');
|
||||
await pageA.waitForURL('**/lists');
|
||||
});
|
||||
|
||||
await step('Mom creates list → Dad sees it appear on overview', async () => {
|
||||
await pageA.click('text=+ New list');
|
||||
await pageA.fill('input[placeholder="List name"]', 'Weekly Groceries');
|
||||
await pageA.selectOption('select', { label: 'Kroger' });
|
||||
await pageA.click('button:has-text("Create")');
|
||||
await pageA.waitForSelector('text=Weekly Groceries');
|
||||
// Dad should see it appear without navigating away
|
||||
await pageB.waitForSelector('text=Weekly Groceries', { timeout: 5000 });
|
||||
});
|
||||
|
||||
// Both open the list
|
||||
await step('Mom opens the list', async () => {
|
||||
await pageA.click('text=Weekly Groceries');
|
||||
await pageA.waitForURL(/\/lists\/\d+/);
|
||||
});
|
||||
|
||||
await step('Dad opens the same list', async () => {
|
||||
await pageB.click('text=Weekly Groceries');
|
||||
await pageB.waitForURL(/\/lists\/\d+/);
|
||||
});
|
||||
|
||||
// Mom adds items — Dad should see them appear in real-time
|
||||
await step('Mom adds "Milk" → Dad sees it appear', async () => {
|
||||
await pageA.fill('input[placeholder="Add an item..."]', 'Milk');
|
||||
await pageA.click('button:has-text("Add")');
|
||||
await pageA.waitForSelector('text=Milk');
|
||||
await pageB.waitForSelector('text=Milk', { timeout: 5000 });
|
||||
});
|
||||
|
||||
await step('Mom adds "Eggs" → Dad sees it appear', async () => {
|
||||
await pageA.fill('input[placeholder="Add an item..."]', 'Eggs');
|
||||
await pageA.click('button:has-text("Add")');
|
||||
await pageA.waitForSelector('text=Eggs');
|
||||
await pageB.waitForSelector('text=Eggs', { timeout: 5000 });
|
||||
});
|
||||
|
||||
await step('Mom adds "Bread" → Dad sees it appear', async () => {
|
||||
await pageA.fill('input[placeholder="Add an item..."]', 'Bread');
|
||||
await pageA.click('button:has-text("Add")');
|
||||
await pageA.waitForSelector('text=Bread');
|
||||
await pageB.waitForSelector('text=Bread', { timeout: 5000 });
|
||||
});
|
||||
|
||||
// Dad checks off Milk — Mom should see it checked
|
||||
await step('Dad checks "Milk" → Mom sees it checked', async () => {
|
||||
await pageB.click('button[aria-label="Check Milk"]');
|
||||
await pageB.waitForSelector('text=Checked (1)');
|
||||
await pageA.waitForSelector('text=Checked (1)', { timeout: 5000 });
|
||||
});
|
||||
|
||||
// Mom checks off Eggs — Dad should see it
|
||||
await step('Mom checks "Eggs" → Dad sees it checked', async () => {
|
||||
await pageA.click('button[aria-label="Check Eggs"]');
|
||||
await pageA.waitForSelector('text=Checked (2)');
|
||||
await pageB.waitForSelector('text=Checked (2)', { timeout: 5000 });
|
||||
});
|
||||
|
||||
// Dad unchecks Milk — Mom should see it unchecked
|
||||
await step('Dad unchecks "Milk" → Mom sees it unchecked', async () => {
|
||||
await pageB.click('button[aria-label="Uncheck Milk"]');
|
||||
await pageB.waitForSelector('text=Checked (1)');
|
||||
await pageA.waitForSelector('text=Checked (1)', { timeout: 5000 });
|
||||
});
|
||||
|
||||
// Dad adds an item — Mom should see it
|
||||
await step('Dad adds "Butter" → Mom sees it appear', async () => {
|
||||
await pageB.fill('input[placeholder="Add an item..."]', 'Butter');
|
||||
await pageB.click('button:has-text("Add")');
|
||||
await pageB.waitForSelector('text=Butter');
|
||||
await pageA.waitForSelector('text=Butter', { timeout: 5000 });
|
||||
});
|
||||
|
||||
// Mom removes Bread — Dad should see it disappear
|
||||
await step('Mom removes "Bread" → Dad sees it disappear', async () => {
|
||||
await pageA.click('button[aria-label="Remove Bread"]');
|
||||
await pageA.waitForFunction(() => !document.body.textContent.includes('Bread'), { timeout: 3000 });
|
||||
await pageB.waitForFunction(() => !document.body.textContent.includes('Bread'), { timeout: 5000 });
|
||||
});
|
||||
|
||||
console.log('\n🎉 All multi-user tests passed! Real-time sync works.');
|
||||
} catch (e) {
|
||||
console.error(`\n💥 Test failed: ${e.message}`);
|
||||
process.exitCode = 1;
|
||||
} finally {
|
||||
await new Promise(r => setTimeout(r, 2000));
|
||||
await browser.close();
|
||||
}
|
||||
+147
@@ -0,0 +1,147 @@
|
||||
import { chromium } from 'playwright';
|
||||
|
||||
const browser = await chromium.launch({ channel: 'chrome', headless: false, slowMo: 300 });
|
||||
const context = await browser.newContext({ viewport: { width: 390, height: 844 } });
|
||||
const page = await context.newPage();
|
||||
|
||||
const BASE = 'http://localhost:5173';
|
||||
let failed = false;
|
||||
|
||||
async function step(name, fn) {
|
||||
process.stdout.write(`▶ ${name}... `);
|
||||
try {
|
||||
await fn();
|
||||
console.log('✅');
|
||||
} catch (e) {
|
||||
console.log(`❌ ${e.message}`);
|
||||
failed = true;
|
||||
throw e;
|
||||
}
|
||||
}
|
||||
|
||||
try {
|
||||
await step('Navigate to app → should redirect to /login', async () => {
|
||||
await page.goto(BASE);
|
||||
await page.waitForURL('**/login', { timeout: 5000 });
|
||||
});
|
||||
|
||||
await step('Register a new account', async () => {
|
||||
await page.click('text=Create account');
|
||||
await page.fill('input[placeholder="Name"]', 'TestChef');
|
||||
await page.fill('input[placeholder="Password"]', 'password123');
|
||||
await page.fill('input[placeholder="Family code"]', 'dev-family-code');
|
||||
await page.click('button[type="submit"]');
|
||||
await page.waitForURL('**/lists', { timeout: 5000 });
|
||||
});
|
||||
|
||||
await step('Create store "Kroger"', async () => {
|
||||
await page.click('a:has-text("Stores")');
|
||||
await page.waitForURL('**/stores');
|
||||
await page.fill('input[placeholder="New store name"]', 'Kroger');
|
||||
await page.click('button:has-text("Add")');
|
||||
await page.waitForSelector('text=Kroger');
|
||||
});
|
||||
|
||||
await step('Create store "Costco"', async () => {
|
||||
await page.fill('input[placeholder="New store name"]', 'Costco');
|
||||
await page.click('button:has-text("Add")');
|
||||
await page.waitForSelector('text=Costco');
|
||||
});
|
||||
|
||||
await step('Create shopping list "Weekly Groceries"', async () => {
|
||||
await page.click('a:has-text("Lists")');
|
||||
await page.waitForURL('**/lists');
|
||||
await page.click('text=+ New list');
|
||||
await page.fill('input[placeholder="List name"]', 'Weekly Groceries');
|
||||
await page.selectOption('select', { label: 'Kroger' });
|
||||
await page.click('button:has-text("Create")');
|
||||
await page.waitForSelector('text=Weekly Groceries');
|
||||
});
|
||||
|
||||
await step('Open list and add 5 items', async () => {
|
||||
await page.click('text=Weekly Groceries');
|
||||
await page.waitForURL(/\/lists\/\d+/);
|
||||
|
||||
for (const item of ['Milk', 'Eggs', 'Bread', 'Chicken breast', 'Broccoli']) {
|
||||
await page.fill('input[placeholder="Add an item..."]', item);
|
||||
await page.click('button:has-text("Add")');
|
||||
await page.waitForSelector(`text=${item}`);
|
||||
}
|
||||
});
|
||||
|
||||
await step('Check off Milk and Eggs', async () => {
|
||||
await page.click('button[aria-label="Check Milk"]');
|
||||
await page.waitForSelector('text=Checked (1)');
|
||||
await page.click('button[aria-label="Check Eggs"]');
|
||||
await page.waitForSelector('text=Checked (2)');
|
||||
});
|
||||
|
||||
await step('Uncheck Milk', async () => {
|
||||
await page.click('button[aria-label="Uncheck Milk"]');
|
||||
await page.waitForSelector('text=Checked (1)');
|
||||
});
|
||||
|
||||
await step('Remove Broccoli', async () => {
|
||||
await page.click('button[aria-label="Remove Broccoli"]');
|
||||
await page.waitForFunction(() => !document.body.textContent.includes('Broccoli'), { timeout: 3000 });
|
||||
});
|
||||
|
||||
await step('Create recipe "Chicken Stir Fry"', async () => {
|
||||
await page.click('a:has-text("Recipes")');
|
||||
await page.waitForURL('**/recipes');
|
||||
await page.click('text=+ New recipe');
|
||||
await page.waitForURL('**/recipes/new');
|
||||
await page.fill('input[placeholder="Recipe title"]', 'Chicken Stir Fry');
|
||||
await page.fill('textarea[placeholder*="description"]', 'Quick weeknight dinner');
|
||||
|
||||
const qtyInputs = page.locator('input[placeholder="Qty"]');
|
||||
const nameInputs = page.locator('input[placeholder="Ingredient name"]');
|
||||
await qtyInputs.nth(0).fill('2 lbs');
|
||||
await nameInputs.nth(0).fill('chicken breast');
|
||||
|
||||
await page.click('text=+ Add ingredient');
|
||||
await qtyInputs.nth(1).fill('1 head');
|
||||
await nameInputs.nth(1).fill('broccoli');
|
||||
|
||||
await page.click('text=+ Add ingredient');
|
||||
await qtyInputs.nth(2).fill('3 tbsp');
|
||||
await nameInputs.nth(2).fill('soy sauce');
|
||||
|
||||
await page.fill('textarea[placeholder*="Instructions"]', '1. Cut chicken\n2. Stir fry\n3. Add sauce');
|
||||
await page.click('button:has-text("Save Recipe")');
|
||||
await page.waitForURL(/\/recipes\/\d+/);
|
||||
});
|
||||
|
||||
await step('Add recipe ingredients to shopping list', async () => {
|
||||
await page.click('button:has-text("Add to list")');
|
||||
await page.waitForSelector('text=Choose a list');
|
||||
await page.click('text=Weekly Groceries');
|
||||
await page.waitForURL(/\/lists\/\d+/);
|
||||
await page.waitForSelector('text=chicken breast');
|
||||
});
|
||||
|
||||
await step('Verify list has recipe items', async () => {
|
||||
const content = await page.textContent('body');
|
||||
for (const item of ['2 lbs chicken breast', '1 head broccoli', '3 tbsp soy sauce']) {
|
||||
if (!content.includes(item)) throw new Error(`Missing recipe item: ${item}`);
|
||||
}
|
||||
});
|
||||
|
||||
await step('Navigate back to lists overview', async () => {
|
||||
await page.click('text=← Back');
|
||||
await page.waitForURL('**/lists');
|
||||
});
|
||||
|
||||
await step('Sign out', async () => {
|
||||
await page.click('text=Sign out');
|
||||
await page.waitForURL('**/login');
|
||||
});
|
||||
|
||||
console.log('\n🎉 All tests passed!');
|
||||
} catch (e) {
|
||||
console.error(`\n💥 Test suite stopped at failure: ${e.message}`);
|
||||
process.exitCode = 1;
|
||||
} finally {
|
||||
await new Promise(r => setTimeout(r, 2000));
|
||||
await browser.close();
|
||||
}
|
||||
Reference in New Issue
Block a user