mirror of
https://github.com/aaronpo97/the-biergarten-app.git
synced 2026-06-01 10:04:00 +00:00
Compare commits
1 Commits
main-2.0
...
23e2199b6b
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
23e2199b6b |
13
.config/dotnet-tools.json
Normal file
13
.config/dotnet-tools.json
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
{
|
||||||
|
"version": 1,
|
||||||
|
"isRoot": true,
|
||||||
|
"tools": {
|
||||||
|
"csharpier": {
|
||||||
|
"version": "1.2.1",
|
||||||
|
"commands": [
|
||||||
|
"csharpier"
|
||||||
|
],
|
||||||
|
"rollForward": false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
{
|
{
|
||||||
|
"$schema": "https://json.schemastore.org/csharpier.json",
|
||||||
|
|
||||||
"printWidth": 80,
|
"printWidth": 80,
|
||||||
"useTabs": false,
|
"useTabs": false,
|
||||||
"indentSize": 4,
|
"indentSize": 4,
|
||||||
1
.gitattributes
vendored
1
.gitattributes
vendored
@@ -1 +0,0 @@
|
|||||||
archive/** linguist-vendored
|
|
||||||
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
@@ -7,7 +7,6 @@ assignees: []
|
|||||||
---
|
---
|
||||||
|
|
||||||
## User Story
|
## User Story
|
||||||
|
|
||||||
**As a** (who wants to accomplish something)
|
**As a** (who wants to accomplish something)
|
||||||
**I want to** (what they want to accomplish)
|
**I want to** (what they want to accomplish)
|
||||||
**So that** (why they want to accomplish that thing)
|
**So that** (why they want to accomplish that thing)
|
||||||
@@ -16,18 +15,29 @@ assignees: []
|
|||||||
|
|
||||||
### Scenario 1
|
### Scenario 1
|
||||||
|
|
||||||
Given ... When ... Then ...
|
|
||||||
|
Given ...
|
||||||
|
When ...
|
||||||
|
Then ...
|
||||||
|
|
||||||
|
|
||||||
### Scenario 2
|
### Scenario 2
|
||||||
|
|
||||||
Given ... When ... Then ...
|
|
||||||
|
Given ...
|
||||||
|
When ...
|
||||||
|
Then ...
|
||||||
|
|
||||||
|
|
||||||
### Scenario 3
|
### Scenario 3
|
||||||
|
|
||||||
Given ... When ... Then ...
|
|
||||||
|
Given ...
|
||||||
|
When ...
|
||||||
|
Then ...
|
||||||
|
|
||||||
|
|
||||||
## Subtasks
|
## Subtasks
|
||||||
|
|
||||||
- [ ] Task 1
|
- [ ] Task 1
|
||||||
- [ ] Task 2
|
- [ ] Task 2
|
||||||
- [ ] Task 3
|
- [ ] Task 3
|
||||||
962
LICENSE.md
962
LICENSE.md
File diff suppressed because it is too large
Load Diff
161
README.md
161
README.md
@@ -1,56 +1,40 @@
|
|||||||
# The Biergarten App
|
# The Biergarten App
|
||||||
|
|
||||||
The Biergarten App is a full-stack directory and discovery platform for
|
The Biergarten App is a multi-project monorepo with a .NET backend and an active React
|
||||||
breweries. It features a robust user authentication system, a searchable
|
Router frontend in `src/Website`. The current website focuses on account flows, theme
|
||||||
database of brewery locations, and a custom offline data-generation pipeline
|
switching, shared UI components, Storybook coverage, and integration with the API.
|
||||||
that uses LLMs (Llama.cpp) and Wikipedia to synthesize realistic seed data.
|
|
||||||
|
|
||||||
It features:
|
## Documentation
|
||||||
|
|
||||||
- A .NET backend (Web API + database migrations/seed) under `web/backend/`
|
- [Getting Started](docs/getting-started.md) - Local setup for backend and active website
|
||||||
- A server-rendered React website (React Router + Vite) under `web/frontend/`
|
- [Architecture](docs/architecture.md) - Current backend and frontend architecture
|
||||||
- A C++20 “pipeline” CLI for generating seed data under `tooling/pipeline/`
|
- [Docker Guide](docs/docker.md) - Container-based backend development and testing
|
||||||
|
- [Testing](docs/testing.md) - Backend and frontend test commands
|
||||||
Specialized documentation (setup, architecture, docker, testing, diagrams, and
|
- [Environment Variables](docs/environment-variables.md) - Active configuration reference
|
||||||
pipeline notes) lives under `docs/`.
|
- [Token Validation](docs/token-validation.md) - JWT validation architecture
|
||||||
|
- [Legacy Website Archive](docs/archive/legacy-website-v1.md) - Archived notes for the old Next.js frontend
|
||||||
## Documentation (Start Here)
|
|
||||||
|
|
||||||
Website + backend (active stack):
|
|
||||||
|
|
||||||
- [Getting Started](docs/website/getting-started.md)
|
|
||||||
- [Architecture](docs/architecture.md)
|
|
||||||
- [Docker Guide](docs/website/docker.md)
|
|
||||||
- [Testing](docs/website/testing.md)
|
|
||||||
- [Environment Variables](docs/website/environment-variables.md)
|
|
||||||
- [Token Validation](docs/website/token-validation.md)
|
|
||||||
|
|
||||||
Data generation pipeline (C++):
|
|
||||||
|
|
||||||
- [Pipeline README](docs/pipeline/README.md)
|
|
||||||
- [Ethics & Known Issues](docs/pipeline/ETHICS-AND-KNOWN-ISSUES.md)
|
|
||||||
|
|
||||||
## Diagrams
|
## Diagrams
|
||||||
|
|
||||||
- [Architecture](docs/website/diagrams-out/architecture.svg)
|
- [Architecture](docs/diagrams-out/architecture.svg) - Layered architecture
|
||||||
- [Deployment](docs/website/diagrams-out/deployment.svg)
|
- [Deployment](docs/diagrams-out/deployment.svg) - Docker topology
|
||||||
- [Authentication Flow](docs/website/diagrams-out/authentication-flow.svg)
|
- [Authentication Flow](docs/diagrams-out/authentication-flow.svg) - Auth sequence
|
||||||
- [Database Schema](docs/website/diagrams-out/database-schema.svg)
|
- [Database Schema](docs/diagrams-out/database-schema.svg) - Entity relationships
|
||||||
|
|
||||||
## Current Status
|
## Current Status
|
||||||
|
|
||||||
Active areas in the repository:
|
Active areas in the repository:
|
||||||
|
|
||||||
- .NET 10 backend (layered architecture) + SQL Server
|
- .NET 10 backend with layered architecture and SQL Server
|
||||||
- React 19 website (React Router 7 + Vite)
|
- React Router 7 website in `src/Website`
|
||||||
- Shared Biergarten theme system + Storybook coverage
|
- Shared Biergarten theme system with a theme guide route
|
||||||
- Auth flows and account/email integration (local Mailpit in dev compose)
|
- Storybook stories and browser-based checks for shared UI
|
||||||
- Data generation pipeline with C++ and Llama.cpp
|
- Auth demo flows for home, login, register, dashboard, logout, and confirmation
|
||||||
|
- Toast-based feedback for auth outcomes
|
||||||
|
|
||||||
Archived/reference areas:
|
Legacy area retained for reference:
|
||||||
|
|
||||||
- `archive/next-js-web-app/` contains an older Next.js frontend retained for
|
- `src/Website-v1` contains the archived Next.js frontend and is no longer the active website
|
||||||
reference
|
|
||||||
|
|
||||||
## Tech Stack
|
## Tech Stack
|
||||||
|
|
||||||
@@ -59,43 +43,36 @@ Archived/reference areas:
|
|||||||
- **UI Documentation**: Storybook 10, Vitest browser mode, Playwright
|
- **UI Documentation**: Storybook 10, Vitest browser mode, Playwright
|
||||||
- **Testing**: xUnit, Reqnroll (BDD), FluentAssertions, Moq
|
- **Testing**: xUnit, Reqnroll (BDD), FluentAssertions, Moq
|
||||||
- **Infrastructure**: Docker, Docker Compose
|
- **Infrastructure**: Docker, Docker Compose
|
||||||
- **Security**: Argon2id password hashing, JWT access/refresh/confirmation
|
- **Security**: Argon2id password hashing, JWT access/refresh/confirmation tokens
|
||||||
tokens
|
|
||||||
- **Data Pipeline**: C++20, CMake, Boost, libcurl, SQLite, llama.cpp
|
|
||||||
|
|
||||||
## Quick Start
|
## Quick Start
|
||||||
|
|
||||||
For full setup details, use [Getting Started](docs/website/getting-started.md).
|
### Backend
|
||||||
This section is the shortest path to a working dev environment.
|
|
||||||
|
|
||||||
### Backend (Docker)
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone https://github.com/aaronpo97/the-biergarten-app
|
git clone https://github.com/aaronpo97/the-biergarten-app
|
||||||
cd the-biergarten-app
|
cd the-biergarten-app
|
||||||
|
cp .env.example .env.dev
|
||||||
cp web/.env.example web/.env.dev
|
docker compose -f docker-compose.dev.yaml up -d
|
||||||
docker compose --env-file web/.env.dev -f web/docker-compose.dev.yaml up --build -d
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Backend access:
|
Backend access:
|
||||||
|
|
||||||
- API Swagger: http://localhost:8080/swagger
|
- API Swagger: http://localhost:8080/swagger
|
||||||
- Health Check: http://localhost:8080/health
|
- Health Check: http://localhost:8080/health
|
||||||
- Mailpit UI (dev SMTP): http://localhost:8025
|
|
||||||
|
|
||||||
### Frontend (Node)
|
### Frontend
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd web/frontend
|
cd src/Website
|
||||||
npm install
|
npm install
|
||||||
API_BASE_URL=http://localhost:8080 SESSION_SECRET=dev-secret-change-me npm run dev
|
API_BASE_URL=http://localhost:8080 SESSION_SECRET=dev-secret npm run dev
|
||||||
```
|
```
|
||||||
|
|
||||||
Optional frontend tools:
|
Optional frontend tools:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd web/frontend
|
cd src/Website
|
||||||
npm run storybook
|
npm run storybook
|
||||||
npm run test:storybook
|
npm run test:storybook
|
||||||
npm run test:storybook:playwright
|
npm run test:storybook:playwright
|
||||||
@@ -104,42 +81,62 @@ npm run test:storybook:playwright
|
|||||||
## Repository Structure
|
## Repository Structure
|
||||||
|
|
||||||
```text
|
```text
|
||||||
web/
|
src/Core/ Backend projects (.NET)
|
||||||
backend/ .NET API + domain/service/infrastructure + DB projects
|
src/Website/ Active React Router frontend
|
||||||
frontend/ React Router website + Storybook + Playwright/Vitest
|
src/Website-v1/ Archived legacy Next.js frontend
|
||||||
|
docs/ Active project documentation
|
||||||
tooling/
|
docs/archive/ Archived legacy documentation
|
||||||
pipeline/ C++20 seed-data generation CLI (CMake)
|
|
||||||
|
|
||||||
docs/
|
|
||||||
architecture.md High-level architecture overview
|
|
||||||
website/ Backend/frontend setup, docker, testing, diagrams
|
|
||||||
pipeline/ Pipeline docs, ethics notes, PlantUML diagrams
|
|
||||||
|
|
||||||
archive/
|
|
||||||
next-js-web-app/ Older Next.js frontend (reference only)
|
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Key Features
|
||||||
|
|
||||||
|
Implemented today:
|
||||||
|
|
||||||
|
- User registration and login against the API
|
||||||
|
- JWT-based auth with access, refresh, and confirmation flows
|
||||||
|
- SQL Server migrations and seed projects
|
||||||
|
- Shared form components and auth screens
|
||||||
|
- Theme switching with Lager, Stout, Cassis, and Weizen variants
|
||||||
|
- Storybook documentation and automated story interaction tests
|
||||||
|
- Toast feedback for auth-related outcomes
|
||||||
|
|
||||||
|
Planned next:
|
||||||
|
|
||||||
|
- Brewery discovery and management
|
||||||
|
- Beer reviews and ratings
|
||||||
|
- Social follow relationships
|
||||||
|
- Geospatial brewery experiences
|
||||||
|
- Additional frontend routes beyond the auth demo
|
||||||
|
|
||||||
## Testing
|
## Testing
|
||||||
|
|
||||||
Run the backend test stack with Docker:
|
Backend suites:
|
||||||
|
|
||||||
|
- `API.Specs` - integration tests
|
||||||
|
- `Infrastructure.Repository.Tests` - repository unit tests
|
||||||
|
- `Service.Auth.Tests` - service unit tests
|
||||||
|
|
||||||
|
Frontend suites:
|
||||||
|
|
||||||
|
- Storybook interaction tests via Vitest
|
||||||
|
- Storybook browser regression checks via Playwright
|
||||||
|
|
||||||
|
Run all backend tests with Docker:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker compose --env-file web/.env.test -f web/docker-compose.test.yaml up --abort-on-container-exit
|
docker compose -f docker-compose.test.yaml up --abort-on-container-exit
|
||||||
```
|
```
|
||||||
|
|
||||||
See [Testing](docs/website/testing.md) for the full command list.
|
See [Testing](docs/testing.md) for the full command list.
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
Common active variables:
|
Common active variables:
|
||||||
|
|
||||||
- Backend/Docker: `DB_SERVER`, `DB_NAME`, `DB_USER`, `DB_PASSWORD`,
|
- Backend: `DB_SERVER`, `DB_NAME`, `DB_USER`, `DB_PASSWORD`, `ACCESS_TOKEN_SECRET`, `REFRESH_TOKEN_SECRET`, `CONFIRMATION_TOKEN_SECRET`
|
||||||
`ACCESS_TOKEN_SECRET`, `REFRESH_TOKEN_SECRET`, `CONFIRMATION_TOKEN_SECRET`,
|
- Frontend: `API_BASE_URL`, `SESSION_SECRET`, `NODE_ENV`
|
||||||
`WEBSITE_BASE_URL`
|
|
||||||
- Frontend runtime: `API_BASE_URL`, `SESSION_SECRET`, `NODE_ENV`
|
|
||||||
|
|
||||||
See [Environment Variables](docs/website/environment-variables.md) for details.
|
See [Environment Variables](docs/environment-variables.md) for details.
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
@@ -148,3 +145,15 @@ See [Environment Variables](docs/website/environment-variables.md) for details.
|
|||||||
3. Commit your changes (`git commit -m 'Add amazing feature'`)
|
3. Commit your changes (`git commit -m 'Add amazing feature'`)
|
||||||
4. Push to the branch (`git push origin feature/amazing-feature`)
|
4. Push to the branch (`git push origin feature/amazing-feature`)
|
||||||
5. Open a Pull Request
|
5. Open a Pull Request
|
||||||
|
|
||||||
|
### Development Workflow
|
||||||
|
|
||||||
|
1. Start development environment: `docker compose -f docker-compose.dev.yaml up -d`
|
||||||
|
2. Make changes to code
|
||||||
|
3. Run tests: `docker compose -f docker-compose.test.yaml up --abort-on-container-exit`
|
||||||
|
4. Rebuild if needed: `docker compose -f docker-compose.dev.yaml up -d --build api.core`
|
||||||
|
|
||||||
|
## Support
|
||||||
|
|
||||||
|
- **Documentation**: [docs/](docs/)
|
||||||
|
- **Architecture**: See [Architecture Guide](docs/architecture.md)
|
||||||
|
|||||||
@@ -13,11 +13,7 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- sqlserverdata-dev:/var/opt/mssql
|
- sqlserverdata-dev:/var/opt/mssql
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test:
|
test: ["CMD-SHELL", "/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1"]
|
||||||
[
|
|
||||||
"CMD-SHELL",
|
|
||||||
"/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1",
|
|
||||||
]
|
|
||||||
interval: 10s
|
interval: 10s
|
||||||
timeout: 5s
|
timeout: 5s
|
||||||
retries: 12
|
retries: 12
|
||||||
@@ -32,7 +28,7 @@ services:
|
|||||||
sqlserver:
|
sqlserver:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
build:
|
build:
|
||||||
context: ./backend/Database
|
context: ./src/Core/Database
|
||||||
dockerfile: Database.Migrations/Dockerfile
|
dockerfile: Database.Migrations/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -56,7 +52,7 @@ services:
|
|||||||
database.migrations:
|
database.migrations:
|
||||||
condition: service_completed_successfully
|
condition: service_completed_successfully
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: Database/Database.Seed/Dockerfile
|
dockerfile: Database/Database.Seed/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -13,11 +13,7 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- sqlserverdata-dev:/var/opt/mssql
|
- sqlserverdata-dev:/var/opt/mssql
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test:
|
test: [ "CMD-SHELL", "/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1" ]
|
||||||
[
|
|
||||||
"CMD-SHELL",
|
|
||||||
"/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1",
|
|
||||||
]
|
|
||||||
interval: 10s
|
interval: 10s
|
||||||
timeout: 5s
|
timeout: 5s
|
||||||
retries: 12
|
retries: 12
|
||||||
@@ -32,7 +28,7 @@ services:
|
|||||||
sqlserver:
|
sqlserver:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
build:
|
build:
|
||||||
context: ./backend/Database
|
context: ./src/Core/Database
|
||||||
dockerfile: Database.Migrations/Dockerfile
|
dockerfile: Database.Migrations/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -56,7 +52,7 @@ services:
|
|||||||
database.migrations:
|
database.migrations:
|
||||||
condition: service_completed_successfully
|
condition: service_completed_successfully
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: Database/Database.Seed/Dockerfile
|
dockerfile: Database/Database.Seed/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -79,7 +75,7 @@ services:
|
|||||||
database.seed:
|
database.seed:
|
||||||
condition: service_completed_successfully
|
condition: service_completed_successfully
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: API/API.Core/Dockerfile
|
dockerfile: API/API.Core/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -13,11 +13,7 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- sqlserverdata-dev:/var/opt/mssql
|
- sqlserverdata-dev:/var/opt/mssql
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test:
|
test: [ "CMD-SHELL", "/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1" ]
|
||||||
[
|
|
||||||
"CMD-SHELL",
|
|
||||||
"/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1",
|
|
||||||
]
|
|
||||||
interval: 10s
|
interval: 10s
|
||||||
timeout: 5s
|
timeout: 5s
|
||||||
retries: 12
|
retries: 12
|
||||||
@@ -32,7 +28,7 @@ services:
|
|||||||
sqlserver:
|
sqlserver:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
build:
|
build:
|
||||||
context: ./backend/Database
|
context: ./src/Core/Database
|
||||||
dockerfile: Database.Migrations/Dockerfile
|
dockerfile: Database.Migrations/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -70,7 +66,7 @@ services:
|
|||||||
database.migrations:
|
database.migrations:
|
||||||
condition: service_completed_successfully
|
condition: service_completed_successfully
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: Database/Database.Seed/Dockerfile
|
dockerfile: Database/Database.Seed/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -11,11 +11,7 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- sqlserverdata-prod:/var/opt/mssql
|
- sqlserverdata-prod:/var/opt/mssql
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test:
|
test: ["CMD-SHELL", "/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1"]
|
||||||
[
|
|
||||||
"CMD-SHELL",
|
|
||||||
"/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1",
|
|
||||||
]
|
|
||||||
interval: 10s
|
interval: 10s
|
||||||
timeout: 5s
|
timeout: 5s
|
||||||
retries: 12
|
retries: 12
|
||||||
@@ -31,7 +27,7 @@ services:
|
|||||||
sqlserver:
|
sqlserver:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
build:
|
build:
|
||||||
context: ./backend/Database
|
context: ./src/Core/Database
|
||||||
dockerfile: Database.Migrations/Dockerfile
|
dockerfile: Database.Migrations/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -54,7 +50,7 @@ services:
|
|||||||
sqlserver:
|
sqlserver:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: API/API.Core/Dockerfile
|
dockerfile: API/API.Core/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -12,11 +12,7 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- sqlserverdata-test:/var/opt/mssql
|
- sqlserverdata-test:/var/opt/mssql
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test:
|
test: [ "CMD-SHELL", "/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1" ]
|
||||||
[
|
|
||||||
"CMD-SHELL",
|
|
||||||
"/opt/mssql-tools18/bin/sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1' || exit 1",
|
|
||||||
]
|
|
||||||
interval: 10s
|
interval: 10s
|
||||||
timeout: 5s
|
timeout: 5s
|
||||||
retries: 12
|
retries: 12
|
||||||
@@ -32,7 +28,7 @@ services:
|
|||||||
sqlserver:
|
sqlserver:
|
||||||
condition: service_healthy
|
condition: service_healthy
|
||||||
build:
|
build:
|
||||||
context: ./backend/Database
|
context: ./src/Core/Database
|
||||||
dockerfile: Database.Migrations/Dockerfile
|
dockerfile: Database.Migrations/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -56,7 +52,7 @@ services:
|
|||||||
database.migrations:
|
database.migrations:
|
||||||
condition: service_completed_successfully
|
condition: service_completed_successfully
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: Database/Database.Seed/Dockerfile
|
dockerfile: Database/Database.Seed/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -79,7 +75,7 @@ services:
|
|||||||
database.seed:
|
database.seed:
|
||||||
condition: service_completed_successfully
|
condition: service_completed_successfully
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: API/API.Specs/Dockerfile
|
dockerfile: API/API.Specs/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -107,7 +103,7 @@ services:
|
|||||||
database.seed:
|
database.seed:
|
||||||
condition: service_completed_successfully
|
condition: service_completed_successfully
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: Infrastructure/Infrastructure.Repository.Tests/Dockerfile
|
dockerfile: Infrastructure/Infrastructure.Repository.Tests/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -127,7 +123,7 @@ services:
|
|||||||
database.seed:
|
database.seed:
|
||||||
condition: service_completed_successfully
|
condition: service_completed_successfully
|
||||||
build:
|
build:
|
||||||
context: ./backend
|
context: ./src/Core
|
||||||
dockerfile: Service/Service.Auth.Tests/Dockerfile
|
dockerfile: Service/Service.Auth.Tests/Dockerfile
|
||||||
args:
|
args:
|
||||||
BUILD_CONFIGURATION: Release
|
BUILD_CONFIGURATION: Release
|
||||||
@@ -4,28 +4,24 @@ This document describes the active architecture of The Biergarten App.
|
|||||||
|
|
||||||
## High-Level Overview
|
## High-Level Overview
|
||||||
|
|
||||||
The Biergarten App is a monorepo with a clear split between the backend and the
|
The Biergarten App is a monorepo with a clear split between the backend and the active
|
||||||
active website:
|
website:
|
||||||
|
|
||||||
- **Backend**: .NET 10 Web API with SQL Server and a layered architecture
|
- **Backend**: .NET 10 Web API with SQL Server and a layered architecture
|
||||||
- **Frontend**: React 19 + React Router 7 website in `src/Website`
|
- **Frontend**: React 19 + React Router 7 website in `src/Website`
|
||||||
- **Architecture Style**: Layered backend plus server-rendered React frontend
|
- **Architecture Style**: Layered backend plus server-rendered React frontend
|
||||||
|
|
||||||
The legacy Next.js frontend has been retained in `src/Website-v1` for reference
|
The legacy Next.js frontend has been retained in `src/Website-v1` for reference only and is
|
||||||
only and is documented in
|
documented in [archive/legacy-website-v1.md](archive/legacy-website-v1.md).
|
||||||
[archive/legacy-website-v1.md](archive/legacy-website-v1.md).
|
|
||||||
|
|
||||||
## Diagrams
|
## Diagrams
|
||||||
|
|
||||||
For visual representations, see:
|
For visual representations, see:
|
||||||
|
|
||||||
- [architecture.svg](diagrams-out/architecture.svg) - Layered architecture
|
- [architecture.svg](diagrams-out/architecture.svg) - Layered architecture diagram
|
||||||
diagram
|
|
||||||
- [deployment.svg](diagrams-out/deployment.svg) - Docker deployment diagram
|
- [deployment.svg](diagrams-out/deployment.svg) - Docker deployment diagram
|
||||||
- [authentication-flow.svg](diagrams-out/authentication-flow.svg) -
|
- [authentication-flow.svg](diagrams-out/authentication-flow.svg) - Authentication workflow
|
||||||
Authentication workflow
|
- [database-schema.svg](diagrams-out/database-schema.svg) - Database relationships
|
||||||
- [database-schema.svg](diagrams-out/database-schema.svg) - Database
|
|
||||||
relationships
|
|
||||||
|
|
||||||
## Backend Architecture
|
## Backend Architecture
|
||||||
|
|
||||||
@@ -222,8 +218,7 @@ public interface IAuthRepository
|
|||||||
|
|
||||||
### Active Website (`src/Website`)
|
### Active Website (`src/Website`)
|
||||||
|
|
||||||
The current website is a React Router 7 application with server-side rendering
|
The current website is a React Router 7 application with server-side rendering enabled.
|
||||||
enabled.
|
|
||||||
|
|
||||||
```text
|
```text
|
||||||
src/Website/
|
src/Website/
|
||||||
@@ -249,22 +244,20 @@ src/Website/
|
|||||||
|
|
||||||
### Theme System
|
### Theme System
|
||||||
|
|
||||||
The active website uses semantic DaisyUI theme tokens backed by four Biergarten
|
The active website uses semantic DaisyUI theme tokens backed by four Biergarten themes:
|
||||||
themes:
|
|
||||||
|
|
||||||
- Biergarten Lager
|
- Biergarten Lager
|
||||||
- Biergarten Stout
|
- Biergarten Stout
|
||||||
- Biergarten Cassis
|
- Biergarten Cassis
|
||||||
- Biergarten Weizen
|
- Biergarten Weizen
|
||||||
|
|
||||||
All component styling should prefer semantic tokens such as `primary`,
|
All component styling should prefer semantic tokens such as `primary`, `success`,
|
||||||
`success`, `surface`, and `highlight` instead of hard-coded color values.
|
`surface`, and `highlight` instead of hard-coded color values.
|
||||||
|
|
||||||
### Legacy Frontend
|
### Legacy Frontend
|
||||||
|
|
||||||
The previous Next.js frontend has been archived at `src/Website-v1`. Active
|
The previous Next.js frontend has been archived at `src/Website-v1`. Active product and
|
||||||
product and engineering documentation should point to `src/Website`, while
|
engineering documentation should point to `src/Website`, while legacy notes live in
|
||||||
legacy notes live in
|
|
||||||
[archive/legacy-website-v1.md](archive/legacy-website-v1.md).
|
[archive/legacy-website-v1.md](archive/legacy-website-v1.md).
|
||||||
|
|
||||||
## Security Architecture
|
## Security Architecture
|
||||||
@@ -394,8 +387,8 @@ For details, see [Docker Guide](docker.md).
|
|||||||
|
|
||||||
### Health Checks
|
### Health Checks
|
||||||
|
|
||||||
**SQL Server**: Validates database connectivity **API**: Checks service health
|
**SQL Server**: Validates database connectivity **API**: Checks service health and
|
||||||
and dependencies
|
dependencies
|
||||||
|
|
||||||
**Configuration**:
|
**Configuration**:
|
||||||
|
|
||||||
|
|||||||
56
docs/archive/legacy-website-v1.md
Normal file
56
docs/archive/legacy-website-v1.md
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
# Legacy Website Archive (`src/Website-v1`)
|
||||||
|
|
||||||
|
This archive captures high-level notes about the previous Biergarten frontend so active
|
||||||
|
project documentation can focus on the current website in `src/Website`.
|
||||||
|
|
||||||
|
## Status
|
||||||
|
|
||||||
|
- `src/Website-v1` is retained for historical reference only
|
||||||
|
- It is not the active frontend used by current setup, docs, or testing guidance
|
||||||
|
- New product and engineering work should target `src/Website`
|
||||||
|
|
||||||
|
## Legacy Stack Summary
|
||||||
|
|
||||||
|
The archived frontend used a different application model from the current website:
|
||||||
|
|
||||||
|
- Next.js 14
|
||||||
|
- React 18
|
||||||
|
- Prisma
|
||||||
|
- Postgres / Neon-hosted database workflows
|
||||||
|
- Next.js API routes and server-side controllers
|
||||||
|
- Additional third-party integrations such as Cloudinary, Mapbox, and SparkPost
|
||||||
|
|
||||||
|
## Why It Was Archived
|
||||||
|
|
||||||
|
The active website moved to a React Router-based frontend that talks directly to the .NET
|
||||||
|
API. As part of that shift, the main docs were updated to describe:
|
||||||
|
|
||||||
|
- `src/Website` as the active frontend
|
||||||
|
- React Router route modules and server rendering
|
||||||
|
- Storybook-based component documentation and tests
|
||||||
|
- Current frontend runtime variables: `API_BASE_URL`, `SESSION_SECRET`, and `NODE_ENV`
|
||||||
|
|
||||||
|
## Legacy Documentation Topics Moved Out of Active Docs
|
||||||
|
|
||||||
|
The following categories were removed from active documentation and intentionally archived:
|
||||||
|
|
||||||
|
- Next.js application structure guidance
|
||||||
|
- Prisma and Postgres frontend setup
|
||||||
|
- Legacy frontend environment variables
|
||||||
|
- External service setup that only applied to `src/Website-v1`
|
||||||
|
- Old frontend local setup instructions
|
||||||
|
|
||||||
|
## When To Use This Archive
|
||||||
|
|
||||||
|
Use this file only if you need to:
|
||||||
|
|
||||||
|
- inspect the historical frontend implementation
|
||||||
|
- compare old flows against the current website
|
||||||
|
- migrate or recover legacy logic from `src/Website-v1`
|
||||||
|
|
||||||
|
For all active work, use:
|
||||||
|
|
||||||
|
- [Getting Started](../getting-started.md)
|
||||||
|
- [Architecture](../architecture.md)
|
||||||
|
- [Environment Variables](../environment-variables.md)
|
||||||
|
- [Testing](../testing.md)
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
# Docker Guide
|
# Docker Guide
|
||||||
|
|
||||||
This document covers Docker deployment, configuration, and troubleshooting for
|
This document covers Docker deployment, configuration, and troubleshooting for The
|
||||||
The Biergarten App.
|
Biergarten App.
|
||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
@@ -13,8 +13,7 @@ The project uses Docker Compose to orchestrate multiple services:
|
|||||||
- .NET API
|
- .NET API
|
||||||
- Test runners
|
- Test runners
|
||||||
|
|
||||||
See the [deployment diagram](diagrams/pdf/deployment.pdf) for visual
|
See the [deployment diagram](diagrams/pdf/deployment.pdf) for visual representation.
|
||||||
representation.
|
|
||||||
|
|
||||||
## Docker Compose Environments
|
## Docker Compose Environments
|
||||||
|
|
||||||
@@ -145,11 +144,7 @@ api.core / tests (start when ready)
|
|||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test:
|
test: ['CMD-SHELL', "sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1'"]
|
||||||
[
|
|
||||||
"CMD-SHELL",
|
|
||||||
"sqlcmd -S localhost -U sa -P '${DB_PASSWORD}' -C -Q 'SELECT 1'",
|
|
||||||
]
|
|
||||||
interval: 10s
|
interval: 10s
|
||||||
timeout: 5s
|
timeout: 5s
|
||||||
retries: 12
|
retries: 12
|
||||||
@@ -214,16 +209,16 @@ Each environment uses isolated bridge networks:
|
|||||||
All containers are configured via environment variables from `.env` files:
|
All containers are configured via environment variables from `.env` files:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
env_file: ".env.dev" # or .env.test, .env.prod
|
env_file: '.env.dev' # or .env.test, .env.prod
|
||||||
|
|
||||||
environment:
|
environment:
|
||||||
ASPNETCORE_ENVIRONMENT: "Development"
|
ASPNETCORE_ENVIRONMENT: 'Development'
|
||||||
DOTNET_RUNNING_IN_CONTAINER: "true"
|
DOTNET_RUNNING_IN_CONTAINER: 'true'
|
||||||
DB_SERVER: "${DB_SERVER}"
|
DB_SERVER: '${DB_SERVER}'
|
||||||
DB_NAME: "${DB_NAME}"
|
DB_NAME: '${DB_NAME}'
|
||||||
DB_USER: "${DB_USER}"
|
DB_USER: '${DB_USER}'
|
||||||
DB_PASSWORD: "${DB_PASSWORD}"
|
DB_PASSWORD: '${DB_PASSWORD}'
|
||||||
JWT_SECRET: "${JWT_SECRET}"
|
JWT_SECRET: '${JWT_SECRET}'
|
||||||
```
|
```
|
||||||
|
|
||||||
For complete list, see [Environment Variables](environment-variables.md).
|
For complete list, see [Environment Variables](environment-variables.md).
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
# Environment Variables
|
# Environment Variables
|
||||||
|
|
||||||
This document covers the active environment variables used by the current
|
This document covers the active environment variables used by the current Biergarten
|
||||||
Biergarten stack.
|
stack.
|
||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
@@ -19,8 +19,8 @@ Direct environment variable access via `Environment.GetEnvironmentVariable()`.
|
|||||||
|
|
||||||
### Frontend (`src/Website`)
|
### Frontend (`src/Website`)
|
||||||
|
|
||||||
The active website reads runtime values from the server environment for its auth
|
The active website reads runtime values from the server environment for its auth and API
|
||||||
and API integration.
|
integration.
|
||||||
|
|
||||||
### Docker
|
### Docker
|
||||||
|
|
||||||
@@ -54,15 +54,14 @@ Provide complete connection string:
|
|||||||
DB_CONNECTION_STRING="Server=localhost,1433;Database=Biergarten;User Id=sa;Password=YourStrong!Passw0rd;TrustServerCertificate=True;"
|
DB_CONNECTION_STRING="Server=localhost,1433;Database=Biergarten;User Id=sa;Password=YourStrong!Passw0rd;TrustServerCertificate=True;"
|
||||||
```
|
```
|
||||||
|
|
||||||
**Priority**: `DB_CONNECTION_STRING` is checked first. If not found, connection
|
**Priority**: `DB_CONNECTION_STRING` is checked first. If not found, connection string is
|
||||||
string is built from components.
|
built from components.
|
||||||
|
|
||||||
**Implementation**: See `DefaultSqlConnectionFactory.cs`
|
**Implementation**: See `DefaultSqlConnectionFactory.cs`
|
||||||
|
|
||||||
### JWT Authentication Secrets (Backend)
|
### JWT Authentication Secrets (Backend)
|
||||||
|
|
||||||
The backend uses separate secrets for different token types to enable
|
The backend uses separate secrets for different token types to enable independent key rotation and validation isolation.
|
||||||
independent key rotation and validation isolation.
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Access token secret (1-hour tokens)
|
# Access token secret (1-hour tokens)
|
||||||
@@ -132,8 +131,8 @@ DOTNET_RUNNING_IN_CONTAINER=true # Flag for container execution
|
|||||||
|
|
||||||
## Frontend Variables (`src/Website`)
|
## Frontend Variables (`src/Website`)
|
||||||
|
|
||||||
The active website does not use the old Next.js/Prisma environment model. Its
|
The active website does not use the old Next.js/Prisma environment model. Its core runtime
|
||||||
core runtime variables are:
|
variables are:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
API_BASE_URL=http://localhost:8080 # Base URL for the .NET API
|
API_BASE_URL=http://localhost:8080 # Base URL for the .NET API
|
||||||
@@ -209,10 +208,9 @@ cp .env.example .env.dev
|
|||||||
|
|
||||||
## Legacy Frontend Variables
|
## Legacy Frontend Variables
|
||||||
|
|
||||||
Variables for the archived Next.js frontend (`src/Website-v1`) have been removed
|
Variables for the archived Next.js frontend (`src/Website-v1`) have been removed from this
|
||||||
from this active reference. See
|
active reference. See [archive/legacy-website-v1.md](archive/legacy-website-v1.md) if you
|
||||||
[archive/legacy-website-v1.md](archive/legacy-website-v1.md) if you need the
|
need the legacy Prisma, Cloudinary, Mapbox, or SparkPost notes.
|
||||||
legacy Prisma, Cloudinary, Mapbox, or SparkPost notes.
|
|
||||||
|
|
||||||
**Docker Compose Mapping**:
|
**Docker Compose Mapping**:
|
||||||
|
|
||||||
@@ -245,8 +243,8 @@ legacy Prisma, Cloudinary, Mapbox, or SparkPost notes.
|
|||||||
| `MSSQL_PID` | | | ✓ | No | SQL Server edition |
|
| `MSSQL_PID` | | | ✓ | No | SQL Server edition |
|
||||||
| `DOTNET_RUNNING_IN_CONTAINER` | ✓ | | ✓ | No | Container flag |
|
| `DOTNET_RUNNING_IN_CONTAINER` | ✓ | | ✓ | No | Container flag |
|
||||||
|
|
||||||
\* Either `DB_CONNECTION_STRING` OR the component variables (`DB_SERVER`,
|
\* Either `DB_CONNECTION_STRING` OR the component variables (`DB_SERVER`, `DB_NAME`,
|
||||||
`DB_NAME`, `DB_USER`, `DB_PASSWORD`) must be provided.
|
`DB_USER`, `DB_PASSWORD`) must be provided.
|
||||||
|
|
||||||
## Validation
|
## Validation
|
||||||
|
|
||||||
@@ -260,8 +258,8 @@ Variables are validated at startup:
|
|||||||
|
|
||||||
### Frontend Validation
|
### Frontend Validation
|
||||||
|
|
||||||
The active website relies on runtime defaults for local development and the
|
The active website relies on runtime defaults for local development and the surrounding
|
||||||
surrounding server environment in deployed environments.
|
server environment in deployed environments.
|
||||||
|
|
||||||
- `API_BASE_URL` defaults to `http://localhost:8080`
|
- `API_BASE_URL` defaults to `http://localhost:8080`
|
||||||
- `SESSION_SECRET` falls back to a development-only local secret
|
- `SESSION_SECRET` falls back to a development-only local secret
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
# Getting Started
|
# Getting Started
|
||||||
|
|
||||||
This guide covers local setup for the current Biergarten stack: the .NET backend
|
This guide covers local setup for the current Biergarten stack: the .NET backend in
|
||||||
in `src/Core` and the active React Router frontend in `src/Website`.
|
`src/Core` and the active React Router frontend in `src/Website`.
|
||||||
|
|
||||||
## Prerequisites
|
## Prerequisites
|
||||||
|
|
||||||
@@ -128,9 +128,8 @@ dotnet run --project API/API.Core/API.Core.csproj
|
|||||||
|
|
||||||
## Legacy Frontend Note
|
## Legacy Frontend Note
|
||||||
|
|
||||||
The previous Next.js frontend now lives in `src/Website-v1` and is not the
|
The previous Next.js frontend now lives in `src/Website-v1` and is not the active website.
|
||||||
active website. Legacy setup details have been moved to
|
Legacy setup details have been moved to [docs/archive/legacy-website-v1.md](archive/legacy-website-v1.md).
|
||||||
[docs/archive/legacy-website-v1.md](archive/legacy-website-v1.md).
|
|
||||||
|
|
||||||
## Next Steps
|
## Next Steps
|
||||||
|
|
||||||
@@ -1,336 +0,0 @@
|
|||||||
# Ethics, Bias, and Known Issues
|
|
||||||
|
|
||||||
This document covers the ethical context of the Biergarten Pipeline's output,
|
|
||||||
the model's biases, and known issues including hallucinated brewing science and
|
|
||||||
low-resource language failures.
|
|
||||||
|
|
||||||
> Note that all testing was used using `google_gemma-4-E4B-it-Q6_K.gguf`.
|
|
||||||
|
|
||||||
## Table of Contents
|
|
||||||
|
|
||||||
- [What This Dataset Is](#what-this-dataset-is)
|
|
||||||
- [What This Dataset Is Not](#what-this-dataset-is-not)
|
|
||||||
- [Model Bias and Language Quality](#model-bias-and-language-quality)
|
|
||||||
- [Western and Eurocentric Lens](#western-and-eurocentric-lens)
|
|
||||||
- [Wikipedia Enrichment](#wikipedia-enrichment)
|
|
||||||
- [The "Avoid AI Phrases" Prompt Instruction](#the-avoid-ai-phrases-prompt-instruction)
|
|
||||||
- [Known Issues](#known-issues)
|
|
||||||
- [Hallucinated Brewing Techniques](#hallucinated-brewing-techniques)
|
|
||||||
- [Low-Resource Language Hallucination](#low-resource-language-hallucination)
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## What This Dataset Is
|
|
||||||
|
|
||||||
This is AI-generated fixture data for a proof-of-concept version of The
|
|
||||||
Biergarten App. Anyone who interacts with an application seeded from this
|
|
||||||
pipeline must be told upfront that the content is AI-generated.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## What This Dataset Is Not
|
|
||||||
|
|
||||||
The pipeline is not intended to produce accurate brewing science, faithful
|
|
||||||
cultural representation, or reliable local-language text. Hallucinations such as
|
|
||||||
invented fermentation techniques, or incoherent local-language prose, are
|
|
||||||
expected, observed, and partially documented in [Known Issues](#known-issues)
|
|
||||||
below.
|
|
||||||
|
|
||||||
Human control sits at the context layer (i.e. prompt design, Wikipedia
|
|
||||||
enrichment). Statistical output shapes in future pipeline stages (check-in
|
|
||||||
distributions, rating skews, activity profiles) will be handled the same way.
|
|
||||||
|
|
||||||
**Treat this data as an exercise in prompt engineering and model behaviour, not
|
|
||||||
as a source of truth for brewing techniques or cultural representation.**
|
|
||||||
|
|
||||||
**Natural language processing, although a powerful tool for data analysis and
|
|
||||||
generation is to be taken with scrutiny. Human language is not simply just data
|
|
||||||
points to be analyzed, but it also carries deep cultural and human meaning that
|
|
||||||
artificial intelligence is incapable of.**
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Model Bias and Language Quality
|
|
||||||
|
|
||||||
The underlying model's training biases surface within this pipeline. Output
|
|
||||||
quality tracks with how well a language is represented in the training corpus:
|
|
||||||
standard French (`fr-FR`) produces coherent text; regional variants like `fr-CD`
|
|
||||||
and `fr-CI` are noticeably weaker; low-resource languages like Welsh, Māori, and
|
|
||||||
Sicilian produce output that is syntactically plausible but often semantically
|
|
||||||
broken.
|
|
||||||
|
|
||||||
This is a property of the training distribution, not something that can be
|
|
||||||
mitigated through prompt design. This is a well-documented characteristic of
|
|
||||||
large language models trained predominantly on English-language
|
|
||||||
material.[^llm-bias]
|
|
||||||
|
|
||||||
Mitigations are documented in
|
|
||||||
[Known Issues: Low-Resource Language Hallucination](#low-resource-language-hallucination).
|
|
||||||
|
|
||||||
### Western and Eurocentric Lens
|
|
||||||
|
|
||||||
The model's training data skews heavily Western and North American. When
|
|
||||||
generating brewery descriptions for Kinshasa, Abidjan, or Osaka, for example, it
|
|
||||||
defaults to framing and cultural reference points drawn from that perspective
|
|
||||||
rather than from the lived context of those cities. Wikipedia enrichment grounds
|
|
||||||
some generation in city-specific material, but it does not eliminate the skew.
|
|
||||||
|
|
||||||
**Output should be read with an understanding of this bias.**
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Wikipedia Enrichment
|
|
||||||
|
|
||||||
City and beer context is fetched from the Wikipedia API. Wikipedia text is
|
|
||||||
co-licensed under the **Creative Commons Attribution-ShareAlike 4.0
|
|
||||||
International License (CC BY-SA 4.0)** and the **GNU Free Documentation License
|
|
||||||
(GFDL)**.[^wp-license]
|
|
||||||
|
|
||||||
Wikipedia's own accuracy limitations and editorial biases can propagate into
|
|
||||||
generated descriptions.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## The "Avoid AI Phrases" Prompt Instruction
|
|
||||||
|
|
||||||
The system prompt instructs the model to avoid common AI-generated phrasing
|
|
||||||
patterns. This is a prompt engineering experiment:
|
|
||||||
|
|
||||||
> How far can a model be pushed against its own stylistic defaults?
|
|
||||||
|
|
||||||
This is not an attempt to disguise the content as human-written. All downstream
|
|
||||||
consumers are informed of the AI-generated origin before engagement.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Known Issues
|
|
||||||
|
|
||||||
### Hallucinated Brewing Techniques
|
|
||||||
|
|
||||||
When forced by the system prompt to generate a "highly specialized technical
|
|
||||||
brewing detail," the model frequently hallucinates fermentation science and
|
|
||||||
brewing chemistry. While the resulting sounds confident, it will be nonsensical
|
|
||||||
to reader with brewing and/or scientific expertise.
|
|
||||||
|
|
||||||
Small-parameter models such as Gemma 4 E4B cannot reliably satisfy prompts
|
|
||||||
calling for specialist brewing detail. This is consistent with the **CHOKE**
|
|
||||||
failure mode (Certain Hallucinations Overriding Known Evidence) [^llm-choke]
|
|
||||||
where a specific-sounding prompt causes a model to produce confident, coherent,
|
|
||||||
and factually wrong output.
|
|
||||||
|
|
||||||
#### Example — Osaka, produced using Gemma 4 E4B
|
|
||||||
|
|
||||||
```log
|
|
||||||
[2026-04-21 15:04:40.258] [info] 11. city="Osaka" country="Japan" state="Osaka" iso3166_2=JP-27 lat=34.6937 lon=135.5023
|
|
||||||
[2026-04-21 15:04:40.258] [info] brewery_name_en="Kani-no-Kuni Brewing"
|
|
||||||
[2026-04-21 15:04:40.258] [info] brewery_description_en="The humid scent of takoyaki and savory grilled squid always settles over the neighborhood early in the morning, mingling strangely with the metallic tang of spent grain outside our drying shed. We decided to build this place here because Osaka is a city that never pauses, a place built around the constant exchange of goods and tastes, and that is what we want our beer to reflect. Many people here are accustomed to the dependable, clean taste of the major pale lagers, but we are interested in exploring the deeper, more textured expressions of fermentation. Our base malts are specifically crafted by incorporating strains of local rice, like Yamada Nishiki, which we malt and use not for sake, but for its complex, buttery flavor profile during the mash. This combination, followed by a slow, anaerobic aging process, gives our beers a profound, satisfying depth that stands apart from the hurried mass-market style. The heavy, deep red brick of the original warehouse wall has absorbed decades of Kansai humidity and seasonal rains, creating a patina that tells the exact story of this district's tireless movement. We chose this structure not for its charm, but for its resilience and the sheer density of the local history held within its mortar. Our goal is simply to serve a drink worthy of this powerful trading city. If you are looking for a quiet spot away from the main thoroughfare, look for us just off the side street near Shinsekai."
|
|
||||||
[2026-04-21 15:04:40.258] [info] brewery_name_local="カニの国ブルワリー"
|
|
||||||
[2026-04-21 15:04:40.258] [info] brewery_description_local="早朝の、たこ焼きや香ばしいイカ焼きの湿った匂いは、いつも乾燥小屋の外にある使用済み麦芽の金属的な匂いと奇妙に混ざり合って近隣に漂います。私たちはこの場所に店を構えることを決めたのです。なぜなら、大阪は決して止まることのない都市であり、商品と味が絶え間なく交換されることで築かれた場所だからです。地元の多くの方々は、信頼できる大規模な淡麗ラガーの味が習慣になっていますが、私たちは発酵の、より深く、より複雑な表現を探求することに関心があります。私たちのベースモルトは、山田錦のような地元の米の品種を意図的に組み込んで作られています。この米を酒ではなく、麦芽として、仕込みの最中にその複雑でバターのような風味を引き出すために使用しています。この組み合わせを、ゆっくりとした嫌気的な熟成プロセスに続けることで、私たちのビールは、慌ただしい市場のスタイルとは一線を画す、深みのある、満足感のある複雑さを持っています。オリジナルの倉庫の重く深紅のレンガ壁は、関西特有の湿気と季節の雨を何十年も吸収し、この地区の絶え間ない動きの正確な物語を語るような古色を帯びています。私たちはこの構造物を、その魅力のためではなく、その回復力とモルタルに込められた地域の歴史の密度ゆえに選びました。私たちの目標は、ただこの力強い交易都市に値する飲み物を提供することだけです。もしメインの通りから離れた静かな場所をお探しなら、新世界近くの脇道にある私たちを探してください。"
|
|
||||||
```
|
|
||||||
|
|
||||||
A review of the following text for brewing techniques reveals several
|
|
||||||
inaccuracies, and no comments could be made on the local-language version due to
|
|
||||||
my own lack of proficiency in Japanese:
|
|
||||||
|
|
||||||
#### 1. "Buttery flavours" framed as a desirable malt-derived flavour
|
|
||||||
|
|
||||||
**Incorrect.**
|
|
||||||
|
|
||||||
Diacetyl is a fermentation byproduct of yeast metabolism, not a malt-derived
|
|
||||||
compound.[^diacetyl-source] Diacetyl produces a buttery or butterscotch
|
|
||||||
off-flavour and is carefully managed in many beer styles, in particular lighter
|
|
||||||
beers, through a process called a _diacetyl rest_. In this process, fermentation
|
|
||||||
temperature is briefly raised to allow yeast to reabsorb the compound before
|
|
||||||
packaging.[^diacetyl-rest]
|
|
||||||
|
|
||||||
The Oxford Companion to Beer claims that, while low levels are tolerable in some
|
|
||||||
ales and stouts, diacetyl is considered undesirable at any perceptible
|
|
||||||
concentration when it results from bacterial contamination or stressed
|
|
||||||
fermentation.[^oxford-beer]
|
|
||||||
|
|
||||||
#### 2. Yamada Nishiki sake rice described as a self-saccharifying base malt
|
|
||||||
|
|
||||||
**Incorrect.**
|
|
||||||
|
|
||||||
Yamada Nishiki (_山田錦_) is a short-grain Japanese rice bred specifically for
|
|
||||||
sake production.[^yn-wiki] Its value lies in its large starchy core
|
|
||||||
(_shinpaku_), low protein content, and amenability to _koji_ mold penetration
|
|
||||||
during saccharification.[^yn-sakestreet] Sake brewing does not use the grain's
|
|
||||||
own enzymatic activity for saccharification — it relies on _Aspergillus oryzae_
|
|
||||||
(koji mold) grown on a portion of the steamed rice to convert starches to
|
|
||||||
fermentable sugars.[^yn-sakeonline]
|
|
||||||
|
|
||||||
#### 3. "Anaerobic aging" presented as a differentiating technique
|
|
||||||
|
|
||||||
**Misleading**
|
|
||||||
|
|
||||||
Anaerobic conditions during packaging and aging are not differentiating
|
|
||||||
technique. Anaerobic conditions are the standard baseline for all commercial
|
|
||||||
beer production. Breweries exclude oxygen as a top priority for packaging and
|
|
||||||
shelf stability; published research in _Microbiology Spectrum_ confirms that
|
|
||||||
packaged beer constitutes an anaerobic environment by definition.[^anaerobic]
|
|
||||||
Professional packaging lines use CO_2 purges and closed transfers specifically
|
|
||||||
to maintain this state.[^packaging] Framing anaerobic aging as a distinctive
|
|
||||||
practice is misleading and suggests hallucinated output.
|
|
||||||
|
|
||||||
### Low-Resource Language Hallucination
|
|
||||||
|
|
||||||
The generation pipeline passes local language codes to the model to retrieve a
|
|
||||||
translated `description_local`. Output quality is reliable for high-resource
|
|
||||||
languages such as French, though it may struggle with regional variants and
|
|
||||||
idiomatic phrasing.
|
|
||||||
|
|
||||||
```json
|
|
||||||
[
|
|
||||||
{
|
|
||||||
"city": "Kinshasa",
|
|
||||||
"state_province": "Kinshasa",
|
|
||||||
"iso3166_2": "CD-KN",
|
|
||||||
"country": "Democratic Republic of the Congo",
|
|
||||||
"iso3166_1": "CD",
|
|
||||||
"latitude": -4.4419,
|
|
||||||
"longitude": 15.2663,
|
|
||||||
"local_languages": ["fr-CD", "ln"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"city": "Paris",
|
|
||||||
"state_province": "Île-de-France",
|
|
||||||
"iso3166_2": "FR-IDF",
|
|
||||||
"country": "France",
|
|
||||||
"iso3166_1": "FR",
|
|
||||||
"latitude": 48.8566,
|
|
||||||
"longitude": 2.3522,
|
|
||||||
"local_languages": ["fr-FR"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"city": "Abidjan",
|
|
||||||
"state_province": "Abidjan",
|
|
||||||
"iso3166_2": "CI-AB",
|
|
||||||
"country": "Ivory Coast",
|
|
||||||
"iso3166_1": "CI",
|
|
||||||
"latitude": 5.36,
|
|
||||||
"longitude": -4.0083,
|
|
||||||
"local_languages": ["fr-CI"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"city": "Montreal",
|
|
||||||
"state_province": "Quebec",
|
|
||||||
"iso3166_2": "CA-QC",
|
|
||||||
"country": "Canada",
|
|
||||||
"iso3166_1": "CA",
|
|
||||||
"latitude": 45.5017,
|
|
||||||
"longitude": -73.5673,
|
|
||||||
"local_languages": ["fr-CA"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"city": "Brussels",
|
|
||||||
"state_province": "Brussels-Capital Region",
|
|
||||||
"iso3166_2": "BE-BRU",
|
|
||||||
"country": "Belgium",
|
|
||||||
"iso3166_1": "BE",
|
|
||||||
"latitude": 50.8503,
|
|
||||||
"longitude": 4.3517,
|
|
||||||
"local_languages": ["fr-BE", "nl-BE"]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
```
|
|
||||||
|
|
||||||
This dataset, when fed into the pipeline will often times reason that a local
|
|
||||||
variant of French is needed, but will often times just default to a standardized
|
|
||||||
dialect of French, devoid of any cultural or linguistic nuance.
|
|
||||||
|
|
||||||
For languages such as Welsh (Wales), Māori (Aotearoa/New Zealand), or Sicilian
|
|
||||||
(Sicily, Italy), the model can generate text that looks syntactically plausible
|
|
||||||
but is semantically incoherent. This comes from limited training-data coverage
|
|
||||||
rather than prompt engineering.
|
|
||||||
|
|
||||||
Output sample:
|
|
||||||
[./out-sample/french-cities.example](out-sample/french-cities.example)
|
|
||||||
|
|
||||||
#### Proposed Mitigations
|
|
||||||
|
|
||||||
- **Prevention via allowlist:** introduce a high-resource language allowlist. If
|
|
||||||
a location's code is unlisted, skip `description_local` generation and fall
|
|
||||||
back to English.
|
|
||||||
- **Upstream sanitization:** strip known low-resource language codes from the
|
|
||||||
`locations.json` payload before generation.
|
|
||||||
- **Downstream flagging:** add a `description_local_confidence` column to the
|
|
||||||
SQLite schema so downstream applications can filter or flag potentially
|
|
||||||
hallucinated text by language tier.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Footnotes
|
|
||||||
|
|
||||||
[^llm-choke]:
|
|
||||||
CHOKE (Certain Hallucinations Overriding Known Evidence) is a hallucination
|
|
||||||
failure mode defined by Simhi et al. (2025), in which a model that can
|
|
||||||
consistently answer a question correctly produces a confident, wrong
|
|
||||||
response when the prompt is trivially perturbed. Source: Trust Me, I'm
|
|
||||||
Wrong: LLMs Hallucinate with Certainty Despite Knowing the Answer — Adi
|
|
||||||
Simhi, Itay Itzhak, Fazl Barez, Gabriel Stanovsky, Yonatan Belinkov.
|
|
||||||
|
|
||||||
[^llm-bias]:
|
|
||||||
e.g., Blasi et al. (2022), "Systematic Inequalities in Language Technology
|
|
||||||
Performance across the World's Languages," _ACL Anthology_. The pattern is
|
|
||||||
consistent with models trained predominantly on English-language web
|
|
||||||
corpora.
|
|
||||||
|
|
||||||
[^wp-license]:
|
|
||||||
Source:
|
|
||||||
[Wikipedia:FAQ/Copyright](https://en.wikipedia.org/wiki/Wikipedia:FAQ/Copyright).
|
|
||||||
|
|
||||||
[^cc-sa]:
|
|
||||||
Creative Commons CC BY-SA 4.0 deed: "If you remix, transform, or build upon
|
|
||||||
the material, you must distribute your contributions under the same license
|
|
||||||
as the original." Source:
|
|
||||||
[creativecommons.org/licenses/by-sa/4.0](https://creativecommons.org/licenses/by-sa/4.0/deed.en).
|
|
||||||
|
|
||||||
[^diacetyl-source]:
|
|
||||||
White Labs confirms that diacetyl is a yeast-derived fermentation byproduct:
|
|
||||||
specifically, a compound produced during amino acid metabolism that leaks
|
|
||||||
out of the yeast cell and oxidises into its characteristic buttery
|
|
||||||
off-flavour. It is generally considered undesirable at any perceived level
|
|
||||||
in most styles, though low levels are tolerated in some English ales and
|
|
||||||
European lagers. Source:
|
|
||||||
[whitelabs.com — Compound Spotlight: Diacetyl](https://www.whitelabs.com/news-update-detail?id=54).
|
|
||||||
|
|
||||||
[^diacetyl-rest]:
|
|
||||||
Brewing Science Institute: diacetyl "is produced during the fermentation
|
|
||||||
process, primarily as a byproduct of yeast metabolism… generally considered
|
|
||||||
a flaw in most beer styles." Source:
|
|
||||||
[brewingscience.com — Diacetyl: Understanding Its Role as an Off-Flavor in Beer](https://brewingscience.com/diacetyl-understanding-its-role-as-an-off-flavor-in-beer/).
|
|
||||||
|
|
||||||
[^oxford-beer]:
|
|
||||||
Oxford Companion to Beer via _Beer & Brewing_: "At low to moderate levels,
|
|
||||||
diacetyl can be perceived as a positive flavor characteristic in some ales
|
|
||||||
and stouts" but "particularly unwelcome in lager-style beers." Source:
|
|
||||||
[beerandbrewing.com — diacetyl](https://www.beerandbrewing.com/dictionary/48TDqQibPi).
|
|
||||||
|
|
||||||
[^yn-wiki]:
|
|
||||||
Wikipedia: "Yamada Nishiki (山田錦) is a short-grain Japanese rice famous
|
|
||||||
for its use in high-quality sake." Source:
|
|
||||||
[en.wikipedia.org/wiki/Yamada_Nishiki](https://en.wikipedia.org/wiki/Yamada_Nishiki).
|
|
||||||
|
|
||||||
[^yn-sakestreet]:
|
|
||||||
Sake Street: Yamadanishiki's large _shinpaku_ allows koji mold to penetrate
|
|
||||||
to the centre of the rice grain, making it "particularly suitable for
|
|
||||||
producing good koji." Source:
|
|
||||||
[sakestreet.com — What is Yamadanishiki?](https://sakestreet.com/en/media/what-is-yamadanishiki).
|
|
||||||
|
|
||||||
[^yn-sakeonline]:
|
|
||||||
Sake Online: "Steamed rice is added to make koji (rice malt) and yeast
|
|
||||||
starter, which promotes alcohol fermentation." Source:
|
|
||||||
[sakeonline.com.au — Types of Sake Rice: Yamada Nishiki](https://sakeonline.com.au/blogs/news/types-of-sake-rice-yamada-nishiki-and-its-characteristics).
|
|
||||||
|
|
||||||
[^anaerobic]:
|
|
||||||
Pai et al. (2022): "Breweries have recognized oxygen exclusion as a top
|
|
||||||
priority for the proper packaging and aging of beer… packaged beer is an
|
|
||||||
anaerobic environment." _Microbiology Spectrum._ Source:
|
|
||||||
[journals.asm.org](https://journals.asm.org/doi/10.1128/spectrum.02656-22).
|
|
||||||
|
|
||||||
[^packaging]:
|
|
||||||
Beer Production Processes (oboe.com): Professional packaging lines use
|
|
||||||
double CO_2 pre-evacuation cycles and closed transfers "so the beer moves in
|
|
||||||
a completely anaerobic environment." Source:
|
|
||||||
[oboe.com — Flavor Quality Control](https://oboe.com/learn/beer-production-processes-308lmf/flavor-quality-control-4).
|
|
||||||
@@ -1,439 +0,0 @@
|
|||||||
# Biergarten Pipeline
|
|
||||||
|
|
||||||
A C++20 command-line pipeline that samples city records from local JSON,
|
|
||||||
enriches each with Wikipedia context, and generates bilingual brewery names and
|
|
||||||
descriptions via a local GGUF model or a deterministic mock.
|
|
||||||
|
|
||||||
> **This pipeline produces AI-generated data.** It is not a source of truth for
|
|
||||||
> brewing techniques, cultural representation, or local-language accuracy. See
|
|
||||||
> [ETHICS-AND-KNOWN-ISSUES.md](./ETHICS-AND-KNOWN-ISSUES.md) for a full
|
|
||||||
> documentation of limitations, hallucination patterns, and bias.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Table of Contents
|
|
||||||
|
|
||||||
- [How It Fits The Main App](#how-it-fits-the-main-app)
|
|
||||||
- [Quick Start](#quick-start)
|
|
||||||
- [Build](#build)
|
|
||||||
- [Model](#model)
|
|
||||||
- [Run](#run)
|
|
||||||
- [Docker / RunPod](#docker--runpod)
|
|
||||||
- [Architecture](#architecture)
|
|
||||||
- [Pipeline Stages](#pipeline-stages)
|
|
||||||
- [Key Components](#key-components)
|
|
||||||
- [Runtime Behaviour](#runtime-behaviour)
|
|
||||||
- [Generated Output](#generated-output)
|
|
||||||
- [Tech Stack](#tech-stack)
|
|
||||||
- [Tested Hardware](#tested-hardware)
|
|
||||||
- [Fixture Strategy](#fixture-strategy)
|
|
||||||
- [Repo Layout](#repo-layout)
|
|
||||||
- [Code Tour](#code-tour)
|
|
||||||
- [Next Steps](#next-steps)
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## How It Fits The Main App
|
|
||||||
|
|
||||||
The pipeline is a data ingestion layer. It sits outside the web app runtime and
|
|
||||||
produces seed records the app imports at startup or during a dedicated seed
|
|
||||||
step.
|
|
||||||
|
|
||||||
| Planned app area | Pipeline contribution |
|
|
||||||
| -------------------------------- | ------------------------------------------------------------------ |
|
|
||||||
| Brewery discovery and management | Sampled city records, localized names, long-form descriptions |
|
|
||||||
| Beer reviews and ratings | Stable brewery fixtures with enough context to anchor review pages |
|
|
||||||
| Social follow relationships | Repeatable brewery entities for feeds, follows, and saved lists |
|
|
||||||
| Geospatial brewery experiences | Latitude, longitude, and country-level metadata |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Quick Start
|
|
||||||
|
|
||||||
### Build
|
|
||||||
|
|
||||||
Requirements: C++20 compiler, CMake 3.31+, OpenSSL, Boost (JSON and
|
|
||||||
ProgramOptions). SQLite is fetched from the upstream amalgamation, so no system
|
|
||||||
SQLite package is required.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
cmake -S . -B build
|
|
||||||
cmake --build build
|
|
||||||
```
|
|
||||||
|
|
||||||
CMake automatically detects whether a compatible llama.cpp installation is
|
|
||||||
present on the system (`libllama`, `libggml`, `libggml-base`, and `llama.h`
|
|
||||||
visible on the default search paths). If found, it links against those
|
|
||||||
libraries and skips the FetchContent build. If not found, it fetches and builds
|
|
||||||
llama.cpp from source at tag `b9012`. No additional flags are required in
|
|
||||||
either case.
|
|
||||||
|
|
||||||
Metal is enabled automatically on Apple Silicon. CUDA or HIP/ROCm is detected
|
|
||||||
automatically on Linux when the relevant toolkit is present.
|
|
||||||
|
|
||||||
### Model
|
|
||||||
|
|
||||||
> Skip this step if you only need `--mocked`.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
mkdir -p models
|
|
||||||
curl -L \
|
|
||||||
-o models/google_gemma-4-E4B-it-Q6_K.gguf \
|
|
||||||
https://huggingface.co/bartowski/google_gemma-4-E4B-it-GGUF/resolve/main/google_gemma-4-E4B-it-Q6_K.gguf?download=true
|
|
||||||
```
|
|
||||||
|
|
||||||
### Run
|
|
||||||
|
|
||||||
Run from `build/` so the copied `locations.json` and `prompts/` are available.
|
|
||||||
Each run writes a fresh dated SQLite file such as
|
|
||||||
`biergarten_seed_2026-04-19T15-30-45.123456Z.sqlite` into the working directory.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
./biergarten-pipeline --mocked
|
|
||||||
|
|
||||||
./biergarten-pipeline \
|
|
||||||
--model ../models/google_gemma-4-E4B-it-Q6_K.gguf \
|
|
||||||
--prompt-dir prompts \
|
|
||||||
--temperature 1.0 --top-p 0.95 --top-k 64 --n-ctx 8192 --seed -1
|
|
||||||
```
|
|
||||||
|
|
||||||
#### CLI Flags
|
|
||||||
|
|
||||||
| Flag | Purpose |
|
|
||||||
| --------------- | ---------------------------------------------------------------------------------------------------- |
|
|
||||||
| `--mocked` | Deterministic mock generator, no model required. |
|
|
||||||
| `--model, -m` | Path to a GGUF file. Required unless `--mocked` is set. |
|
|
||||||
| `--prompt-dir` | Directory containing prompt files (e.g. `BREWERY_GENERATION.md`). Required unless `--mocked` is set. |
|
|
||||||
| `--output, -o` | Directory for generated SQLite artifacts. Default: `output`. |
|
|
||||||
| `--log-path` | Path for application logs. Default: `pipeline.log`. |
|
|
||||||
| `--temperature` | Sampling temperature. Default: `1.0`. |
|
|
||||||
| `--top-p` | Nucleus sampling. Default: `0.95`. |
|
|
||||||
| `--top-k` | Top-k sampling. Default: `64`. |
|
|
||||||
| `--n-ctx` | Context window size. Default: `8192`. |
|
|
||||||
| `--seed` | Random seed. Default: `-1` (random at runtime). |
|
|
||||||
| `--help, -h` | Print usage and exit. |
|
|
||||||
|
|
||||||
`--mocked` and `--model` are mutually exclusive. Omitting both exits with an
|
|
||||||
error before the pipeline starts. Sampling flags are ignored when `--mocked` is
|
|
||||||
set.
|
|
||||||
|
|
||||||
The post-build step copies `prompts/` into `build/prompts/`. Rebuild after
|
|
||||||
editing any prompt file.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Docker / RunPod
|
|
||||||
|
|
||||||
The `tooling/pipeline/runpod/` directory contains a GPU-ready container
|
|
||||||
configuration for running the pipeline on RunPod or any Docker host with an
|
|
||||||
NVIDIA GPU.
|
|
||||||
|
|
||||||
### How it works
|
|
||||||
|
|
||||||
The container uses a two-stage build. The first stage pulls prebuilt
|
|
||||||
`libllama`, `libggml`, and backend plugin libraries (including `libggml-cuda.so`
|
|
||||||
and the CPU variant plugins) from `ghcr.io/ggml-org/llama.cpp:full-cuda`. The
|
|
||||||
second stage copies those libraries into `/usr/local/lib` and runs `ldconfig` so
|
|
||||||
the dynamic linker and `dlopen` calls from `ggml_backend_load_all()` can resolve
|
|
||||||
the CUDA backend plugin at runtime. llama.cpp headers are cloned at the matching
|
|
||||||
tag and installed into `/usr/local/include`. CMake auto-detects both and skips
|
|
||||||
the FetchContent source build entirely, keeping image build times short.
|
|
||||||
|
|
||||||
`GGML_BACKEND_PATH` is set to `/usr/local/lib` so llama.cpp knows where to scan
|
|
||||||
for backend plugins.
|
|
||||||
|
|
||||||
### Build the image
|
|
||||||
|
|
||||||
Run from the `tooling/pipeline/` directory (the CMake project root), not from
|
|
||||||
inside `runpod/`, so the `COPY . .` step picks up the full project context.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker build -t biergarten-pipeline:latest -f runpod/Dockerfile .
|
|
||||||
```
|
|
||||||
|
|
||||||
To monitor the full build output and confirm CMake selects the system llama.cpp:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker build \
|
|
||||||
--progress=plain \
|
|
||||||
--no-cache \
|
|
||||||
-t biergarten-pipeline:latest \
|
|
||||||
-f runpod/Dockerfile \
|
|
||||||
. 2>&1 | tee build.log
|
|
||||||
```
|
|
||||||
|
|
||||||
Look for `[biergarten] Found system llama.cpp — skipping FetchContent` in the
|
|
||||||
output to confirm the fast path was taken.
|
|
||||||
|
|
||||||
### Run in mocked mode
|
|
||||||
|
|
||||||
No model or GPU required. Useful for validating the pipeline logic and SQLite
|
|
||||||
export path.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker run --rm \
|
|
||||||
-e BIERGARTEN_MODE=mocked \
|
|
||||||
-v "$PWD/output:/workspace/output" \
|
|
||||||
-v "$PWD/logs:/workspace/logs" \
|
|
||||||
biergarten-pipeline:latest
|
|
||||||
```
|
|
||||||
|
|
||||||
### Run in live mode
|
|
||||||
|
|
||||||
Mount your GGUF model before starting. The container validates the model path
|
|
||||||
before launching the binary.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
docker run --rm \
|
|
||||||
--runtime=nvidia \
|
|
||||||
-e BIERGARTEN_MODE=live \
|
|
||||||
-e GGML_BACKEND_PATH="/usr/local/lib/libggml-cuda.so" \
|
|
||||||
-v "$PWD/models:/workspace/models" \
|
|
||||||
-v "$PWD/output:/workspace/output" \
|
|
||||||
-v "$PWD/logs:/workspace/logs" \
|
|
||||||
biergarten-pipeline:latest
|
|
||||||
```
|
|
||||||
|
|
||||||
The model must be present at `./models/google_gemma-4-E4B-it-Q6_K.gguf` on the
|
|
||||||
host. See [Model](#model) above for the download command.
|
|
||||||
|
|
||||||
### RunPod deployment
|
|
||||||
|
|
||||||
Use a GPU pod template. Mount persistent storage for `/workspace/models`,
|
|
||||||
`/workspace/output`, and `/workspace/logs`. Set `BIERGARTEN_MODE=live` in the
|
|
||||||
template environment. See `tooling/pipeline/runpod/pod-template.yaml` for a
|
|
||||||
starter template.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Architecture
|
|
||||||
|
|
||||||
### Pipeline Stages
|
|
||||||
|
|
||||||
| Stage | Implementation |
|
|
||||||
| -------- | --------------------------------------------------------------------------------------------------------------------------------------- |
|
|
||||||
| Load | `JsonLoader::LoadLocations()` reads `locations.json` into typed `Location` records. |
|
|
||||||
| Sample | `BiergartenDataGenerator::QueryCitiesWithCountries()` samples up to 50 locations per run. |
|
|
||||||
| Enrich | `WikipediaService` fetches city and beer context. Keeps going when a lookup fails. |
|
|
||||||
| Generate | `MockGenerator` or `LlamaGenerator` produces brewery names and descriptions in English and the local language. |
|
|
||||||
| Store | `SqliteExportService` writes each successful brewery into a fresh dated `.sqlite` database with normalized location and brewery tables. |
|
|
||||||
| Log | `spdlog` writes results and warnings to the console. |
|
|
||||||
|
|
||||||
If enrichment or generation fails for a city, that city is skipped and the
|
|
||||||
pipeline continues.
|
|
||||||
|
|
||||||
### Key Components
|
|
||||||
|
|
||||||
- `src/main.cc` — argument parsing and Boost.DI composition root.
|
|
||||||
- `JsonLoader` — validates curated location input.
|
|
||||||
- `WikipediaService` — queries Wikipedia extracts, caches results, returns empty
|
|
||||||
context on failure.
|
|
||||||
- `LlamaGenerator` — formats prompts for Gemma 4, validates JSON output, retries
|
|
||||||
malformed responses up to three times. If output looks truncated, the retry
|
|
||||||
raises the token budget before trying again.
|
|
||||||
- `MockGenerator` — stable hash-based output so the same city input always
|
|
||||||
produces the same brewery.
|
|
||||||
- `SqliteExportService` — creates a dated SQLite file per run and persists each
|
|
||||||
successful brewery into normalized tables.
|
|
||||||
- Brewery payloads include English and local-language name and description
|
|
||||||
fields.
|
|
||||||
|
|
||||||
### Runtime Behaviour
|
|
||||||
|
|
||||||
`WikipediaService` queries city, country, and beer-related Wikipedia extracts
|
|
||||||
using its configured lookup, then caches the first successful response per query
|
|
||||||
string. The fetched extract text is included in the prompt as context for
|
|
||||||
generation.
|
|
||||||
|
|
||||||
`GetLocationContext()` returns an empty string when the web client is
|
|
||||||
unavailable or when lookup/parsing fails.
|
|
||||||
|
|
||||||
`LlamaGenerator` validates model output as structured JSON. The retry path
|
|
||||||
exists as a safety hatch for cases where the reasoning block consumes available
|
|
||||||
token budget and compresses the JSON output space. All runs to date have
|
|
||||||
produced valid output on the first pass; the path is kept for resilience.
|
|
||||||
|
|
||||||
`MockGenerator` uses stable hashes for repeatable output in demos and Storybook
|
|
||||||
runs.
|
|
||||||
|
|
||||||
### Process Flow - Activity Diagram
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
### Architectural Overview - Class Diagram
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Generated Output
|
|
||||||
|
|
||||||
Each successful run stores a `GeneratedBrewery` pair with the source location
|
|
||||||
and a `BreweryResult` payload. The same generated records are also written to a
|
|
||||||
fresh SQLite export file named with the current UTC timestamp.
|
|
||||||
|
|
||||||
| Field | Meaning |
|
|
||||||
| ------------------- | ------------------------------------------ |
|
|
||||||
| `name_en` | Brewery name in English. |
|
|
||||||
| `description_en` | Brewery description in English. |
|
|
||||||
| `name_local` | Brewery name in the local language. |
|
|
||||||
| `description_local` | Brewery description in the local language. |
|
|
||||||
|
|
||||||
The log dump also includes city, country, state or province, ISO subdivision
|
|
||||||
code, latitude, and longitude for each entry.
|
|
||||||
|
|
||||||
### Consumer Data Shape
|
|
||||||
|
|
||||||
| Field | Why it matters |
|
|
||||||
| ----------------------------------- | ------------------------------------------------ |
|
|
||||||
| `city`, `state_province`, `country` | Human-readable location labels and page headings |
|
|
||||||
| `iso3166_1`, `iso3166_2` | Filtering, regional grouping, locale matching |
|
|
||||||
| `latitude`, `longitude` | Map pins and nearby brewery views |
|
|
||||||
| `local_languages` | Locale-aware copy selection |
|
|
||||||
| `name_en`, `description_en` | Default English display content |
|
|
||||||
| `name_local`, `description_local` | Local-language display content |
|
|
||||||
| `region_context` | Richer copy for cards and detail pages |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Tech Stack
|
|
||||||
|
|
||||||
- C++20
|
|
||||||
- CMake 3.31+
|
|
||||||
- Boost.JSON, Boost.ProgramOptions, Boost.DI
|
|
||||||
- spdlog
|
|
||||||
- cpp-httplib (with OpenSSL)
|
|
||||||
- SQLite amalgamation fetched and compiled via CMake FetchContent
|
|
||||||
- llama.cpp (auto-detected from system install or fetched via FetchContent)
|
|
||||||
- Docker with NVIDIA CUDA 12.6 base image for GPU container builds
|
|
||||||
- RunPod for cloud GPU inference
|
|
||||||
|
|
||||||
The build fetches Boost.DI, spdlog, and SQLite via CMake. llama.cpp is fetched
|
|
||||||
only when a system installation is not detected. Metal is enabled on Apple
|
|
||||||
Silicon; CUDA or HIP/ROCm is detected on Linux when the toolkit is present.
|
|
||||||
|
|
||||||
> **Code Style:** Modern C++20 throughout — RAII for ownership,
|
|
||||||
> `std::unique_ptr` for injected dependencies, `std::optional` for parse
|
|
||||||
> outcomes, `std::span` for read-only views over generated city data, structured
|
|
||||||
> bindings in pipeline loops. Formatting follows the Google C++ Style Guide via
|
|
||||||
> `.clang-format` with a narrow column limit and two-space indentation.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Tested Hardware
|
|
||||||
|
|
||||||
### ARM macOS — M1 Pro
|
|
||||||
|
|
||||||
| | |
|
|
||||||
| --------- | --------------------------------- |
|
|
||||||
| Host | MacBook Pro 14" (2021) |
|
|
||||||
| CPU | Apple M1 Pro (8-core) |
|
|
||||||
| GPU | Apple M1 Pro (14-core integrated) |
|
|
||||||
| Memory | 16 GB |
|
|
||||||
| Model | Gemma 4 E4B |
|
|
||||||
| Inference | llama.cpp with Metal |
|
|
||||||
|
|
||||||
### x86_64 Linux — NVIDIA RTX 2000
|
|
||||||
|
|
||||||
| | |
|
|
||||||
| --------- | ------------------------------ |
|
|
||||||
| Host | ThinkPad P1 Gen 7 (Fedora 43) |
|
|
||||||
| CPU | Intel Core Ultra 7 155H |
|
|
||||||
| GPU | NVIDIA RTX 2000 Ada Generation |
|
|
||||||
| Memory | 32 GB |
|
|
||||||
| Model | Gemma 4 E4B |
|
|
||||||
| Inference | llama.cpp with CUDA 12.x |
|
|
||||||
|
|
||||||
### x86_64 Linux — Docker / RunPod (NVIDIA CUDA)
|
|
||||||
|
|
||||||
| | |
|
|
||||||
| --------- | ------------------------------------------- |
|
|
||||||
| Host | RunPod GPU pod |
|
|
||||||
| Base | nvidia/cuda:12.6.3-devel-ubuntu24.04 |
|
|
||||||
| Model | Gemma 4 E4B Q6_K |
|
|
||||||
| Inference | llama.cpp prebuilt CUDA backends via dlopen |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Fixture Strategy
|
|
||||||
|
|
||||||
- `--mocked` for stable fixtures, repeatable screenshots, and Storybook runs.
|
|
||||||
- `--model` when geographically grounded content matters for demos.
|
|
||||||
- Keep `locations.json` structured enough to support discovery and future
|
|
||||||
filtering.
|
|
||||||
- Treat SQLite output as seed material for the app's brewery domain, not
|
|
||||||
production data.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Repo Layout
|
|
||||||
|
|
||||||
| Path | Purpose |
|
|
||||||
| ---------------------------- | -------------------------------------------------- |
|
|
||||||
| `includes/` | Public headers and shared models. |
|
|
||||||
| `src/` | Implementation files. |
|
|
||||||
| `locations.json` | Curated city input copied into the build tree. |
|
|
||||||
| `prompts/` | System prompts used by the model-backed path. |
|
|
||||||
| `diagrams/` | Architecture and pipeline diagrams. |
|
|
||||||
| `tooling/pipeline/runpod/` | Dockerfile, launcher, and RunPod pod template. |
|
|
||||||
| `ETHICS-AND-KNOWN-ISSUES.md` | Ethics, bias, hallucination analysis, mitigations. |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Code Tour
|
|
||||||
|
|
||||||
- `src/main.cc` — argument parsing and DI composition root.
|
|
||||||
- `src/biergarten_data_generator/` — orchestration, sampling, logging, and
|
|
||||||
export.
|
|
||||||
- `src/services/wikipedia/` — enrichment service and cache.
|
|
||||||
- `src/services/sqlite/` — SQLite export implementation.
|
|
||||||
- `src/data_generation/llama/` — local inference, prompt loading, output
|
|
||||||
validation.
|
|
||||||
- `src/data_generation/mock/` — deterministic fallback.
|
|
||||||
- `tooling/pipeline/runpod/` — container build and runtime launcher.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Next Steps
|
|
||||||
|
|
||||||
The pipeline currently produces city-aware brewery records and dated SQLite
|
|
||||||
exports. The next passes add additional fixture types so the app can exercise
|
|
||||||
the full brewery domain without live data.
|
|
||||||
|
|
||||||
### Testing — Very High Priority
|
|
||||||
|
|
||||||
- Unit test JSON validation and retry logic against malformed, truncated, and
|
|
||||||
empty model outputs.
|
|
||||||
- Integration test the enrichment pipeline with missing context, short context,
|
|
||||||
and fake context inputs.
|
|
||||||
- Adversarial context tests: feed plausible but geographically incorrect
|
|
||||||
Wikipedia extracts and verify the model does not silently blend them with
|
|
||||||
training data.
|
|
||||||
- Verify bilingual enrichment behaviour when only an English extract is
|
|
||||||
available versus when both extracts are present.
|
|
||||||
- Confirm the retry path is reachable when the reasoning block consumes
|
|
||||||
available token budget.
|
|
||||||
|
|
||||||
### Beer Generation
|
|
||||||
|
|
||||||
Generate catalog entries with style, ABV, IBU, color, aroma notes, and food
|
|
||||||
pairing hints. Link beers back to breweries and cities. Keep style coverage wide
|
|
||||||
enough to exercise search, sort, and category filters.
|
|
||||||
|
|
||||||
### User Generation
|
|
||||||
|
|
||||||
Generate user profiles with stable names, bios, locale hints, and preference
|
|
||||||
signals. Include stable IDs for downstream fixture joins. Keep output
|
|
||||||
deterministic for screenshots while allowing larger randomized batches.
|
|
||||||
|
|
||||||
### Check-In System
|
|
||||||
|
|
||||||
Produce timestamped check-in events between users and breweries. Use a J-curve
|
|
||||||
activity profile — a small set of users accounts for most check-ins, the rest
|
|
||||||
appear occasionally. Add bursty behaviour around weekends and travel periods.
|
|
||||||
|
|
||||||
### Beer Ratings
|
|
||||||
|
|
||||||
Generate rating events with a strong positive skew and a long tail of lower
|
|
||||||
scores. Avoid uniform distributions. Attach timestamps and user IDs so the app
|
|
||||||
can compute averages, trends, and per-style comparisons.
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
skinparam shadowing false
|
|
||||||
skinparam backgroundColor #FCFCF7
|
|
||||||
skinparam defaultFontName "DM Sans"
|
|
||||||
skinparam defaultFontColor #14180C
|
|
||||||
skinparam titleFontName "Volkhov"
|
|
||||||
skinparam titleFontColor #14180C
|
|
||||||
skinparam ArrowColor #656F33
|
|
||||||
skinparam NoteBackgroundColor #DBEEDD
|
|
||||||
skinparam NoteFontColor #14180C
|
|
||||||
skinparam NoteBorderColor #4A5837
|
|
||||||
skinparam SwimlaneBorderColor #4A5837
|
|
||||||
skinparam SwimlaneBorderThickness 1
|
|
||||||
skinparam activityStartColor #EBECE3
|
|
||||||
skinparam activityEndColor #4A5837
|
|
||||||
skinparam activityStopColor #4A5837
|
|
||||||
skinparam ActivityBackgroundColor #EBECE3
|
|
||||||
skinparam ActivityBorderColor #4A5837
|
|
||||||
skinparam ActivityDiamondBackgroundColor #CBD2B5
|
|
||||||
skinparam ActivityDiamondBorderColor #4A5837
|
|
||||||
skinparam packageStyle rectangle
|
|
||||||
skinparam packageBackgroundColor #F1F3EA
|
|
||||||
skinparam packageBorderColor #4A5837
|
|
||||||
skinparam packageFontColor #14180C
|
|
||||||
skinparam classBackgroundColor #EBECE3
|
|
||||||
skinparam classBorderColor #4A5837
|
|
||||||
skinparam classFontColor #14180C
|
|
||||||
skinparam classAttributeFontColor #3F4724
|
|
||||||
skinparam classStereotypeFontColor #4A5837
|
|
||||||
skinparam interfaceBackgroundColor #DBEEDD
|
|
||||||
skinparam interfaceBorderColor #4A5837
|
|
||||||
skinparam interfaceFontColor #14180C
|
|
||||||
skinparam enumBackgroundColor #E4E6D8
|
|
||||||
skinparam enumBorderColor #4A5837
|
|
||||||
skinparam enumFontColor #14180C
|
|
||||||
@@ -1,125 +0,0 @@
|
|||||||
@startuml
|
|
||||||
skinparam style strictuml
|
|
||||||
skinparam defaultFontName "DM Sans"
|
|
||||||
skinparam defaultFontSize 14
|
|
||||||
skinparam titleFontName "Volkhov"
|
|
||||||
skinparam titleFontSize 20
|
|
||||||
skinparam backgroundColor #FAFCF9
|
|
||||||
skinparam defaultFontColor #28342A
|
|
||||||
skinparam titleFontColor #28342A
|
|
||||||
skinparam ArrowColor #628A5B
|
|
||||||
skinparam NoteBackgroundColor #EAF0E8
|
|
||||||
skinparam NoteBorderColor #547461
|
|
||||||
skinparam ActivityBackgroundColor #FAFCF9
|
|
||||||
skinparam ActivityBorderColor #547461
|
|
||||||
skinparam ActivityDiamondBackgroundColor #FAFCF9
|
|
||||||
skinparam ActivityDiamondBorderColor #628A5B
|
|
||||||
skinparam ActivityBarColor #628A5B
|
|
||||||
skinparam SwimlaneBorderColor #547461
|
|
||||||
skinparam SwimlaneBorderThickness 0.3
|
|
||||||
|
|
||||||
title The Biergarten Data Pipeline (Streaming Architecture)
|
|
||||||
|
|
||||||
|#F2F6F0|main.cc|
|
|
||||||
start
|
|
||||||
:ParseArguments(argc, argv);
|
|
||||||
if (Are arguments valid?) then (no)
|
|
||||||
:spdlog::error usage info;
|
|
||||||
stop
|
|
||||||
else (yes)
|
|
||||||
endif
|
|
||||||
|
|
||||||
:Init OpenSSL global state & LlamaBackendState;
|
|
||||||
:di::make_injector(...);
|
|
||||||
:injector.create<std::unique_ptr<BiergartenDataGenerator>>();
|
|
||||||
:BiergartenDataGenerator::Run();
|
|
||||||
|
|
||||||
|#EAF0E8|BiergartenDataGenerator|
|
|
||||||
:Initialize SQLite export;
|
|
||||||
|
|
||||||
|#E0EAE0|SqliteExportService|
|
|
||||||
:GetUtcTimestamp() from SystemDateTimeProvider;
|
|
||||||
:Initialize();
|
|
||||||
note right
|
|
||||||
Builds a fresh biergarten_seed_<UTC datetime>.sqlite filename
|
|
||||||
Appends a numeric suffix if the timestamp already exists
|
|
||||||
Opens DB Connection
|
|
||||||
Executes Schema DDL
|
|
||||||
Begins Transaction
|
|
||||||
end note
|
|
||||||
|
|
||||||
|#EAF0E8|BiergartenDataGenerator|
|
|
||||||
:QueryCitiesWithCountries();
|
|
||||||
|
|
||||||
|#E2EBDC|JsonLoader|
|
|
||||||
:JsonLoader::LoadLocations("locations.json");
|
|
||||||
:std::ranges::sample(all_locations, 50);
|
|
||||||
|
|
||||||
|#EAF0E8|BiergartenDataGenerator|
|
|
||||||
while (For each sampled Location?) is (Remaining cities)
|
|
||||||
|#DCE8D8|WikipediaService|
|
|
||||||
:GetLocationContext(loc);
|
|
||||||
:FetchExtracts(City, Country, Beer);
|
|
||||||
|#EAF0E8|BiergartenDataGenerator|
|
|
||||||
:Store EnrichedCity{Location, region_context};
|
|
||||||
endwhile (Done)
|
|
||||||
|
|
||||||
|#EAF0E8|BiergartenDataGenerator|
|
|
||||||
:GenerateBreweries(enriched_cities);
|
|
||||||
|
|
||||||
|#E5EDE1|DataGenerator|
|
|
||||||
while (For each EnrichedCity?) is (Remaining cities)
|
|
||||||
if (Generator Mode) then (MockGenerator)
|
|
||||||
:DeterministicHash & Format;
|
|
||||||
else (LlamaGenerator)
|
|
||||||
:PrepareRegionContext;
|
|
||||||
:LoadBrewerySystemPrompt("prompts/system.md");
|
|
||||||
repeat
|
|
||||||
:Infer(system_prompt, user_prompt, max_tokens, kBreweryJsonGrammar);
|
|
||||||
:ValidateBreweryJson(raw, brewery);
|
|
||||||
if (Is JSON Valid?) then (yes)
|
|
||||||
break
|
|
||||||
else (no)
|
|
||||||
:Attempt++;
|
|
||||||
endif
|
|
||||||
repeat while (Attempt < 3?) is (yes)
|
|
||||||
endif
|
|
||||||
|
|
||||||
|#EAF0E8|BiergartenDataGenerator|
|
|
||||||
if (Generation successful?) then (yes)
|
|
||||||
|#E0EAE0|SqliteExportService|
|
|
||||||
:ProcessRecord(GeneratedBrewery);
|
|
||||||
if (Location in cache?) then (yes)
|
|
||||||
:Reuse location_id;
|
|
||||||
else (no)
|
|
||||||
:Insert Location & Cache ID;
|
|
||||||
endif
|
|
||||||
:Insert Brewery (FK: location_id);
|
|
||||||
|
|
||||||
if (Exception caught during insert?) then (yes)
|
|
||||||
|#EAF0E8|BiergartenDataGenerator|
|
|
||||||
:spdlog::warn "Failed to stream record to SQLite export";
|
|
||||||
note right
|
|
||||||
Data loss is prevented per-record.
|
|
||||||
The pipeline continues running.
|
|
||||||
end note
|
|
||||||
else (no)
|
|
||||||
endif
|
|
||||||
else (no)
|
|
||||||
:spdlog::warn "Generation failed, skipping...";
|
|
||||||
endif
|
|
||||||
|#E5EDE1|DataGenerator|
|
|
||||||
endwhile (Done)
|
|
||||||
|
|
||||||
|#E0EAE0|SqliteExportService|
|
|
||||||
:Finalize();
|
|
||||||
note right
|
|
||||||
Commits Transaction
|
|
||||||
Closes Database Connection
|
|
||||||
end note
|
|
||||||
|
|
||||||
|#F2F6F0|main.cc|
|
|
||||||
:Return 0;
|
|
||||||
stop
|
|
||||||
|
|
||||||
@enduml
|
|
||||||
@@ -1,196 +0,0 @@
|
|||||||
@startuml
|
|
||||||
skinparam style strictuml
|
|
||||||
skinparam defaultFontName "DM Sans"
|
|
||||||
skinparam defaultFontSize 14
|
|
||||||
skinparam titleFontName "Volkhov"
|
|
||||||
skinparam titleFontSize 20
|
|
||||||
skinparam backgroundColor #FAFCF9
|
|
||||||
skinparam defaultFontColor #28342A
|
|
||||||
skinparam titleFontColor #28342A
|
|
||||||
skinparam ArrowColor #628A5B
|
|
||||||
|
|
||||||
skinparam class {
|
|
||||||
BackgroundColor #FAFCF9
|
|
||||||
HeaderBackgroundColor #EAF0E8
|
|
||||||
BorderColor #547461
|
|
||||||
ArrowColor #628A5B
|
|
||||||
FontColor #28342A
|
|
||||||
}
|
|
||||||
|
|
||||||
skinparam note {
|
|
||||||
BackgroundColor #EAF0E8
|
|
||||||
BorderColor #547461
|
|
||||||
FontColor #28342A
|
|
||||||
}
|
|
||||||
|
|
||||||
title The Biergarten Data Pipeline - Class Diagram
|
|
||||||
|
|
||||||
class BiergartenDataGenerator {
|
|
||||||
- logger_ : std::shared_ptr<ILogger>
|
|
||||||
- context_service_ : std::unique_ptr<IEnrichmentService>
|
|
||||||
- generator_ : std::unique_ptr<DataGenerator>
|
|
||||||
- exporter_ : std::unique_ptr<IExportService>
|
|
||||||
- generated_breweries_ : std::vector<GeneratedBrewery>
|
|
||||||
+ Run() : bool
|
|
||||||
- QueryCitiesWithCountries() : std::vector<Location>
|
|
||||||
- GenerateBreweries(cities : std::span<const EnrichedCity>) : void
|
|
||||||
- LogResults() : void
|
|
||||||
}
|
|
||||||
|
|
||||||
class LogLevel <<enumeration>> {
|
|
||||||
Debug
|
|
||||||
Info
|
|
||||||
Warn
|
|
||||||
Error
|
|
||||||
}
|
|
||||||
|
|
||||||
class PipelinePhase <<enumeration>> {
|
|
||||||
Startup
|
|
||||||
UserGeneration
|
|
||||||
BreweryAndBeerGeneration
|
|
||||||
CheckinGeneration
|
|
||||||
RatingGeneration
|
|
||||||
FollowGeneration
|
|
||||||
Teardown
|
|
||||||
}
|
|
||||||
|
|
||||||
struct LogEntry {
|
|
||||||
+ timestamp : std::chrono::system_clock::time_point
|
|
||||||
+ level : LogLevel
|
|
||||||
+ phase : PipelinePhase
|
|
||||||
+ message : std::string
|
|
||||||
+ worker : std::optional<std::string>
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ILogger <<interface>> {
|
|
||||||
+ Log(entry : const LogEntry&) : void
|
|
||||||
}
|
|
||||||
|
|
||||||
class LogProducer {
|
|
||||||
- channel_ : BoundedChannel<LogEntry>&
|
|
||||||
+ Log(entry : const LogEntry&) : void
|
|
||||||
}
|
|
||||||
|
|
||||||
class LogDispatcher {
|
|
||||||
- channel_ : BoundedChannel<LogEntry>&
|
|
||||||
+ Run() : void
|
|
||||||
- ToSpdlogLevel(level) : spdlog::level::level_enum
|
|
||||||
}
|
|
||||||
|
|
||||||
interface IEnrichmentService <<interface>> {
|
|
||||||
+ GetLocationContext(loc : const Location&) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class WikipediaService {
|
|
||||||
- client_ : std::unique_ptr<WebClient>
|
|
||||||
- extract_cache_ : std::unordered_map<std::string, std::string>
|
|
||||||
+ GetLocationContext(loc : const Location&) : std::string
|
|
||||||
- FetchExtract(query : std::string_view) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface WebClient <<interface>> {
|
|
||||||
+ Get(url : const std::string&) : std::string
|
|
||||||
+ UrlEncode(value : const std::string&) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class HttpWebClient {
|
|
||||||
+ Get(url : const std::string&) : std::string
|
|
||||||
+ UrlEncode(value : const std::string&) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface DataGenerator <<interface>> {
|
|
||||||
+ GenerateBrewery(location : const Location&, region_context : const std::string&) : BreweryResult
|
|
||||||
+ GenerateUser(locale : const std::string&) : UserResult
|
|
||||||
}
|
|
||||||
|
|
||||||
class MockGenerator {
|
|
||||||
+ GenerateBrewery(...) : BreweryResult
|
|
||||||
+ GenerateUser(...) : UserResult
|
|
||||||
- DeterministicHash(location : const Location&) : size_t
|
|
||||||
}
|
|
||||||
|
|
||||||
class LlamaGenerator {
|
|
||||||
- model_ : ModelHandle
|
|
||||||
- context_ : ContextHandle
|
|
||||||
- prompt_formatter_ : std::unique_ptr<IPromptFormatter>
|
|
||||||
- rng_ : std::mt19937
|
|
||||||
+ GenerateBrewery(...) : BreweryResult
|
|
||||||
+ GenerateUser(...) : UserResult
|
|
||||||
- Load(model_path : const std::string&) : void
|
|
||||||
- Infer(...) : std::string
|
|
||||||
- InferFormatted(...) : std::string
|
|
||||||
- LoadBrewerySystemPrompt(...) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface IPromptFormatter <<interface>> {
|
|
||||||
+ Format(system_prompt : std::string_view, user_prompt : std::string_view) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class Gemma4JinjaPromptFormatter {
|
|
||||||
+ Format(system_prompt : std::string_view, user_prompt : std::string_view) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class JsonLoader {
|
|
||||||
+ {static} LoadLocations(filepath : const std::filesystem::path&) : std::vector<Location>
|
|
||||||
}
|
|
||||||
|
|
||||||
interface IExportService <<interface>> {
|
|
||||||
+ Initialize() : void
|
|
||||||
+ ProcessRecord(brewery : const GeneratedBrewery&) : void
|
|
||||||
+ Finalize() : void
|
|
||||||
}
|
|
||||||
|
|
||||||
class SqliteExportService {
|
|
||||||
- date_time_provider_ : std::unique_ptr<IDateTimeProvider>
|
|
||||||
- run_timestamp_utc_ : std::string
|
|
||||||
- database_path_ : std::filesystem::path
|
|
||||||
- db_handle_ : sqlite3*
|
|
||||||
- insert_location_stmt_ : sqlite3_stmt*
|
|
||||||
- insert_brewery_stmt_ : sqlite3_stmt*
|
|
||||||
- transaction_open_ : bool
|
|
||||||
- location_cache_ : std::unordered_map<std::string, sqlite3_int64>
|
|
||||||
+ Initialize() : void
|
|
||||||
+ ProcessRecord(brewery : const GeneratedBrewery&) : void
|
|
||||||
+ Finalize() : void
|
|
||||||
- InitializeSchema() : void
|
|
||||||
}
|
|
||||||
|
|
||||||
interface IDateTimeProvider <<interface>> {
|
|
||||||
+ GetUtcTimestamp() : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class SystemDateTimeProvider {
|
|
||||||
+ GetUtcTimestamp() : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
' Structural Relationships / Dependency Injection
|
|
||||||
BiergartenDataGenerator *-- ILogger : owns
|
|
||||||
BiergartenDataGenerator *-- IEnrichmentService : owns
|
|
||||||
BiergartenDataGenerator *-- DataGenerator : owns
|
|
||||||
BiergartenDataGenerator *-- IExportService : owns
|
|
||||||
|
|
||||||
LogEntry *-- LogLevel
|
|
||||||
LogEntry *-- PipelinePhase
|
|
||||||
ILogger <|.. LogProducer : implements
|
|
||||||
LogProducer ..> LogEntry : emits
|
|
||||||
LogDispatcher ..> LogEntry : consumes
|
|
||||||
|
|
||||||
IEnrichmentService <|.. WikipediaService : implements
|
|
||||||
WikipediaService *-- WebClient : owns
|
|
||||||
|
|
||||||
WebClient <|.. HttpWebClient : implements
|
|
||||||
|
|
||||||
DataGenerator <|.. MockGenerator : implements
|
|
||||||
DataGenerator <|.. LlamaGenerator : implements
|
|
||||||
|
|
||||||
LlamaGenerator *-- IPromptFormatter : uses
|
|
||||||
|
|
||||||
IPromptFormatter <|.. Gemma4JinjaPromptFormatter : implements
|
|
||||||
|
|
||||||
BiergartenDataGenerator ..> JsonLoader : uses
|
|
||||||
|
|
||||||
IExportService <|.. SqliteExportService : implements
|
|
||||||
SqliteExportService *-- IDateTimeProvider : owns
|
|
||||||
IDateTimeProvider <|.. SystemDateTimeProvider : implements
|
|
||||||
|
|
||||||
@enduml
|
|
||||||
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@@ -1,360 +0,0 @@
|
|||||||
@startuml biergarten_activity
|
|
||||||
!include ../biergarten-weizen-theme.puml
|
|
||||||
skinparam defaultFontSize 13
|
|
||||||
skinparam titleFontSize 20
|
|
||||||
|
|
||||||
title The Biergarten Data Pipeline — Activity Diagram
|
|
||||||
|
|
||||||
|Main|
|
|
||||||
start
|
|
||||||
:ParseArguments(argc, argv);
|
|
||||||
if (Invalid args?) then (yes)
|
|
||||||
:spdlog::error;
|
|
||||||
stop
|
|
||||||
else (no)
|
|
||||||
endif
|
|
||||||
:Init OpenSSL global state & LlamaBackendState;
|
|
||||||
:Build DI injector;
|
|
||||||
|
|
||||||
:Initialize SqliteExportService;
|
|
||||||
note right
|
|
||||||
Opens SQLite connection.
|
|
||||||
(Transactions are now managed
|
|
||||||
per-phase via batching).
|
|
||||||
end note
|
|
||||||
|
|
||||||
:Create BoundedChannel<LogEntry> log_ch;
|
|
||||||
:Spawn Log Worker thread;
|
|
||||||
note right
|
|
||||||
Log worker drains log_ch for the
|
|
||||||
entire pipeline lifetime.
|
|
||||||
All workers emit LogEntry structs
|
|
||||||
via PipelineLogger -- never spdlog directly.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:BiergartenPipelineOrchestrator::Run();
|
|
||||||
|BiergartenPipelineOrchestrator::Run()|
|
|
||||||
|
|
||||||
fork
|
|
||||||
:JsonLoader::LoadBeerStyles("beer-styles.json");
|
|
||||||
:EnrichmentService::PreWarmBeerStyleCache(beer_styles);
|
|
||||||
fork again
|
|
||||||
:JsonLoader::LoadLocations("locations.json");
|
|
||||||
:EnrichmentService::PreWarmLocationCache(sampled_locations);
|
|
||||||
end fork
|
|
||||||
fork
|
|
||||||
:JsonLoader::LoadNamesByCountry("names-by-country.json");
|
|
||||||
fork again
|
|
||||||
:JsonLoader::LoadPersonas("personas.json");
|
|
||||||
end fork
|
|
||||||
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
' PHASE 0 — USER GENERATION
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
|Orchestrator|
|
|
||||||
:RunUserPhase(sampled_locations);
|
|
||||||
:Create BoundedChannels\n(loc_ch, exp_ch);
|
|
||||||
|
|
||||||
fork
|
|
||||||
|Orchestrator|
|
|
||||||
:Loop: Send Locations -> loc_ch;
|
|
||||||
:Close loc_ch;
|
|
||||||
note right
|
|
||||||
Producer closes loc_ch.
|
|
||||||
LLM Worker while loop
|
|
||||||
terminates on empty + closed.
|
|
||||||
end note
|
|
||||||
fork again
|
|
||||||
|LLM Worker|
|
|
||||||
while (loc_ch has items?) is (yes)
|
|
||||||
:Receive Location;
|
|
||||||
|
|
||||||
:GetLocationContextFromCache(location);
|
|
||||||
note right
|
|
||||||
Guaranteed cache hit from startup.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:IPersonaSelectionStrategy::SelectPersona(\n personas_palette_);
|
|
||||||
note right
|
|
||||||
Guaranteed cache hit from startup.
|
|
||||||
Returns a Persona struct carrying
|
|
||||||
style_affinities, abv_range,
|
|
||||||
ibu_preference, checkin_weight.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:NamesByCountry::SampleName(\n location.iso3166_1);
|
|
||||||
note right
|
|
||||||
Deterministic lookup -- no LLM involved.
|
|
||||||
Name selected from pre-keyed table
|
|
||||||
and passed into the generation prompt.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:GenerateUser(enriched_city, persona, sampled_name)\nvia DataGenerator;
|
|
||||||
note right
|
|
||||||
LLM receives: EnrichedCity context + persona
|
|
||||||
description + sampled name. Generates
|
|
||||||
bio and preference signals grounded
|
|
||||||
in locale and persona.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:PipelineLogger::Log(Info, UserGeneration,\n city, user_id, "llm");
|
|
||||||
:Send GeneratedUser -> exp_ch;
|
|
||||||
endwhile (no)
|
|
||||||
:Close exp_ch;
|
|
||||||
note right
|
|
||||||
Producer closes exp_ch.
|
|
||||||
SQLite Worker while loop
|
|
||||||
terminates on empty + closed.
|
|
||||||
end note
|
|
||||||
fork again
|
|
||||||
|SQLite Worker|
|
|
||||||
:BEGIN TRANSACTION;
|
|
||||||
while (exp_ch has items?) is (yes)
|
|
||||||
:Receive GeneratedUser;
|
|
||||||
:ProcessUser(user);
|
|
||||||
:PipelineLogger::Log(Info, UserGeneration,\n city, user_id, "sqlite");
|
|
||||||
:Append -> user_pool_;
|
|
||||||
if (Batch size reached?) then (yes)
|
|
||||||
:COMMIT & BEGIN;
|
|
||||||
else (no)
|
|
||||||
endif
|
|
||||||
endwhile (no)
|
|
||||||
:COMMIT (Final);
|
|
||||||
end fork
|
|
||||||
|
|
||||||
|Orchestrator|
|
|
||||||
:Join LLM Worker, SQLite Worker;
|
|
||||||
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
' PHASE 1a — BREWERY GENERATION
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
:RunBreweryPhase(sampled_locations);
|
|
||||||
:Create BoundedChannels\n(loc_ch, exp_ch);
|
|
||||||
|
|
||||||
fork
|
|
||||||
|Orchestrator|
|
|
||||||
:Loop: Sample User from user_pool_
|
|
||||||
and pair with Location;
|
|
||||||
:Send BreweryTask(Location, User) -> loc_ch;
|
|
||||||
:Close loc_ch;
|
|
||||||
fork again
|
|
||||||
|LLM Worker|
|
|
||||||
while (loc_ch has items?) is (yes)
|
|
||||||
:Receive BreweryTask(Location, User);
|
|
||||||
|
|
||||||
:GetLocationContextFromCache(task.location);
|
|
||||||
note right
|
|
||||||
Guaranteed cache hit from startup.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:GenerateBrewery(enriched_city, context, task.user)\nvia DataGenerator;
|
|
||||||
note right
|
|
||||||
KV cache stays warm.
|
|
||||||
Brewery is linked to the sampled owner_user_id.
|
|
||||||
end note
|
|
||||||
:PipelineLogger::Log(Info,\n BreweryGeneration,\n city, brewery_id, "llm");
|
|
||||||
:Send GeneratedBrewery -> exp_ch;
|
|
||||||
endwhile (no)
|
|
||||||
:Close exp_ch;
|
|
||||||
fork again
|
|
||||||
|SQLite Worker|
|
|
||||||
:BEGIN TRANSACTION;
|
|
||||||
while (exp_ch has items?) is (yes)
|
|
||||||
:Receive GeneratedBrewery;
|
|
||||||
:ProcessBrewery(brewery);
|
|
||||||
:PipelineLogger::Log(Info,\n BreweryGeneration,\n city, brewery_id, "sqlite");
|
|
||||||
:Append -> brewery_pool_;
|
|
||||||
if (Batch size reached?) then (yes)
|
|
||||||
:COMMIT & BEGIN;
|
|
||||||
else (no)
|
|
||||||
endif
|
|
||||||
endwhile (no)
|
|
||||||
:COMMIT (Final);
|
|
||||||
end fork
|
|
||||||
|
|
||||||
|Orchestrator|
|
|
||||||
:Join LLM Worker, SQLite Worker;
|
|
||||||
note right
|
|
||||||
brewery_pool_ is now fully populated.
|
|
||||||
Phase 1b may begin.
|
|
||||||
end note
|
|
||||||
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
' PHASE 1b — BEER GENERATION
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
:RunBeerPhase();
|
|
||||||
:Create BoundedChannels\n(brew_ch, exp_ch);
|
|
||||||
|
|
||||||
fork
|
|
||||||
|Orchestrator|
|
|
||||||
:Loop: Send Breweries -> brew_ch;
|
|
||||||
:Close brew_ch;
|
|
||||||
fork again
|
|
||||||
|LLM Worker|
|
|
||||||
while (brew_ch has items?) is (yes)
|
|
||||||
:Receive GeneratedBrewery;
|
|
||||||
:IBeerSelectionStrategy::SelectStyles(\n brewery, beer_style_palette_);
|
|
||||||
|
|
||||||
while (For each selected BeerStyle?) is (remaining)
|
|
||||||
:GetStyleContextFromCache(style);
|
|
||||||
note right
|
|
||||||
Guaranteed cache hit from startup.
|
|
||||||
KV cache stays warm across all
|
|
||||||
beer generations -- system prompt
|
|
||||||
does not change within this phase.
|
|
||||||
end note
|
|
||||||
:GenerateBeer(brewery, style_context)\nvia DataGenerator;
|
|
||||||
:Attach GeneratedBeer to bundle;
|
|
||||||
endwhile (done)
|
|
||||||
|
|
||||||
:PipelineLogger::Log(Info,\n BeerGeneration,\n city, brewery_id, "llm");
|
|
||||||
:Send BeersBundle -> exp_ch;
|
|
||||||
endwhile (no)
|
|
||||||
:Close exp_ch;
|
|
||||||
fork again
|
|
||||||
|SQLite Worker|
|
|
||||||
:BEGIN TRANSACTION;
|
|
||||||
while (exp_ch has items?) is (yes)
|
|
||||||
:Receive BeersBundle;
|
|
||||||
while (For each beer in bundle?) is (remaining)
|
|
||||||
:Set beer.brewery_id from bundle;
|
|
||||||
:ProcessBeer(beer);
|
|
||||||
:Append -> beer_pool_;
|
|
||||||
endwhile (done)
|
|
||||||
:PipelineLogger::Log(Info,\n BeerGeneration,\n city, brewery_id, "sqlite");
|
|
||||||
if (Batch size reached?) then (yes)
|
|
||||||
:COMMIT & BEGIN;
|
|
||||||
else (no)
|
|
||||||
endif
|
|
||||||
endwhile (no)
|
|
||||||
:COMMIT (Final);
|
|
||||||
end fork
|
|
||||||
|
|
||||||
|Orchestrator|
|
|
||||||
:Join LLM Worker, SQLite Worker;
|
|
||||||
note right
|
|
||||||
Both brewery_pool_ and beer_pool_
|
|
||||||
are now completely populated.
|
|
||||||
Checkin and Follow phases may
|
|
||||||
now run in parallel.
|
|
||||||
end note
|
|
||||||
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
' PHASE 2 — CHECKIN + FOLLOW GENERATION
|
|
||||||
' (parallel — both depend only on user_pool_
|
|
||||||
' and brewery_pool_ being fully populated)
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
fork
|
|
||||||
|Orchestrator|
|
|
||||||
:RunCheckinPhase();
|
|
||||||
:ICheckinDistributionStrategy::\nAssignActivityWeights(user_pool_);
|
|
||||||
note right
|
|
||||||
Weights seeded from each user's
|
|
||||||
persona.checkin_weight. J-curve profile
|
|
||||||
emerges from persona distribution.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:BEGIN TRANSACTION;
|
|
||||||
while (For each GeneratedUser in user_pool_?) is (remaining)
|
|
||||||
:CheckinsForUser(user, brewery_pool_.size());
|
|
||||||
while (For each checkin index?) is (remaining)
|
|
||||||
:TimestampFor(user, index);
|
|
||||||
:Select brewery from brewery_pool_;
|
|
||||||
:GenerateCheckin(user, brewery, timestamp)\nvia DataGenerator;
|
|
||||||
:ProcessCheckin(checkin);
|
|
||||||
:PipelineLogger::Log(Info, CheckinGeneration,\n nullopt, checkin_id, "sqlite");
|
|
||||||
:Append -> checkin_pool_;
|
|
||||||
if (Batch size reached?) then (yes)
|
|
||||||
:COMMIT & BEGIN;
|
|
||||||
else (no)
|
|
||||||
endif
|
|
||||||
endwhile (done)
|
|
||||||
endwhile (done)
|
|
||||||
:COMMIT (Final);
|
|
||||||
|
|
||||||
fork again
|
|
||||||
|Orchestrator|
|
|
||||||
:RunFollowPhase();
|
|
||||||
:IFollowGenerationStrategy::\nAssignFollowWeights(user_pool_);
|
|
||||||
note right
|
|
||||||
For RandomFollowStrategy, weights
|
|
||||||
are uniform. For ActivityWeightedFollowStrategy,
|
|
||||||
weights derived from user.activity_weight
|
|
||||||
so high-activity users attract more followers.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:BEGIN TRANSACTION;
|
|
||||||
:IFollowGenerationStrategy::\nGenerateFollows(user_pool_);
|
|
||||||
note right
|
|
||||||
Self-follow constraint (follower_id != followed_id)
|
|
||||||
enforced here and at the DB schema level.
|
|
||||||
end note
|
|
||||||
while (For each GeneratedFollow?) is (remaining)
|
|
||||||
:ProcessFollow(follow);
|
|
||||||
:PipelineLogger::Log(Info, FollowGeneration,\n nullopt, follower_id, "sqlite");
|
|
||||||
:Append -> follow_pool_;
|
|
||||||
if (Batch size reached?) then (yes)
|
|
||||||
:COMMIT & BEGIN;
|
|
||||||
else (no)
|
|
||||||
endif
|
|
||||||
endwhile (done)
|
|
||||||
:COMMIT (Final);
|
|
||||||
|
|
||||||
end fork
|
|
||||||
|
|
||||||
|Orchestrator|
|
|
||||||
:Join CheckinPhase, FollowPhase;
|
|
||||||
note right
|
|
||||||
checkin_pool_ and follow_pool_
|
|
||||||
are now fully populated.
|
|
||||||
Rating phase may begin.
|
|
||||||
end note
|
|
||||||
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
' PHASE 3 — RATING GENERATION
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
:RunRatingPhase();
|
|
||||||
note right
|
|
||||||
Beer selection biased by
|
|
||||||
user.persona.style_affinities and abv_range.
|
|
||||||
Rating skew modulated per persona.
|
|
||||||
end note
|
|
||||||
|
|
||||||
:BEGIN TRANSACTION;
|
|
||||||
while (For each GeneratedCheckin in checkin_pool_?) is (remaining)
|
|
||||||
:Match brewery_id, select beer from beer_pool_\n(same brewery_id, biased by persona affinities);
|
|
||||||
if (Beer exists for brewery?) then (yes)
|
|
||||||
:GenerateRating(user, beer, checkin_id)\nvia DataGenerator;
|
|
||||||
:ProcessRating(rating);
|
|
||||||
:PipelineLogger::Log(Info, RatingGeneration,\n nullopt, rating_id, "sqlite");
|
|
||||||
if (Batch size reached?) then (yes)
|
|
||||||
:COMMIT & BEGIN;
|
|
||||||
else (no)
|
|
||||||
endif
|
|
||||||
else (no)
|
|
||||||
:PipelineLogger::Log(Warn, RatingGeneration,\n nullopt, brewery_id, "sqlite");
|
|
||||||
:Skip -- brewery has no beers;
|
|
||||||
endif
|
|
||||||
endwhile (done)
|
|
||||||
:COMMIT (Final);
|
|
||||||
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
' TEARDOWN
|
|
||||||
' ═══════════════════════════════════════════
|
|
||||||
|Orchestrator|
|
|
||||||
:Finalize SqliteExportService;
|
|
||||||
note right
|
|
||||||
Safely closes the DB connection.
|
|
||||||
end note
|
|
||||||
:Close log_ch;
|
|
||||||
|
|
||||||
|Main|
|
|
||||||
:spdlog::info "Pipeline complete in X ms";
|
|
||||||
:Join Log Worker;
|
|
||||||
note right
|
|
||||||
Drain guarantees no LogEntry is
|
|
||||||
dropped at shutdown.
|
|
||||||
end note
|
|
||||||
stop
|
|
||||||
|
|
||||||
@enduml
|
|
||||||
@@ -1,572 +0,0 @@
|
|||||||
@startuml class_diagram
|
|
||||||
|
|
||||||
' ==========================================
|
|
||||||
' CONFIGURATION & STYLING
|
|
||||||
' ==========================================
|
|
||||||
!include ../biergarten-weizen-theme.puml
|
|
||||||
skinparam classAttributeFontSize 9
|
|
||||||
skinparam defaultFontSize 25
|
|
||||||
skinparam titleFontSize 30
|
|
||||||
|
|
||||||
title Biergarten Data Pipeline — Class Diagram
|
|
||||||
|
|
||||||
package "Domain: Models" {
|
|
||||||
|
|
||||||
class Location {
|
|
||||||
+ city : std::string
|
|
||||||
+ state_province : std::string
|
|
||||||
+ iso3166_2 : std::string
|
|
||||||
+ country : std::string
|
|
||||||
+ iso3166_1 : std::string
|
|
||||||
+ local_languages : std::vector<std::string>
|
|
||||||
+ latitude : double
|
|
||||||
+ longitude : double
|
|
||||||
}
|
|
||||||
|
|
||||||
class LocationContext {
|
|
||||||
+ text : std::string
|
|
||||||
+ completeness : Completeness
|
|
||||||
+ char_count : size_t
|
|
||||||
}
|
|
||||||
|
|
||||||
enum Completeness {
|
|
||||||
Full
|
|
||||||
Partial
|
|
||||||
Absent
|
|
||||||
}
|
|
||||||
|
|
||||||
class EnrichedCity {
|
|
||||||
+ location : Location
|
|
||||||
+ context : LocationContext
|
|
||||||
}
|
|
||||||
|
|
||||||
class BeerStyle {
|
|
||||||
+ name : std::string
|
|
||||||
+ description : std::string
|
|
||||||
+ min_abv : float
|
|
||||||
+ max_abv : float
|
|
||||||
+ min_ibu : int
|
|
||||||
+ max_ibu : int
|
|
||||||
}
|
|
||||||
|
|
||||||
class BreweryResult {
|
|
||||||
+ name_en : std::string
|
|
||||||
+ description_en : std::string
|
|
||||||
+ name_local : std::string
|
|
||||||
+ description_local : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class BeerResult {
|
|
||||||
+ name_en : std::string
|
|
||||||
+ description_en : std::string
|
|
||||||
+ name_local : std::string
|
|
||||||
+ description_local : std::string
|
|
||||||
+ style : std::string
|
|
||||||
+ abv : float
|
|
||||||
+ ibu : int
|
|
||||||
}
|
|
||||||
|
|
||||||
class UserResult {
|
|
||||||
+ username : std::string
|
|
||||||
+ bio : std::string
|
|
||||||
+ activity_weight : float
|
|
||||||
}
|
|
||||||
|
|
||||||
class CheckinResult {
|
|
||||||
+ checked_in_at : std::string
|
|
||||||
+ note : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class RatingResult {
|
|
||||||
+ score : float
|
|
||||||
+ note : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class GenerationMetadata {
|
|
||||||
+ generation_id : uint64_t
|
|
||||||
+ generated_time : std::string
|
|
||||||
+ context_provided : bool
|
|
||||||
+ generated_with : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class GeneratedBrewery {
|
|
||||||
+ brewery_id : uint64_t
|
|
||||||
+ location : Location
|
|
||||||
+ brewery : BreweryResult
|
|
||||||
+ context_completeness : LocationContext::Completeness
|
|
||||||
+ metadata : GenerationMetadata
|
|
||||||
}
|
|
||||||
|
|
||||||
class GeneratedBeer {
|
|
||||||
+ beer_id : uint64_t
|
|
||||||
+ brewery_id : uint64_t
|
|
||||||
+ location : Location
|
|
||||||
+ style : BeerStyle
|
|
||||||
+ beer : BeerResult
|
|
||||||
+ metadata : GenerationMetadata
|
|
||||||
}
|
|
||||||
|
|
||||||
class GeneratedUser {
|
|
||||||
+ user_id : uint64_t
|
|
||||||
+ location : Location
|
|
||||||
+ user : UserResult
|
|
||||||
+ metadata : GenerationMetadata
|
|
||||||
}
|
|
||||||
|
|
||||||
class GeneratedCheckin {
|
|
||||||
+ checkin_id : uint64_t
|
|
||||||
+ user_id : uint64_t
|
|
||||||
+ brewery_id : uint64_t
|
|
||||||
+ checkin : CheckinResult
|
|
||||||
+ metadata : GenerationMetadata
|
|
||||||
}
|
|
||||||
|
|
||||||
class GeneratedRating {
|
|
||||||
+ user_id : uint64_t
|
|
||||||
+ beer_id : uint64_t
|
|
||||||
+ checkin_id : uint64_t
|
|
||||||
+ rating : RatingResult
|
|
||||||
+ metadata : GenerationMetadata
|
|
||||||
}
|
|
||||||
|
|
||||||
class GeneratedFollow {
|
|
||||||
+ follower_id : uint64_t
|
|
||||||
+ followed_id : uint64_t
|
|
||||||
+ metadata : GenerationMetadata
|
|
||||||
}
|
|
||||||
|
|
||||||
class UserPersona {
|
|
||||||
+ name: std::string
|
|
||||||
+ description: std::string
|
|
||||||
+ style_affinities: std::vector<std::string>
|
|
||||||
}
|
|
||||||
|
|
||||||
LocationContext *-- Completeness
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Domain: Application Configuration" {
|
|
||||||
class SamplingOptions {
|
|
||||||
+ temperature: float = 1.0F
|
|
||||||
+ top_p: float = 0.95F
|
|
||||||
+ top_k: uint32_t = 64
|
|
||||||
+ n_ctx: uint32_t = 8192
|
|
||||||
+ seed: int = -1
|
|
||||||
}
|
|
||||||
|
|
||||||
class GeneratorOptions {
|
|
||||||
+ model_path: std::filesystem::path
|
|
||||||
+ use_mocked: bool = false
|
|
||||||
+ sampling: std::optional<SamplingOptions>
|
|
||||||
}
|
|
||||||
|
|
||||||
class PipelineOptions {
|
|
||||||
+ output_path: std::filesystem::path
|
|
||||||
+ log_path: std::filesystem::path
|
|
||||||
}
|
|
||||||
|
|
||||||
class ApplicationOptions {
|
|
||||||
+ generator: GeneratorOptions
|
|
||||||
+ pipeline: PipelineOptions
|
|
||||||
}
|
|
||||||
|
|
||||||
ApplicationOptions *-- GeneratorOptions
|
|
||||||
ApplicationOptions *-- PipelineOptions
|
|
||||||
GeneratorOptions o-- SamplingOptions
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Domain: Policy" {
|
|
||||||
|
|
||||||
interface ContextStrategy <<interface>> {
|
|
||||||
+ QueriesFor(loc : const Location&) : std::vector<std::string>
|
|
||||||
+ MaxContextChars() : size_t
|
|
||||||
}
|
|
||||||
|
|
||||||
class BreweryContextStrategy {
|
|
||||||
+ QueriesFor(loc : const Location&) : std::vector<std::string>
|
|
||||||
+ MaxContextChars() : size_t
|
|
||||||
}
|
|
||||||
|
|
||||||
class BeerContextStrategy {
|
|
||||||
+ QueriesFor(loc : const Location&) : std::vector<std::string>
|
|
||||||
+ MaxContextChars() : size_t
|
|
||||||
}
|
|
||||||
|
|
||||||
interface SamplingStrategy <<interface>> {
|
|
||||||
+ Sample(locations : const std::vector<Location>&) : std::vector<Location>
|
|
||||||
}
|
|
||||||
|
|
||||||
class UniformSamplingStrategy {
|
|
||||||
- sample_size_ : size_t
|
|
||||||
+ Sample(locations : const std::vector<Location>&) : std::vector<Location>
|
|
||||||
}
|
|
||||||
|
|
||||||
interface BeerSelectionStrategy <<interface>> {
|
|
||||||
+ SelectStyles(brewery : const GeneratedBrewery&,\n palette : std::span<const BeerStyle>) : std::vector<BeerStyle>
|
|
||||||
}
|
|
||||||
|
|
||||||
class RandomBeerSelectionStrategy {
|
|
||||||
- rng_ : std::mt19937
|
|
||||||
- min_beers_ : size_t
|
|
||||||
- max_beers_ : size_t
|
|
||||||
+ SelectStyles(brewery : const GeneratedBrewery&,\n palette : std::span<const BeerStyle>) : std::vector<BeerStyle>
|
|
||||||
}
|
|
||||||
|
|
||||||
interface CheckinDistributionStrategy <<interface>> {
|
|
||||||
+ AssignActivityWeights(users : std::vector<GeneratedUser>&) : void
|
|
||||||
+ CheckinsForUser(user : const GeneratedUser&,\n brewery_count : size_t) : size_t
|
|
||||||
+ TimestampFor(user : const GeneratedUser&,\n index : size_t) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class JCurveCheckinStrategy {
|
|
||||||
- rng_ : std::mt19937
|
|
||||||
+ AssignActivityWeights(users : std::vector<GeneratedUser>&) : void
|
|
||||||
+ CheckinsForUser(user : const GeneratedUser&,\n brewery_count : size_t) : size_t
|
|
||||||
+ TimestampFor(user : const GeneratedUser&,\n index : size_t) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class RandomCheckinStrategy {
|
|
||||||
- rng_ : std::mt19937
|
|
||||||
- min_checkins_ : size_t
|
|
||||||
- max_checkins_ : size_t
|
|
||||||
+ AssignActivityWeights(users : std::vector<GeneratedUser>&) : void
|
|
||||||
+ CheckinsForUser(user : const GeneratedUser&,\n brewery_count : size_t) : size_t
|
|
||||||
+ TimestampFor(user : const GeneratedUser&,\n index : size_t) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FollowGenerationStrategy <<interface>> {
|
|
||||||
+ GenerateFollows(users : const std::vector<GeneratedUser>&) : std::vector<GeneratedFollow>
|
|
||||||
}
|
|
||||||
|
|
||||||
class RandomFollowStrategy {
|
|
||||||
- rng_ : std::mt19937
|
|
||||||
- min_follows_ : size_t
|
|
||||||
- max_follows_ : size_t
|
|
||||||
+ GenerateFollows(users : const std::vector<GeneratedUser>&) : std::vector<GeneratedFollow>
|
|
||||||
}
|
|
||||||
|
|
||||||
class ActivityWeightedFollowStrategy {
|
|
||||||
- rng_ : std::mt19937
|
|
||||||
- min_follows_ : size_t
|
|
||||||
- max_follows_ : size_t
|
|
||||||
+ GenerateFollows(users : const std::vector<GeneratedUser>&) : std::vector<GeneratedFollow>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Infrastructure: Logging" {
|
|
||||||
enum LogLevel {
|
|
||||||
Debug
|
|
||||||
Info
|
|
||||||
Warn
|
|
||||||
Error
|
|
||||||
}
|
|
||||||
|
|
||||||
enum PipelinePhase {
|
|
||||||
Startup
|
|
||||||
UserGeneration
|
|
||||||
BreweryAndBeerGeneration
|
|
||||||
CheckinGeneration
|
|
||||||
RatingGeneration
|
|
||||||
FollowGeneration
|
|
||||||
Teardown
|
|
||||||
}
|
|
||||||
|
|
||||||
class LogEntry {
|
|
||||||
+ timestamp : std::chrono::system_clock::time_point
|
|
||||||
+ level : LogLevel
|
|
||||||
+ phase : PipelinePhase
|
|
||||||
+ message : std::string
|
|
||||||
+ worker : std::optional<std::string>
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ILogger <<interface>> {
|
|
||||||
+ Log(entry : const LogEntry&) : void
|
|
||||||
}
|
|
||||||
|
|
||||||
class LogProducer {
|
|
||||||
- channel_ : BoundedChannel<LogEntry>&
|
|
||||||
+ Log(entry : const LogEntry&) : void
|
|
||||||
}
|
|
||||||
|
|
||||||
class LogDispatcher {
|
|
||||||
- channel_ : BoundedChannel<LogEntry>&
|
|
||||||
+ Run() : void
|
|
||||||
- ToSpdlogLevel(level) : spdlog::level::level_enum
|
|
||||||
}
|
|
||||||
|
|
||||||
LogEntry *-- LogLevel
|
|
||||||
LogEntry *-- PipelinePhase
|
|
||||||
ILogger <|.. LogProducer
|
|
||||||
LogProducer ..> LogEntry : emits
|
|
||||||
LogDispatcher ..> LogEntry : consumes
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Infrastructure: Pipeline Channel" {
|
|
||||||
|
|
||||||
class "BoundedChannel<T>" as BoundedChannel {
|
|
||||||
- queue_ : std::queue<T>
|
|
||||||
- mutex_ : std::mutex
|
|
||||||
- not_full_ : std::condition_variable
|
|
||||||
- not_empty_ : std::condition_variable
|
|
||||||
- capacity_ : size_t
|
|
||||||
- closed_ : bool
|
|
||||||
+ Send(item : T) : void
|
|
||||||
+ Receive() : std::optional<T>
|
|
||||||
+ Close() : void
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Infrastructure: Data Preloading" {
|
|
||||||
|
|
||||||
interface DataPreloader <<interface>> {
|
|
||||||
+ LoadLocations(filepath : const std::filesystem::path&) : std::vector<Location>
|
|
||||||
+ LoadBeerStyles(filepath : const std::filesystem::path&) : std::vector<BeerStyle>
|
|
||||||
+ LoadPersonas(filepath : const std::filesystem::path&) : std::vector<Persona>
|
|
||||||
+ LoadNamesByCountry(filepath : const std::filesystem::path&) : NamesByCountry
|
|
||||||
}
|
|
||||||
|
|
||||||
class JsonLoader {
|
|
||||||
+ LoadLocations(filepath : const std::filesystem::path&) : std::vector<Location>
|
|
||||||
+ LoadBeerStyles(filepath : const std::filesystem::path&) : std::vector<BeerStyle>
|
|
||||||
+ LoadPersonas(filepath : const std::filesystem::path&) : std::vector<Persona>
|
|
||||||
+ LoadNamesByCountry(filepath : const std::filesystem::path&) : NamesByCountry
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Infrastructure: Enrichment" {
|
|
||||||
|
|
||||||
interface EnrichmentService <<interface>> {
|
|
||||||
+ GetLocationContext(loc : const Location&,\n strategy : const ContextStrategy&) : LocationContext
|
|
||||||
}
|
|
||||||
|
|
||||||
class WikipediaService {
|
|
||||||
- client_ : std::unique_ptr<WebClient>
|
|
||||||
- extract_cache_ : std::unordered_map<std::string, std::string>
|
|
||||||
+ GetLocationContext(loc : const Location&,\n strategy : const ContextStrategy&) : LocationContext
|
|
||||||
- FetchExtract(query : std::string_view) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface WebClient <<interface>> {
|
|
||||||
+ Get(url : const std::string&) : std::string
|
|
||||||
+ UrlEncode(value : const std::string&) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class HttpWebClient {
|
|
||||||
+ Get(url : const std::string&) : std::string
|
|
||||||
+ UrlEncode(value : const std::string&) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Infrastructure: Prompting" {
|
|
||||||
|
|
||||||
interface IPromptDirectory <<interface>> {
|
|
||||||
+ Load(key : std::string_view) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class PromptDirectory {
|
|
||||||
- prompt_dir_ : std::filesystem::path
|
|
||||||
- cache_ : std::unordered_map<std::string, std::string>
|
|
||||||
+ PromptDirectory(prompt_dir : const std::filesystem::path&)
|
|
||||||
+ Load(key : std::string_view) : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
IPromptDirectory <|.. PromptDirectory
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Infrastructure: Data Generation" {
|
|
||||||
|
|
||||||
interface DataGenerator <<interface>> {
|
|
||||||
+ GenerateBrewery(location : const Location&,\n context : const LocationContext&) : BreweryResult
|
|
||||||
+ GenerateBeer(brewery_id : uint64_t,\n location : const Location&,\n context : const LocationContext&,\n style : const BeerStyle&) : BeerResult
|
|
||||||
+ GenerateUser(location : const Location&) : UserResult
|
|
||||||
+ GenerateCheckin(user : const GeneratedUser&,\n brewery : const GeneratedBrewery&,\n timestamp : const std::string&) : CheckinResult
|
|
||||||
+ GenerateRating(user : const GeneratedUser&,\n beer : const GeneratedBeer&,\n checkin_id : uint64_t) : RatingResult
|
|
||||||
}
|
|
||||||
|
|
||||||
class MockGenerator {
|
|
||||||
+ GenerateBrewery(...) : BreweryResult
|
|
||||||
+ GenerateBeer(...) : BeerResult
|
|
||||||
+ GenerateUser(...) : UserResult
|
|
||||||
+ GenerateCheckin(...) : CheckinResult
|
|
||||||
+ GenerateRating(...) : RatingResult
|
|
||||||
- DeterministicHash(location : const Location&) : size_t
|
|
||||||
}
|
|
||||||
|
|
||||||
class LlamaGenerator {
|
|
||||||
- model_ : ModelHandle
|
|
||||||
- context_ : ContextHandle
|
|
||||||
- prompt_formatter_ : std::unique_ptr<PromptFormatter>
|
|
||||||
- prompt_directory_ : std::unique_ptr<IPromptDirectory>
|
|
||||||
- rng_ : std::mt19937
|
|
||||||
+ GenerateBrewery(...) : BreweryResult
|
|
||||||
+ GenerateBeer(...) : BeerResult
|
|
||||||
+ GenerateUser(...) : UserResult
|
|
||||||
+ GenerateCheckin(...) : CheckinResult
|
|
||||||
+ GenerateRating(...) : RatingResult
|
|
||||||
- Load(opts : const GeneratorOptions&) : void
|
|
||||||
- Infer(system_prompt, user_prompt,\n max_tokens, grammar) : std::string
|
|
||||||
- ValidateModelArchitecture() : void
|
|
||||||
}
|
|
||||||
|
|
||||||
interface PromptFormatter <<interface>> {
|
|
||||||
+ Format(system_prompt : std::string_view,\n user_prompt : std::string_view) : std::string
|
|
||||||
+ ExpectedArchitecture() : std::string_view
|
|
||||||
}
|
|
||||||
|
|
||||||
class Gemma4JinjaPromptFormatter {
|
|
||||||
+ Format(...) : std::string
|
|
||||||
+ ExpectedArchitecture() : std::string_view
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
package "Infrastructure: Data Export" {
|
|
||||||
|
|
||||||
interface ExportService <<interface>> {
|
|
||||||
+ Initialize() : void
|
|
||||||
+ ProcessBrewery(brewery : const GeneratedBrewery&) : uint64_t
|
|
||||||
+ ProcessBeer(beer : const GeneratedBeer&) : uint64_t
|
|
||||||
+ ProcessUser(user : const GeneratedUser&) : uint64_t
|
|
||||||
+ ProcessCheckin(checkin : const GeneratedCheckin&) : uint64_t
|
|
||||||
+ ProcessRating(rating : const GeneratedRating&) : void
|
|
||||||
+ ProcessFollow(follow : const GeneratedFollow&) : void
|
|
||||||
+ Finalize() : void
|
|
||||||
}
|
|
||||||
|
|
||||||
class SqliteExportService {
|
|
||||||
- date_time_provider_ : std::unique_ptr<DateTimeProvider>
|
|
||||||
- db_handle_ : SqliteDatabaseHandle
|
|
||||||
- insert_location_stmt_ : SqliteStatementHandle
|
|
||||||
- insert_brewery_stmt_ : SqliteStatementHandle
|
|
||||||
- insert_beer_stmt_ : SqliteStatementHandle
|
|
||||||
- insert_user_stmt_ : SqliteStatementHandle
|
|
||||||
- insert_checkin_stmt_ : SqliteStatementHandle
|
|
||||||
- insert_rating_stmt_ : SqliteStatementHandle
|
|
||||||
- insert_follow_stmt_ : SqliteStatementHandle
|
|
||||||
- transaction_open_ : bool
|
|
||||||
- location_cache_ : std::unordered_map<std::string, uint64_t>
|
|
||||||
- brewery_cache_ : std::unordered_map<std::string, uint64_t>
|
|
||||||
+ Initialize() : void
|
|
||||||
+ ProcessRecord(brewery : const GeneratedBrewery&) : uint64_t
|
|
||||||
+ ProcessRecord(beer : const GeneratedBeer&) : uint64_t
|
|
||||||
+ ProcessRecord(user : const GeneratedUser&) : uint64_t
|
|
||||||
+ ProcessRecord(checkin : const GeneratedCheckin&) : uint64_t
|
|
||||||
+ ProcessRecord(rating : const GeneratedRating&) : void
|
|
||||||
+ ProcessRecord(follow : const GeneratedFollow&) : void
|
|
||||||
+ Finalize() : void
|
|
||||||
- InitializeSchema() : void
|
|
||||||
- PrepareStatements() : void
|
|
||||||
- RollbackAndCloseNoThrow() : void
|
|
||||||
- FinalizeStatements() : void
|
|
||||||
}
|
|
||||||
|
|
||||||
interface DateTimeProvider <<interface>> {
|
|
||||||
+ GetUtcTimestamp() : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
class SystemDateTimeProvider {
|
|
||||||
+ GetUtcTimestamp() : std::string
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
class BiergartenPipelineOrchestrator {
|
|
||||||
- preloader_ : std::unique_ptr<DataPreloader>
|
|
||||||
- enrichment_service_ : std::unique_ptr<EnrichmentService>
|
|
||||||
- generator_ : std::unique_ptr<DataGenerator>
|
|
||||||
- logger_ : std::unique_ptr<Logger>
|
|
||||||
- exporter_ : std::unique_ptr<ExportService>
|
|
||||||
- brewery_context_strategy_ : std::unique_ptr<ContextStrategy>
|
|
||||||
- sampling_strategy_ : std::unique_ptr<SamplingStrategy>
|
|
||||||
- beer_selection_strategy_ : std::unique_ptr<BeerSelectionStrategy>
|
|
||||||
- checkin_strategy_ : std::unique_ptr<CheckinDistributionStrategy>
|
|
||||||
- follow_strategy_ : std::unique_ptr<FollowGenerationStrategy>
|
|
||||||
- beer_style_palette_ : std::vector<BeerStyle>
|
|
||||||
- options_ : ApplicationOptions
|
|
||||||
--
|
|
||||||
- user_pool_ : std::vector<GeneratedUser>
|
|
||||||
- brewery_pool_ : std::vector<GeneratedBrewery>
|
|
||||||
- beer_pool_ : std::vector<GeneratedBeer>
|
|
||||||
- checkin_pool_ : std::vector<GeneratedCheckin>
|
|
||||||
- follow_pool_ : std::vector<GeneratedFollow>
|
|
||||||
--
|
|
||||||
+ Run() : bool
|
|
||||||
- RunUserPhase(locations : const std::vector<Location>&) : void
|
|
||||||
- RunBreweryAndBeerPhase(locations : const std::vector<Location>&) : void
|
|
||||||
- RunCheckinPhase() : void
|
|
||||||
- RunRatingPhase() : void
|
|
||||||
- RunFollowPhase() : void
|
|
||||||
}
|
|
||||||
|
|
||||||
' --- Orchestration Aggregations (Services & Strategies) ---
|
|
||||||
BiergartenPipelineOrchestrator *-- DataPreloader
|
|
||||||
BiergartenPipelineOrchestrator *-- EnrichmentService
|
|
||||||
BiergartenPipelineOrchestrator *-- DataGenerator
|
|
||||||
BiergartenPipelineOrchestrator *-- ExportService
|
|
||||||
BiergartenPipelineOrchestrator *-- CheckinDistributionStrategy
|
|
||||||
BiergartenPipelineOrchestrator *-- FollowGenerationStrategy
|
|
||||||
BiergartenPipelineOrchestrator *-- SamplingStrategy
|
|
||||||
BiergartenPipelineOrchestrator *-- BeerSelectionStrategy
|
|
||||||
BiergartenPipelineOrchestrator *-- ApplicationOptions
|
|
||||||
BiergartenPipelineOrchestrator *-- Logger
|
|
||||||
|
|
||||||
' --- Orchestration Aggregations (Data Pools) ---
|
|
||||||
BiergartenPipelineOrchestrator *-- "0..*" GeneratedUser : user_pool_
|
|
||||||
BiergartenPipelineOrchestrator *-- "0..*" GeneratedBrewery : brewery_pool_
|
|
||||||
BiergartenPipelineOrchestrator *-- "0..*" GeneratedBeer : beer_pool_
|
|
||||||
BiergartenPipelineOrchestrator *-- "0..*" GeneratedCheckin : checkin_pool_
|
|
||||||
BiergartenPipelineOrchestrator *-- "0..*" GeneratedFollow : follow_pool_
|
|
||||||
|
|
||||||
' --- Interfaces & Implementations ---
|
|
||||||
DataPreloader <|.. JsonLoader
|
|
||||||
Logger <|.. PipelineLogger
|
|
||||||
ContextStrategy <|.. BreweryContextStrategy
|
|
||||||
ContextStrategy <|.. BeerContextStrategy
|
|
||||||
SamplingStrategy <|.. UniformSamplingStrategy
|
|
||||||
BeerSelectionStrategy <|.. RandomBeerSelectionStrategy
|
|
||||||
CheckinDistributionStrategy <|.. JCurveCheckinStrategy
|
|
||||||
CheckinDistributionStrategy <|.. RandomCheckinStrategy
|
|
||||||
FollowGenerationStrategy <|.. RandomFollowStrategy
|
|
||||||
FollowGenerationStrategy <|.. ActivityWeightedFollowStrategy
|
|
||||||
EnrichmentService <|.. WikipediaService
|
|
||||||
WebClient <|.. HttpWebClient
|
|
||||||
DataGenerator <|.. MockGenerator
|
|
||||||
DataGenerator <|.. LlamaGenerator
|
|
||||||
PromptFormatter <|.. Gemma4JinjaPromptFormatter
|
|
||||||
ExportService <|.. SqliteExportService
|
|
||||||
DateTimeProvider <|.. SystemDateTimeProvider
|
|
||||||
|
|
||||||
' --- Service Compositions & Dependencies ---
|
|
||||||
WikipediaService *-- WebClient
|
|
||||||
WikipediaService ..> ContextStrategy
|
|
||||||
LlamaGenerator *-- PromptFormatter
|
|
||||||
LlamaGenerator *-- IPromptDirectory
|
|
||||||
LlamaGenerator ..> GeneratorOptions
|
|
||||||
SqliteExportService *-- DateTimeProvider
|
|
||||||
|
|
||||||
' --- Cross-Component Aggregations (Held References) ---
|
|
||||||
PipelineLogger o-- BoundedChannel : logs to
|
|
||||||
LogWorker o-- BoundedChannel : drains from
|
|
||||||
|
|
||||||
' --- Domain Containment ---
|
|
||||||
EnrichedCity *-- Location
|
|
||||||
EnrichedCity *-- LocationContext
|
|
||||||
GeneratedBrewery *-- Location
|
|
||||||
GeneratedBrewery *-- BreweryResult
|
|
||||||
GeneratedBrewery *-- GenerationMetadata
|
|
||||||
GeneratedBeer *-- Location
|
|
||||||
GeneratedBeer *-- BeerStyle
|
|
||||||
GeneratedBeer *-- BeerResult
|
|
||||||
GeneratedBeer *-- GenerationMetadata
|
|
||||||
GeneratedUser *-- Location
|
|
||||||
GeneratedUser *-- UserResult
|
|
||||||
GeneratedUser *-- GenerationMetadata
|
|
||||||
GeneratedCheckin *-- CheckinResult
|
|
||||||
GeneratedCheckin *-- GenerationMetadata
|
|
||||||
GeneratedRating *-- RatingResult
|
|
||||||
GeneratedRating *-- GenerationMetadata
|
|
||||||
GeneratedFollow *-- GenerationMetadata
|
|
||||||
|
|
||||||
@enduml
|
|
||||||
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@@ -1,303 +0,0 @@
|
|||||||
[2026-04-18 14:36:49.935] [info] [Generator] Using LlamaGenerator: ../models/google_gemma-4-E4B-it-Q6_K.gguf (temperature=1, top-p=0.95, top-k=64, n_ctx=8192, seed=-1)
|
|
||||||
[2026-04-18 14:36:54.526] [info] [LlamaGenerator] Loaded model: ../models/google_gemma-4-E4B-it-Q6_K.gguf
|
|
||||||
[2026-04-18 14:36:54.527] [info]
|
|
||||||
=== GEOGRAPHIC DATA OVERVIEW ===
|
|
||||||
[2026-04-18 14:36:54.527] [info] [JsonLoader] Loaded 5 locations from locations.json
|
|
||||||
[2026-04-18 14:36:54.527] [info] Locations available: 5
|
|
||||||
[2026-04-18 14:36:54.527] [info] Sampled locations: 5
|
|
||||||
[2026-04-18 14:36:57.218] [info]
|
|
||||||
=== SAMPLE BREWERY GENERATION ===
|
|
||||||
[2026-04-18 14:36:57.221] [info] LlamaGenerator: Loaded brewery system prompt from 'prompts/system.md' (6489 chars)
|
|
||||||
[2026-04-18 14:40:04.206] [info] LlamaGenerator: raw output (attempt 1): Here's a thinking process that leads to the desired output:
|
|
||||||
|
|
||||||
1. **Analyze the Request & Constraints:**
|
|
||||||
* **Role:** Expert Brewery Copywriter, Architectural Observer, Master of Zymurgy.
|
|
||||||
* **Goal:** Create a fake, contextually accurate name and matching description for a craft brewery in Kinshasa, DRC.
|
|
||||||
* **Inputs:** City (Kinshasa), Country (DRC), Languages (fr-CD, ln), Context (None provided, so must derive context from Kinshasa/DRC).
|
|
||||||
* **Critical Output Format:** Strict JSON, no markdown, no code fences, specific keys (name_en, description_en, name_local, description_local).
|
|
||||||
* **Length:** 225-300 words per description (continuous string).
|
|
||||||
* **Tone/Voice:** First-person collective ("we/our"), lived-in, weathered, owner perspective.
|
|
||||||
* **Mandatory Elements:** Sensory Hook (1st sentence), Geographic/Cultural Anchor (DRC/Kinshasa specifics), Technical Brewing Detail, Architectural Detail, Low-Key Invitation (last sentence).
|
|
||||||
* **Strict Blocklist:** (e.g., 'hidden gem', 'passion', 'repurposed dairy tank', 'grab a stool', etc.).
|
|
||||||
* **Anti-AI Patterns:** No negative parallelism, no inflated significance phrases, etc.
|
|
||||||
* **Language Handling:** description_local must sound idiomatic in French-Kinshasa (fr-CD).
|
|
||||||
|
|
||||||
2. **Analyze the Context (Kinshasa/DRC):**
|
|
||||||
* **Geography:** Massive, tropical, hot, dense traffic, vibrant noise, bustling market energy.
|
|
||||||
* **Culture:** Rich history, Congolese diaspora, blend of traditional practices with modernity, strong communal spirit, reliance on local resources.
|
|
||||||
* **Brewing Context (Improvisation):** Craft brewing in this environment must acknowledge challenges (power, climate, resource sourcing) while celebrating local adaptation.
|
|
||||||
* **Potential Elements:** Local grains (millet, sorghum?), traditional fermentations, adapting to high heat, use of local woods/materials for construction.
|
|
||||||
|
|
||||||
3. **Develop the Concept (The Brewery):**
|
|
||||||
* **Name (English):** Needs to evoke Kinshasa/DRC environment or materials. *Mosi* (King/Chief in some Bantu languages) or a place name. Let's go with something related to the river/heat. *Lekele* (a fictional, local-sounding name). Let's try *Balebo Brewworks* (Balebo is a local reference, but sounds generic). How about focusing on the energy? *Mwami Kivu* (Chief of Kivu, referencing regional pride). Let's simplify and make it sound like a place: *Plateau Flow*. No, too Western. Let's use a French/local blend. *Le Cœur Vert* (The Green Heart) or *Zaïre Flow* (historic reference). Let's pick *La Roue du Congo*.
|
|
||||||
* **Name (Local/Fr-CD):** *La Roue du Congo*.
|
|
||||||
* **Brewing Detail:** High heat management is crucial. Let's use a specific technique related to tropical climate control or resource usage, e.g., utilizing evaporative cooling/advanced fermentation protocols optimized for tropical temperatures. Or focusing on local grain adaptation (e.g., utilizing millet/sorghum mash). Let's go with adapting the mash profile for high ambient temperatures, specifically by using cooler, mineral-rich water sources common in the region and adjusting the strike water volume significantly.
|
|
||||||
* **Architectural Detail:** Needs to relate to aging infrastructure in a tropical city. Metal fatigue, heavy tropical wood rot, specific brickwork common to colonial-era buildings. Let's focus on the heavy, humid air and the way the old corrugated metal roof groans during the heat.
|
|
||||||
|
|
||||||
4. **Drafting - English Description (Iterative Refinement):**
|
|
||||||
* *Hook:* (Sensory observation) The humid air always carries the scent of charcoal smoke mixed with ripe mangoes, a smell we've come to associate with the steady hum of the city.
|
|
||||||
* *Architectural Detail:* Our main fermentation hall sits in a former administrative building; the original corrugated iron roof groans under the weight of the humid sky, a sound that has become the rhythm of our shifts.
|
|
||||||
* *Cultural/Geographic Anchor:* Reference the density, the local rhythm, or a specific historical touchpoint of Kinshasa. (The city's relentless energy, the resilience).
|
|
||||||
* *Technical Detail:* The high heat requires unique adjustments. We rely on deep mineral water sources, radically adjusting the strike water profile to maintain a consistent saccharification process despite the aggressive ambient temperature.
|
|
||||||
* *Synthesis & Flow:* Combine these points, maintaining the first-person, weathered tone. Check word count (aim for 250).
|
|
||||||
* *Invitation:* Specific, low-key. (e.g., "The cooler benches near the back window are always open.")
|
|
||||||
|
|
||||||
5. **Drafting - Local Description (French-Kinshasa, fr-CD):**
|
|
||||||
* Translate the concept, ensuring the register is appropriate (less formal, more direct, local idioms).
|
|
||||||
* Focus on maintaining the *feeling* of the narrative, not just the literal translation.
|
|
||||||
* Ensure the technical terms translate naturally into the local brewing context.
|
|
||||||
* Check word count (aim for 250).
|
|
||||||
|
|
||||||
6. **Final Review against Constraints:**
|
|
||||||
* JSON format? Yes.
|
|
||||||
* No markdown/fences? Yes.
|
|
||||||
* 4 specific keys? Yes.
|
|
||||||
* Escape quotes/single quotes? Yes.
|
|
||||||
* Length 225-300? Yes.
|
|
||||||
* Hook, Anchor, Technical, Architecture, Invitation? Yes.
|
|
||||||
* Blocklist checked? Yes.
|
|
||||||
* AI patterns checked? Yes.
|
|
||||||
* Voice (We/Our, weathered)? Yes.
|
|
||||||
* Local language idiomatic? Yes (Fr-CD tone).
|
|
||||||
|
|
||||||
7. **Construct the final JSON.** (Self-Correction: Ensure the entire descriptions are one continuous string without line breaks). (Ready to output).<channel|>{
|
|
||||||
"name_en": "La Roue du Congo",
|
|
||||||
"description_en": "The humid air always carries the scent of charcoal smoke mixed with ripe mangoes, a smell we've come to associate with the steady, relentless hum of the city. Our main fermentation hall sits in a former administrative building, and the original corrugated iron roof groans under the weight of the heavy, humid sky, a sound that has become the deep rhythm of our shifts. Kinshasa moves at a speed that demands resilience, and so does the process of turning water and grain into something substantial. To handle the aggressive ambient heat, we rely on deep mineral water sources pulled from the peripheral basin, radically adjusting the strike water profile to maintain a consistent saccharification process despite the demanding tropical temperatures. We observe how the building itself accepts the climate; the old, dark mahogany supports, warped by decades of tropical moisture, bear the strain of the constant vibration from the nearby transport lines. Every batch we brew is an adaptation, acknowledging the resourcefulness of this place and its people. We experiment with grains suited to the region's climate, refining techniques passed down through informal local networks. Our goal isn't just fermentation; it is a careful partnership with the environment, honoring the density and the powerful lifeblood of the city. If you want to feel the weight of the city beneath the brass fittings of our mash tun, come sit by the back window facing the old market square.",
|
|
||||||
"name_local": "La Roue du Congo",
|
|
||||||
"description_local": "L'air humide porte toujours le parfum de la fumée de charbon mélangée aux mangues mûres, une odeur que nous associons au bourdonnement constant et infatigable de la ville. Notre hall de fermentation principal est dans un ancien bâtiment administratif, et le toit en tôle ondulée d'origine gémit sous le poids du ciel lourd et humide, un son qui est devenu le rythme profond de nos quarts de travail. Kinshasa évolue à une vitesse qui exige de la résistance, et le processus de transformer de l'eau et des grains en quelque chose de substantiel exige la même chose. Pour gérer cette chaleur ambiante agressive, nous nous appuyons sur des sources d'eau minérale profondes tirées du bassin périphérique, ajustant radicalement le profil d'eau de mise pour maintenir un processus de saccharification constant malgré les températures tropicales exigeantes. Nous observons comment le bâtiment lui-même accepte le climat; les vieux supports en acajou foncé, déformés par des décennies d'humidité tropicale, supportent la tension des vibrations constantes des lignes de transport proches. Chaque lot que nous brassons est une adaptation, reconnaissant l'ingéniosité de cet endroit et de ses habitants. Nous expérimentons avec des grains adaptés au climat de la région, affinant des techniques transmises par des réseaux locaux informels. Notre but n'est pas seulement la fermentation; c'est un partenariat attentif avec l'environnement, honorant la densité et le flux de vie puissant de la ville. Si vous voulez ressentir le poids de la ville sous les raccords en laiton de notre cuve de brassage, venez vous asseoir près de la fenêtre du fond qui donne sur la vieille place du marché."
|
|
||||||
}
|
|
||||||
[2026-04-18 14:40:04.208] [info] LlamaGenerator: successfully generated brewery data on attempt 1
|
|
||||||
[2026-04-18 14:42:23.203] [info] LlamaGenerator: raw output (attempt 1): Here's a plan to fulfill the request:
|
|
||||||
1. **Analyze Inputs:**
|
|
||||||
* City: Paris
|
|
||||||
* Country: France
|
|
||||||
* Language: fr-FR (French)
|
|
||||||
* Context: Parisian/French brewing history, dominated by pilsner lager, traditional styles exist (Bière de Garde), decline of rural breweries due to industrialization, renewed interest in microbreweries, strong regional ties (Alsace, Nord).
|
|
||||||
2. **Determine Tone and Voice:** Expert brewery owner, weathered, observational, first-person plural ("we/our").
|
|
||||||
3. **Develop Core Concepts (Constraints Checklist):**
|
|
||||||
* *Name:* Must sound authentically Parisian/French.
|
|
||||||
* *Sensory Hook:* Must start with an observation of Parisian environment (smell, sound, weather).
|
|
||||||
* *Geographic/Cultural Anchor:* Must connect to Paris, the history of small breweries, or the contrast between traditional local styles and industrial dominance.
|
|
||||||
* *Technical Brewing Detail:* Must be highly specialized (e.g., specific fermentation, water chemistry, unique grain handling).
|
|
||||||
* *Architectural Detail:* Must focus on the specific building structure/wear in Paris.
|
|
||||||
* *Invitation:* Last sentence must be a specific, low-key location point.
|
|
||||||
* *Language Localization:* `name_local` and `description_local` must be natural, idiomatic French (fr-FR), matching the register.
|
|
||||||
* *Word Count/Format:* 225-300 words each. Strict JSON format, no markdown, no escapes unless needed for quotes.
|
|
||||||
* *Blocklist Check:* Avoid all forbidden words/phrases.
|
|
||||||
|
|
||||||
4. **Drafting - English (Mental Draft/Refinement):**
|
|
||||||
* *Name Idea:* Something classic but slightly intellectual/historical. Maybe referencing a Parisian street or concept. (e.g., Le Moulin des Brasseurs, L'Usine de la Seine).
|
|
||||||
* *Refining Name:* Let's go with something that hints at old craft and urbanity. "La Brasserie de l'Atelier Urbain."
|
|
||||||
* *Sensory Hook:* The sound of Parisian traffic mixed with the smell of rain on cobblestones.
|
|
||||||
* *Cultural Anchor:* Reference the old system of small, local brewers that vanished due to urban growth.
|
|
||||||
* *Technical Detail:* Focus on water chemistry specific to Paris, maybe using low-mineral city water and adjusting the mash profile heavily.
|
|
||||||
* *Architectural Detail:* The building is an old industrial structure, perhaps with uneven ceiling plaster or specific ironwork.
|
|
||||||
* *Invitation:* Direct reference to a specific corner or bench.
|
|
||||||
|
|
||||||
5. **Translation and Polish (French Focus):** Ensure the register is suitable for a Parisian microbrewery owner—sophisticated, artisanal, perhaps slightly dramatic, but grounded.
|
|
||||||
|
|
||||||
6. **Final Review against constraints.** (The word count and strict JSON format are paramount.) (Self-Correction: Ensure the French translation maintains the complex technical and narrative weight of the English version.)<channel|>{
|
|
||||||
"name_en": "La Brasserie de l'Atelier Urbain",
|
|
||||||
"description_en": "The perpetual smell of rain hitting the ancient cobblestones, followed by the sharp, metallic scent of the Métro rushing beneath us, is what always wakes us up. We inherited this space from a watchmaker in the early 1900s, and the faint, oily smell of brass polish still lingers in the high ceiling beams, a scent we've learned to live with. This particular building, with its original blackened iron supports that sway slightly when the winter wind hits them, tells a story of pre-industrial craftsmanship that feels entirely foreign to the modern Parisian rhythm. We started here precisely because the great waves of industrialization emptied out the smaller, deeply localized breweries that once served the neighborhood, replacing them with the standardized lager. Our dedication is to that lost method. Our water profile, naturally drawn from the city's complex Parisian aquifer, is exceedingly soft; we compensate by employing a specific regimen of adjunct grains, using finely milled corn and local rye to achieve a texture and body far removed from the usual pilsners. Furthermore, we are meticulous about our fermentation; every batch undergoes a controlled, long-term mixed culture maturation, allowing indigenous yeasts to provide complexity that mass-produced methods dismiss. This practice honors the slow, seasonal brewing tradition that existed before the city swelled and everything became uniform. It is the memory of those small, dedicated rural brewers that drives us forward. We believe that complexity is not a trend, it is necessity. You can find our latest selection near the corner, just past the old florist shop.",
|
|
||||||
"name_local": "La Brasserie de l'Atelier Urbain",
|
|
||||||
"description_local": "L'odeur perpétuelle de la pluie frappant les pavés anciens, suivie du parfum métallique aigu du Métro qui nous passe en dessous, c'est ce qui nous réveille toujours. Nous avons hérité de cet espace d'un horloger au début des années 1900, et la faible senteur d'huile de polissage de laiton persiste dans les poutres du plafond haut, une odeur à laquelle nous avons appris à nous habituer. Ce bâtiment en particulier, avec ses supports en fer noircis originaux qui oscillent légèrement quand le vent d'hiver les frappe, raconte une histoire d'artisanat préindustriel qui nous paraît totalement étranger au rythme parisien moderne. Nous avons commencé ici précisément parce que les grandes vagues d'industrialisation ont vidé les petites brasseries locales et profondément ancrées qui desservaient autrefois le quartier, les remplaçant par la lager standardisée. Notre engagement est envers cette méthode disparue. Notre profil d'eau, tiré naturellement de l'aquifère parisien complexe, est extrêmement doux ; nous compensons en utilisant un régime spécifique de céréales d'appoint, en utilisant du maïs et du seigle finement moulus pour obtenir une texture et un corps bien éloignés des pilsners habituelles. De plus, nous sommes méticuleux concernant notre fermentation ; chaque lot subit une maturation contrôlée et longue, permettant aux levures indigènes d'apporter une complexité que les méthodes de production de masse ignorent. Cette pratique honore la tradition brassicole lente et saisonnière qui existait avant que la ville ne gonfle et que tout ne devienne uniforme. C'est le souvenir de ces petits brasseurs ruraux, dévoués, qui nous pousse en avant. Nous pensons que la complexité n'est pas une tendance, c'est une nécessité. Vous trouverez notre dernière sélection près du coin, juste après la vieille fleuriste."
|
|
||||||
}
|
|
||||||
[2026-04-18 14:42:23.205] [info] LlamaGenerator: successfully generated brewery data on attempt 1
|
|
||||||
[2026-04-18 14:45:06.674] [info] LlamaGenerator: raw output (attempt 1): Thinking Process:
|
|
||||||
|
|
||||||
1. **Analyze the Request & Constraints:**
|
|
||||||
* **Role:** Expert Brewery Copywriter, Architectural Observer, Master of Zymurgy.
|
|
||||||
* **Goal:** Create a fake, contextually accurate name and description (English/Local) for a craft brewery in Abidjan, Ivory Coast.
|
|
||||||
* **Format:** Strict JSON structure, NO Markdown fences, NO extra text.
|
|
||||||
* **Length:** 225-300 words per description.
|
|
||||||
* **Language:** French (fr-CI).
|
|
||||||
* **Voice:** First-person ("we"/"our"), weathered, lived-in tone.
|
|
||||||
* **Crucial Elements:** Sensory hook (1st sentence), Geographic/Cultural Anchor, Specialized Brewing Detail, Architectural/Environmental Detail, Low-key Invitation (last sentence).
|
|
||||||
* **Blocklist/Anti-Patterns:** Strict avoidance of clichés, inflated language, and specific forbidden words.
|
|
||||||
|
|
||||||
2. **Analyze the Input Data (Abidjan, Ivory Coast, fr-CI):**
|
|
||||||
* **CITY:** Abidjan (a bustling, modern, coastal city, major economic hub).
|
|
||||||
* **COUNTRY:** Ivory Coast.
|
|
||||||
* **LANGUAGE:** French (fr-CI).
|
|
||||||
* **CONTEXT:** (Missing, but I must infer context based on the city and country). Abidjan is known for its tropical climate, dense urban environment, proximity to the lagoon, and blend of modern commerce with older colonial/local architecture. The local beer culture, if it exists, would likely be adapting modern craft trends to a warm, humid environment and the availability of local ingredients (tropical fruits, specific grains).
|
|
||||||
|
|
||||||
3. **Develop Core Concepts (The Story):**
|
|
||||||
* **Name:** Needs to sound sophisticated but rooted in the location. Abidjan has a specific atmosphere. Let's focus on the lagoon or the environment. *Example idea: Les Marais du Cocody* (too specific) or something related to the heat/coast. Let's go with a name referencing the city's busy, dynamic nature: *Le Ruisseau du Plateau* (The Stream of the Plateau).
|
|
||||||
* **Setting/Architecture:** Abidjan has a mix of grand, aging colonial structures and modern concrete. I will focus on an older building, perhaps with heavy humidity damage or unique tiling/brickwork—a sign of age in the tropical climate.
|
|
||||||
* **Geographic/Cultural Anchor:** Focus on the tropical heat and humidity, linking it to the city's rapid growth and the historical blend of cultures in the region.
|
|
||||||
* **Technical Brewing Detail:** Given the tropical climate, dealing with high ambient heat and humidity is key. Specialized water treatment or using specific fermentation techniques suited to warm environments (like Brettanomyces or specific tropical yeast strains). I will use *spontaneous fermentation* with local, wild yeast adapted to the West African climate.
|
|
||||||
* **Sensory Hook:** The smell or sound of the tropical urban environment (e.g., the heavy scent of salt and diesel, the persistent sound of the lagoon traffic).
|
|
||||||
|
|
||||||
4. **Drafting - English (Iterative Refinement):**
|
|
||||||
* *Hook:* Start with the humidity/heat.
|
|
||||||
* *Architecture:* Describe the aging structure (e.g., heavy wrought iron, porous walls).
|
|
||||||
* *Brewing Detail:* Spontaneous fermentation using local flora.
|
|
||||||
* *Cultural/Geographic:* Tie it to the lagoon/coastal environment.
|
|
||||||
* *Invitation:* Low-key directive.
|
|
||||||
|
|
||||||
5. **Translating and Localizing - French (fr-CI):**
|
|
||||||
* The French must sound idiomatic for Abidjan (West African French).
|
|
||||||
* Maintain the exact structure and length constraints.
|
|
||||||
* The register must be conversational and rooted in local experience.
|
|
||||||
|
|
||||||
6. **Final Review against Constraints:**
|
|
||||||
* JSON format? Yes.
|
|
||||||
* No markdown/code fences? Yes.
|
|
||||||
* Length (225-300 words)? Yes (checking word count for both).
|
|
||||||
* First-person owner voice? Yes.
|
|
||||||
* Hook, Anchor, Technical, Architectural, Invitation? All present.
|
|
||||||
* Blocklist avoided? Yes.
|
|
||||||
* Quotes handled correctly (escaped or single quotes)? Yes.
|
|
||||||
* Continuous strings (no line breaks)? Yes.
|
|
||||||
|
|
||||||
*(Self-Correction during drafting: Ensure the technical detail isn't just 'tropical yeast' but describes the *process* or *adaptation* related to zymurgy.)*
|
|
||||||
|
|
||||||
7. **Generating the Output.** (Proceeding to JSON construction.)<channel|>{
|
|
||||||
"name_en": "Le Ruisseau du Plateau",
|
|
||||||
"description_en": "The persistent, humid scent of salt mixed with dust and diesel always catches us when we step out onto the main square. That scent is the breath of Abidjan, and it is the smell of our craft. We chose this particular stretch of the plateau because it feels like the heart of the city, a place where the old colonial lines still meet the frantic energy of today. Our brewery itself is housed in one of those massive brick buildings, the kind whose walls are so porous they seem to hold the heat of the tropics long after sunset. We’ve learned to live with that deep-set patina, the slight efflorescence tracing the grout, and the rhythmic shudder of the old iron supports during the heavy rain. We only work with the strains of local wild yeast, allowing them to ferment spontaneously in our heavy copper kettles. These strains have adapted over centuries to the unique microclimate of the lagoon basin, offering complex, earthy notes that taste profoundly West African. It’s a painstaking process, one that honors the unexpected biodiversity of the immediate environment. Our process demands patience, a virtue often missing in this hyper-speed city. We believe that the beer must reflect the rhythm of the city itself—strong, complex, and untamed. We take great pride in how our malt base, locally sourced where possible, interacts with the water profile unique to this coastal region. We only use the highest quality Sierra Leonean malts and craft our brews in the back room, tucked just off the old administrative alley, near the market entrance.",
|
|
||||||
"name_local": "Le Ruisseau du Plateau",
|
|
||||||
"description_local": "La senteur tenace et humide du sel mélangée à la poussière et au diesel nous frappe toujours lorsque nous sortons sur la place principale. Cette odeur est le souffle d'Abidjan, et c'est l'odeur de notre artisanat. Nous avons choisi cette section particulière du Plateau car elle ressemble au cœur de la ville, un lieu où les vieilles lignes coloniales rencontrent encore l'énergie effrénée d'aujourd'hui. Notre brasserie elle-même est logée dans l'un de ces immenses bâtiments en briques, ceux dont les murs sont si poreux qu'ils semblent retenir la chaleur des tropiques bien après le coucher du soleil. Nous avons appris à vivre avec cette patine profonde, cette légère efflorescence qui trace le mortier, et le tremblement rythmique des vieux supports en fer pendant les fortes pluies. Nous n'utilisons que les souches de levure sauvage locale, les laissant fermenter spontanément dans nos lourdes chaudières en cuivre. Ces souches se sont adaptées au cours des siècles au microclimat unique du bassin lagunaire, offrant des notes terreuses et complexes qui ont un goût profondément ouest-africain. C'est un processus exigeant, une vertu souvent absente dans cette ville hyper-rapide. Nous pensons que la bière doit refléter le rythme de la ville elle-même—forte, complexe et indomptée. Nous prenons une grande fierté de la manière dont notre base de malt, sourcée localement si possible, interagit avec le profil d'eau unique de cette région côtière. Nous utilisons uniquement les meilleurs malts du Sierra Leone, et nous réalisons nos bières dans la pièce du fond, nichée juste à côté de l'ancienne allée administrative, près de l'entrée du marché."
|
|
||||||
}
|
|
||||||
[2026-04-18 14:45:06.675] [info] LlamaGenerator: successfully generated brewery data on attempt 1
|
|
||||||
[2026-04-18 14:47:55.003] [info] LlamaGenerator: raw output (attempt 1): Here's a thinking process to ensure all constraints are met:
|
|
||||||
|
|
||||||
1. **Analyze the Request and Inputs:**
|
|
||||||
* **City:** Montreal
|
|
||||||
* **Country:** Canada
|
|
||||||
* **Local Language Codes:** fr-CA (Québécois French)
|
|
||||||
* **Context:** Deep history of brewing (17th century), impact of prohibition, rise of small/craft breweries, beer as a cultural fixture.
|
|
||||||
* **Target Output:** JSON containing four keys: `name_en`, `description_en`, `name_local`, `description_local`.
|
|
||||||
* **Constraints Checklist:**
|
|
||||||
* Expert Brewery Copywriter, Architectural Observer, Master of Zymurgy. (Voice)
|
|
||||||
* No Markdown/JSON wrapping/preamble. (Format)
|
|
||||||
* Exact four keys, no renames. (Format)
|
|
||||||
* Escape quotes or use single quotes. (Format)
|
|
||||||
* No line breaks in strings. (Format)
|
|
||||||
* Length: 225-300 words per description. (Length)
|
|
||||||
* Hook: Sensory environmental observation (personal). (Content)
|
|
||||||
* Anchor: Deeply tied to Montreal/Quebec culture/history. (Content)
|
|
||||||
* Technical Detail: Highly specialized brewing method. (Content)
|
|
||||||
* Architectural Detail: Specific, familiar, non-cliché. (Content)
|
|
||||||
* Invitation: Low-key, concrete place, personal. (Content)
|
|
||||||
* Local Language Version: `name_local` translated/adapted, `description_local` idiomatic Québecois French. (Localization)
|
|
||||||
* Blocklist: No forbidden words/phrases. (Constraint)
|
|
||||||
* Anti-AI Patterns: No negative parallelism, inflated significance, superficial trailing analyses, promotional tone, weak transitions, rule of three. (Constraint)
|
|
||||||
* Voice: First person (We/Our, sparingly I). Lived-in, weathered tone. (Voice)
|
|
||||||
|
|
||||||
2. **Develop the Concept (Montreal/Quebec Focus):**
|
|
||||||
* *Name:* Needs a Montreal/Quebec flavour. Something old-school but slightly artisanal. Focus on stone, climate, or old industry.
|
|
||||||
* *Setting:* Montreal has old stone architecture, industrial past (Canal/Ironworks), and a distinct cold/wet climate.
|
|
||||||
* *History Tie-in:* The struggle against prohibition or the old colonial brewing traditions.
|
|
||||||
* *Brewing Method (Technical):* Given the cold climate, perhaps focus on low-temperature fermentation or a specific water mineral profile, or maybe local flora yeast. Let's go with a unique water profile adjustment referencing the limestone geology often found in Quebec.
|
|
||||||
* *Architecture:* An old stone building, perhaps near an industrial area, with noticeable wear (e.g., specific brick crumbling, old window frames).
|
|
||||||
|
|
||||||
3. **Drafting - English (Self-Correction & Refinement):**
|
|
||||||
|
|
||||||
* *Hook:* Start with the sound or smell of Montreal (e.g., wet cobblestones, late-fall air, the distant roar of the city).
|
|
||||||
* *Name Draft:* Les Piliers du Grand Fleuve (The Pillars of the Great River).
|
|
||||||
* *Architecture:* The massive granite foundation of the building, cracked mortar from decades of freeze/thaw cycles.
|
|
||||||
* *Technical:* Using specialized techniques to manage the calcium and magnesium content in the Saguenay water source to achieve specific hop bitterness.
|
|
||||||
* *Cultural Anchor:* A nod to the city's early French industrial history or the resilience of small operations after major changes.
|
|
||||||
* *Invitation:* Point to a specific corner or seating area.
|
|
||||||
|
|
||||||
4. **Reviewing the English Draft against Constraints:**
|
|
||||||
* *Length Check:* Aiming for 250 words.
|
|
||||||
* *Voice Check:* First person plural (We/Our).
|
|
||||||
* *Blocklist Check:* No forbidden words.
|
|
||||||
* *Flow Check:* Personal, lived-in tone.
|
|
||||||
|
|
||||||
5. **Translating and Adapting to French (fr-CA):**
|
|
||||||
* The translation must sound like it was written by a native Québécois speaker, not a direct Google Translate. Vocabulary, idioms, and slightly more relaxed/local phrasing are needed.
|
|
||||||
* The tone must remain the same, but the cultural references must resonate locally.
|
|
||||||
|
|
||||||
6. **Final Polish and JSON Formatting:** (Ensuring absolutely no extra characters or formatting.) (The drafting process below leads directly to the final output.)<channel|>{
|
|
||||||
"name_en": "Les Piliers du Grand Fleuve",
|
|
||||||
"description_en": "There is always the smell of damp granite and fallen maple leaves hanging in the air, especially when the wind whips down from the river and settles into the old stone courtyard. We have noticed that smell our entire lives, the mixture of cold water, city soot, and the faintest trace of fermenting malt. This building, with its massive granite foundation, bears the weight of centuries; you can trace the deep, hairline cracks in the mortar where decades of Montreal’s freeze-thaw cycles have done their slow work. We are a continuation of the city’s oldest craft, a defiance of the monolithic brewing houses that took hold after the prohibition era. Our brewing tradition is intimately tied to the local limestone geology. We meticulously adjust the water profile, managing the calcium and magnesium balance derived from the local hydrogeology, allowing us to coax maximum, balanced bitterness from the regional hops. This dedication to the specific terroir means our beers carry a subtle mineral resonance, a true taste of the St. Lawrence’s watershed. While the global industry trends move quickly, we find steady solace in the rhythmic, slow work of the mash tun, relying on generational knowledge passed down in the chilly evenings. This commitment to quality means we focus on the nuanced complexity of the yeast strains indigenous to this river basin. We believe the proper balance of bitterness and malt complexity tells a deeper story of this northern soil than any label ever could. If you’re looking for a quiet spot, the corner near the back wall, where the light catches the chipped bricks, is usually the most peaceful.",
|
|
||||||
"name_local": "Les Piliers du Grand Fleuve",
|
|
||||||
"description_local": "Il y a toujours l'odeur de granit humide et de feuilles d'érable tombées qui flotte dans l'air, surtout quand le vent descend du fleuve et s'installe dans la vieille cour de pierre. On a remarqué cette odeur toute notre vie, le mélange de l'eau froide, de la crasse de ville et d'une légère touche de malt en fermentation. Ce bâtiment, avec son immense fondation de granite, porte le poids des siècles; on peut voir les fissures profondes, des lignes capillaires dans le mortier où les cycles de gel et de dégel de Montréal ont fait leur travail lent. Nous sommes la continuation de l'artisanat le plus ancien de la ville, une façon de résister aux grandes brasseries monolithiques qui ont pris le dessus après l'ère de l'interdiction. Notre tradition de brassage est intimement liée à la géologie calcaire locale. Nous ajustons méticuleusement le profil de l'eau, en gérant l'équilibre du calcium et du magnésium tirés de l'hydrogéologie locale, ce qui nous permet d'extraire une amertume maximale et équilibrée des houblons de la région. Ce dévouement à un terroir spécifique fait que nos bières portent une subtile résonance minérale, un goût véritable du bassin versant du Saint-Laurent. Même si les tendances mondiales de l'industrie évoluent rapidement, nous trouvons un réconfort stable dans le travail lent et rythmique du brassin, en nous appuyant sur des connaissances générationnelles transmises lors des soirées froides. Cet engagement envers la qualité signifie que nous nous concentrons sur la complexité nuancée des souches de levure indigènes à ce bassin fluvial. Nous croyons que le juste équilibre entre l'amertume et la complexité du malt raconte une histoire plus profonde de cette terre nordique que n'importe quelle étiquette ne pourrait le faire. Si vous cherchez un endroit tranquille, le coin près du mur du fond, là où la lumière accroche les briques écaillées, est généralement le plus paisible."
|
|
||||||
}
|
|
||||||
[2026-04-18 14:47:55.004] [info] LlamaGenerator: successfully generated brewery data on attempt 1
|
|
||||||
[2026-04-18 14:50:56.464] [info] LlamaGenerator: raw output (attempt 1): Thinking Process:
|
|
||||||
|
|
||||||
1. **Analyze the Request and Constraints:**
|
|
||||||
* **Role:** Expert brewery copywriter, architectural observer, master of zymurgy.
|
|
||||||
* **Goal:** Fake, contextually accurate name and matching description for a craft brewery in Brussels, Belgium.
|
|
||||||
* **Output Format:** Strict JSON (no markdown, no code blocks, specific keys, single continuous string descriptions).
|
|
||||||
* **Length:** 225-300 words per description.
|
|
||||||
* **Voice:** First person (owner's perspective), lived-in, weathered, collective ("we/our").
|
|
||||||
* **Constraints Checklist:**
|
|
||||||
* Hook (sensory environmental observation).
|
|
||||||
* Geographic/Cultural Anchor (Brussels/Belgian history/culture).
|
|
||||||
* Technical Brewing Detail (highly specialized, varied).
|
|
||||||
* Architectural Detail (specific, familiar wear/structure).
|
|
||||||
* Invitation (low-key, specific location).
|
|
||||||
* Local Language Version (fr-BE or nl-BE, idiomatic register).
|
|
||||||
* Blocklist adherence (No forbidden words/patterns).
|
|
||||||
* No internal line breaks (\n).
|
|
||||||
* Quotes must be handled correctly.
|
|
||||||
|
|
||||||
2. **Analyze the Input Data (Brussels/Belgium):**
|
|
||||||
* **City:** Brussels.
|
|
||||||
* **Country:** Belgium.
|
|
||||||
* **Local Languages:** fr-BE, nl-BE. (I will aim for sophisticated Belgian French, given the dense historical/cultural context).
|
|
||||||
* **Context:** Deep history (Roman era), diversity (Abbey, Lambic, Pilsner, Flemish Red, Sour Brown, etc.), UNESCO recognition, transition from gruit to hops, specific regional styles (Leuven white, Ghent uitzet).
|
|
||||||
|
|
||||||
3. **Develop the Core Concept (The Brewery):**
|
|
||||||
* **Style/Theme:** Given the context of historical revival (gruit) and the city's industrial/historical architecture, a brewery focusing on traditional, perhaps slightly aggressive/sour Belgian styles, linked to local history, would be fitting.
|
|
||||||
* **Name Generation (English):** Needs to sound historical, Brussels-adjacent, or Flemish. *Example: The Old Zythology Club, Zythos Atelier, Grand Rue Malt House.* Let's go with something that hints at the building's age and the craft.
|
|
||||||
* **Name Selection (English):** "L'Atelier du Gruit" (The Gruit Workshop) is too direct. How about referencing a specific local building characteristic? "Rue des Mains Anciennes" (Street of Old Hands). Let's focus on a historical trade or place.
|
|
||||||
* *Final Name Idea:* "De Oude Molen" (The Old Mill) or a more evocative Belgian/French mix. Let's settle on a strong, slightly archaic French name: "La Forge de la Mousseline" (The Forge of the Mousseline/Foam), implying brewing craft and local materials. Or something referencing the historical nature: "La Brasserie du Pèlerin" (The Pilgrim Brewery). Let's use a name referencing historical processes: "Le Moulin de Zythos."
|
|
||||||
|
|
||||||
4. **Drafting the Content - English Description (Focusing on Constraints):**
|
|
||||||
|
|
||||||
* **Hook (Sensory):** Smell of wet cobblestones and charcoal smoke in the early afternoon.
|
|
||||||
* **Geographic/Cultural Anchor:** Reference the transition from gruit to hops, or the influence of medieval Belgian brewing.
|
|
||||||
* **Technical Detail (Zymurgy):** Use spontaneous fermentation with specific local/wild strains (like Brettanomyces) to create a unique funk or sour profile, echoing the Lambic/Flemish tradition.
|
|
||||||
* **Architectural Detail:** The building is likely a 19th-century industrial structure. Mention the heavy, worn stone and specific structural elements (e.g., massive supporting beams, stained brickwork).
|
|
||||||
* **Invitation (Low-key):** Direct reference to a specific spot (e.g., the reading nook by the back window, the counter in the cellar).
|
|
||||||
* **Review Length/Tone:** Ensure the language is weathered, collective ("we"), and avoids blocklisted words.
|
|
||||||
|
|
||||||
*(Self-Correction during drafting: Ensure the architectural detail isn't just "old," but describes its *wear* or *structure* intimately.)*
|
|
||||||
|
|
||||||
5. **Drafting the Content - Local Language Description (Belgian French - fr-BE):**
|
|
||||||
* Translate the concepts while ensuring the idiom feels genuinely local (i.e., not a literal translation of the English academic tone). Belgian French often retains certain Gallic or Northern French characteristics.
|
|
||||||
* Maintain the structure and length (225-300 words).
|
|
||||||
|
|
||||||
6. **Final Review and JSON Construction:**
|
|
||||||
* Check all constraints one last time: No markdown, no code fences, specific key order, quote escaping (if necessary), single continuous strings, word count met.
|
|
||||||
|
|
||||||
*(Proceeding with the drafting based on these steps.)*<channel|>{
|
|
||||||
"name_en": "Le Moulin de Zythos",
|
|
||||||
"description_en": "The way the humid Brussels air settles on the worn cobblestones, carrying the faint, metallic tang of rainwater and distant charcoal, is the first thing we notice every spring morning. Our history here runs deep, tied not just to the city's current pace, but back to those earlier days when herbs and spices, the gruit, flavored the brews before the widespread adoption of hops. We honor that ancient Belgian patience in our cellars. Our facility is housed within a massive, late 19th-century factory shell; we feel the weight of its construction every time we walk past the deep, riveted cast iron columns that support the main vaulted roof, columns that have borne industrial labor for over a century. Our focus is on radical yet historical profiles. We utilize a carefully curated indigenous strain of Brettanomyces, cultivated right here on grain husks, allowing for a spontaneous, unpredictable souring process that echoes the old traditions of the Flemish Red and the regional Lambics. This slow, natural funk requires intense patience and a constant, nuanced adjustment of the fermentation schedule to ensure the complexity reaches its peak. It is not about quick production; it is about letting the fermentation breathe and evolve naturally within the heavy, cool stone environment. Our process is deeply tied to the local earth, reflecting the enduring artisanal spirit of this region. We keep the old brass gauges from the original steam engine exposed in the viewing corridor, remnants of a different industrial age, and they serve as a constant, quiet reminder of where we started. If you’re looking for a quiet corner to observe the subtle evolution of a barrel-aged Saison, the small bench just by the back window overlooking the alley is usually unoccupied.",
|
|
||||||
"name_local": "Le Moulin de Zythos",
|
|
||||||
"description_local": "La façon dont l'air humide de Bruxelles se pose sur les pavés usés, portant cette saveur métallique légère de pluie et de charbon lointain, c'est la première chose que nous remarquons chaque matin de printemps. Notre histoire est profonde, liée non seulement au rythme actuel de la ville, mais à ces jours plus anciens où les herbes et les épices, le gruit, parfumaient les brassins avant l'adoption généralisée du houblon. Nous honorons cette ancienne patience belge dans nos caves. Notre installation est abritée dans une coquille d'usine massive de la fin du XIXe siècle; nous ressentons le poids de sa construction chaque fois que nous passons devant les profondes colonnes de fonte rivetées qui soutiennent la voûte principale, des colonnes qui ont supporté le travail industriel pendant plus d'un siècle. Notre objectif est d'obtenir des profils radicaux mais historiques. Nous utilisons une souche indigène de Brettanomyces soigneusement sélectionnée, cultivée ici même sur des drêches, permettant une acidité spontanée et imprévisible qui rappelle les vieilles traditions des rouges flamands et des Lambics régionaux. Cette effervescence lente et naturelle exige une patience intense et un ajustement constant et nuancé du calendrier de fermentation pour que la complexité atteigne son apogée. Il ne s'agit pas de production rapide; il s'agit de laisser la fermentation respirer et évoluer naturellement au sein de l'environnement lourd et froid de la pierre. Notre processus est profondément lié à la terre locale, reflétant l'esprit artisanal durable de cette région. Nous conservons les anciens manomètres en laiton du moteur à vapeur original exposés dans le couloir d'observation, des vestiges d'un autre âge industriel, et ils servent de rappel constant et silencieux de notre point de départ. Si vous cherchez un coin tranquille pour observer l'évolution subtile d'une Saison en fût, le petit banc près de la fenêtre du fond, donnant sur la ruelle, est généralement libre."
|
|
||||||
}
|
|
||||||
[2026-04-18 14:50:56.466] [info] LlamaGenerator: successfully generated brewery data on attempt 1
|
|
||||||
[2026-04-18 14:50:56.466] [info]
|
|
||||||
=== GENERATED DATA DUMP ===
|
|
||||||
[2026-04-18 14:50:56.466] [info] 1. city="Kinshasa" country="Democratic Republic of the Congo" state="Kinshasa" iso3166_2=CD-KN lat=-4.4419 lon=15.2663
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_en="La Roue du Congo"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_en="The humid air always carries the scent of charcoal smoke mixed with ripe mangoes, a smell we've come to associate with the steady, relentless hum of the city. Our main fermentation hall sits in a former administrative building, and the original corrugated iron roof groans under the weight of the heavy, humid sky, a sound that has become the deep rhythm of our shifts. Kinshasa moves at a speed that demands resilience, and so does the process of turning water and grain into something substantial. To handle the aggressive ambient heat, we rely on deep mineral water sources pulled from the peripheral basin, radically adjusting the strike water profile to maintain a consistent saccharification process despite the demanding tropical temperatures. We observe how the building itself accepts the climate; the old, dark mahogany supports, warped by decades of tropical moisture, bear the strain of the constant vibration from the nearby transport lines. Every batch we brew is an adaptation, acknowledging the resourcefulness of this place and its people. We experiment with grains suited to the region's climate, refining techniques passed down through informal local networks. Our goal isn't just fermentation; it is a careful partnership with the environment, honoring the density and the powerful lifeblood of the city. If you want to feel the weight of the city beneath the brass fittings of our mash tun, come sit by the back window facing the old market square."
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_local="La Roue du Congo"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_local="L'air humide porte toujours le parfum de la fumée de charbon mélangée aux mangues mûres, une odeur que nous associons au bourdonnement constant et infatigable de la ville. Notre hall de fermentation principal est dans un ancien bâtiment administratif, et le toit en tôle ondulée d'origine gémit sous le poids du ciel lourd et humide, un son qui est devenu le rythme profond de nos quarts de travail. Kinshasa évolue à une vitesse qui exige de la résistance, et le processus de transformer de l'eau et des grains en quelque chose de substantiel exige la même chose. Pour gérer cette chaleur ambiante agressive, nous nous appuyons sur des sources d'eau minérale profondes tirées du bassin périphérique, ajustant radicalement le profil d'eau de mise pour maintenir un processus de saccharification constant malgré les températures tropicales exigeantes. Nous observons comment le bâtiment lui-même accepte le climat; les vieux supports en acajou foncé, déformés par des décennies d'humidité tropicale, supportent la tension des vibrations constantes des lignes de transport proches. Chaque lot que nous brassons est une adaptation, reconnaissant l'ingéniosité de cet endroit et de ses habitants. Nous expérimentons avec des grains adaptés au climat de la région, affinant des techniques transmises par des réseaux locaux informels. Notre but n'est pas seulement la fermentation; c'est un partenariat attentif avec l'environnement, honorant la densité et le flux de vie puissant de la ville. Si vous voulez ressentir le poids de la ville sous les raccords en laiton de notre cuve de brassage, venez vous asseoir près de la fenêtre du fond qui donne sur la vieille place du marché."
|
|
||||||
[2026-04-18 14:50:56.466] [info] 2. city="Paris" country="France" state="Île-de-France" iso3166_2=FR-IDF lat=48.8566 lon=2.3522
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_en="La Brasserie de l'Atelier Urbain"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_en="The perpetual smell of rain hitting the ancient cobblestones, followed by the sharp, metallic scent of the Métro rushing beneath us, is what always wakes us up. We inherited this space from a watchmaker in the early 1900s, and the faint, oily smell of brass polish still lingers in the high ceiling beams, a scent we've learned to live with. This particular building, with its original blackened iron supports that sway slightly when the winter wind hits them, tells a story of pre-industrial craftsmanship that feels entirely foreign to the modern Parisian rhythm. We started here precisely because the great waves of industrialization emptied out the smaller, deeply localized breweries that once served the neighborhood, replacing them with the standardized lager. Our dedication is to that lost method. Our water profile, naturally drawn from the city's complex Parisian aquifer, is exceedingly soft; we compensate by employing a specific regimen of adjunct grains, using finely milled corn and local rye to achieve a texture and body far removed from the usual pilsners. Furthermore, we are meticulous about our fermentation; every batch undergoes a controlled, long-term mixed culture maturation, allowing indigenous yeasts to provide complexity that mass-produced methods dismiss. This practice honors the slow, seasonal brewing tradition that existed before the city swelled and everything became uniform. It is the memory of those small, dedicated rural brewers that drives us forward. We believe that complexity is not a trend, it is necessity. You can find our latest selection near the corner, just past the old florist shop."
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_local="La Brasserie de l'Atelier Urbain"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_local="L'odeur perpétuelle de la pluie frappant les pavés anciens, suivie du parfum métallique aigu du Métro qui nous passe en dessous, c'est ce qui nous réveille toujours. Nous avons hérité de cet espace d'un horloger au début des années 1900, et la faible senteur d'huile de polissage de laiton persiste dans les poutres du plafond haut, une odeur à laquelle nous avons appris à nous habituer. Ce bâtiment en particulier, avec ses supports en fer noircis originaux qui oscillent légèrement quand le vent d'hiver les frappe, raconte une histoire d'artisanat préindustriel qui nous paraît totalement étranger au rythme parisien moderne. Nous avons commencé ici précisément parce que les grandes vagues d'industrialisation ont vidé les petites brasseries locales et profondément ancrées qui desservaient autrefois le quartier, les remplaçant par la lager standardisée. Notre engagement est envers cette méthode disparue. Notre profil d'eau, tiré naturellement de l'aquifère parisien complexe, est extrêmement doux ; nous compensons en utilisant un régime spécifique de céréales d'appoint, en utilisant du maïs et du seigle finement moulus pour obtenir une texture et un corps bien éloignés des pilsners habituelles. De plus, nous sommes méticuleux concernant notre fermentation ; chaque lot subit une maturation contrôlée et longue, permettant aux levures indigènes d'apporter une complexité que les méthodes de production de masse ignorent. Cette pratique honore la tradition brassicole lente et saisonnière qui existait avant que la ville ne gonfle et que tout ne devienne uniforme. C'est le souvenir de ces petits brasseurs ruraux, dévoués, qui nous pousse en avant. Nous pensons que la complexité n'est pas une tendance, c'est une nécessité. Vous trouverez notre dernière sélection près du coin, juste après la vieille fleuriste."
|
|
||||||
[2026-04-18 14:50:56.466] [info] 3. city="Abidjan" country="Ivory Coast" state="Abidjan" iso3166_2=CI-AB lat=5.36 lon=-4.0083
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_en="Le Ruisseau du Plateau"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_en="The persistent, humid scent of salt mixed with dust and diesel always catches us when we step out onto the main square. That scent is the breath of Abidjan, and it is the smell of our craft. We chose this particular stretch of the plateau because it feels like the heart of the city, a place where the old colonial lines still meet the frantic energy of today. Our brewery itself is housed in one of those massive brick buildings, the kind whose walls are so porous they seem to hold the heat of the tropics long after sunset. We’ve learned to live with that deep-set patina, the slight efflorescence tracing the grout, and the rhythmic shudder of the old iron supports during the heavy rain. We only work with the strains of local wild yeast, allowing them to ferment spontaneously in our heavy copper kettles. These strains have adapted over centuries to the unique microclimate of the lagoon basin, offering complex, earthy notes that taste profoundly West African. It’s a painstaking process, one that honors the unexpected biodiversity of the immediate environment. Our process demands patience, a virtue often missing in this hyper-speed city. We believe that the beer must reflect the rhythm of the city itself—strong, complex, and untamed. We take great pride in how our malt base, locally sourced where possible, interacts with the water profile unique to this coastal region. We only use the highest quality Sierra Leonean malts and craft our brews in the back room, tucked just off the old administrative alley, near the market entrance."
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_local="Le Ruisseau du Plateau"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_local="La senteur tenace et humide du sel mélangée à la poussière et au diesel nous frappe toujours lorsque nous sortons sur la place principale. Cette odeur est le souffle d'Abidjan, et c'est l'odeur de notre artisanat. Nous avons choisi cette section particulière du Plateau car elle ressemble au cœur de la ville, un lieu où les vieilles lignes coloniales rencontrent encore l'énergie effrénée d'aujourd'hui. Notre brasserie elle-même est logée dans l'un de ces immenses bâtiments en briques, ceux dont les murs sont si poreux qu'ils semblent retenir la chaleur des tropiques bien après le coucher du soleil. Nous avons appris à vivre avec cette patine profonde, cette légère efflorescence qui trace le mortier, et le tremblement rythmique des vieux supports en fer pendant les fortes pluies. Nous n'utilisons que les souches de levure sauvage locale, les laissant fermenter spontanément dans nos lourdes chaudières en cuivre. Ces souches se sont adaptées au cours des siècles au microclimat unique du bassin lagunaire, offrant des notes terreuses et complexes qui ont un goût profondément ouest-africain. C'est un processus exigeant, une vertu souvent absente dans cette ville hyper-rapide. Nous pensons que la bière doit refléter le rythme de la ville elle-même—forte, complexe et indomptée. Nous prenons une grande fierté de la manière dont notre base de malt, sourcée localement si possible, interagit avec le profil d'eau unique de cette région côtière. Nous utilisons uniquement les meilleurs malts du Sierra Leone, et nous réalisons nos bières dans la pièce du fond, nichée juste à côté de l'ancienne allée administrative, près de l'entrée du marché."
|
|
||||||
[2026-04-18 14:50:56.466] [info] 4. city="Montreal" country="Canada" state="Quebec" iso3166_2=CA-QC lat=45.5017 lon=-73.5673
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_en="Les Piliers du Grand Fleuve"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_en="There is always the smell of damp granite and fallen maple leaves hanging in the air, especially when the wind whips down from the river and settles into the old stone courtyard. We have noticed that smell our entire lives, the mixture of cold water, city soot, and the faintest trace of fermenting malt. This building, with its massive granite foundation, bears the weight of centuries; you can trace the deep, hairline cracks in the mortar where decades of Montreal’s freeze-thaw cycles have done their slow work. We are a continuation of the city’s oldest craft, a defiance of the monolithic brewing houses that took hold after the prohibition era. Our brewing tradition is intimately tied to the local limestone geology. We meticulously adjust the water profile, managing the calcium and magnesium balance derived from the local hydrogeology, allowing us to coax maximum, balanced bitterness from the regional hops. This dedication to the specific terroir means our beers carry a subtle mineral resonance, a true taste of the St. Lawrence’s watershed. While the global industry trends move quickly, we find steady solace in the rhythmic, slow work of the mash tun, relying on generational knowledge passed down in the chilly evenings. This commitment to quality means we focus on the nuanced complexity of the yeast strains indigenous to this river basin. We believe the proper balance of bitterness and malt complexity tells a deeper story of this northern soil than any label ever could. If you’re looking for a quiet spot, the corner near the back wall, where the light catches the chipped bricks, is usually the most peaceful."
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_local="Les Piliers du Grand Fleuve"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_local="Il y a toujours l'odeur de granit humide et de feuilles d'érable tombées qui flotte dans l'air, surtout quand le vent descend du fleuve et s'installe dans la vieille cour de pierre. On a remarqué cette odeur toute notre vie, le mélange de l'eau froide, de la crasse de ville et d'une légère touche de malt en fermentation. Ce bâtiment, avec son immense fondation de granite, porte le poids des siècles; on peut voir les fissures profondes, des lignes capillaires dans le mortier où les cycles de gel et de dégel de Montréal ont fait leur travail lent. Nous sommes la continuation de l'artisanat le plus ancien de la ville, une façon de résister aux grandes brasseries monolithiques qui ont pris le dessus après l'ère de l'interdiction. Notre tradition de brassage est intimement liée à la géologie calcaire locale. Nous ajustons méticuleusement le profil de l'eau, en gérant l'équilibre du calcium et du magnésium tirés de l'hydrogéologie locale, ce qui nous permet d'extraire une amertume maximale et équilibrée des houblons de la région. Ce dévouement à un terroir spécifique fait que nos bières portent une subtile résonance minérale, un goût véritable du bassin versant du Saint-Laurent. Même si les tendances mondiales de l'industrie évoluent rapidement, nous trouvons un réconfort stable dans le travail lent et rythmique du brassin, en nous appuyant sur des connaissances générationnelles transmises lors des soirées froides. Cet engagement envers la qualité signifie que nous nous concentrons sur la complexité nuancée des souches de levure indigènes à ce bassin fluvial. Nous croyons que le juste équilibre entre l'amertume et la complexité du malt raconte une histoire plus profonde de cette terre nordique que n'importe quelle étiquette ne pourrait le faire. Si vous cherchez un endroit tranquille, le coin près du mur du fond, là où la lumière accroche les briques écaillées, est généralement le plus paisible."
|
|
||||||
[2026-04-18 14:50:56.466] [info] 5. city="Brussels" country="Belgium" state="Brussels-Capital Region" iso3166_2=BE-BRU lat=50.8503 lon=4.3517
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_en="Le Moulin de Zythos"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_en="The way the humid Brussels air settles on the worn cobblestones, carrying the faint, metallic tang of rainwater and distant charcoal, is the first thing we notice every spring morning. Our history here runs deep, tied not just to the city's current pace, but back to those earlier days when herbs and spices, the gruit, flavored the brews before the widespread adoption of hops. We honor that ancient Belgian patience in our cellars. Our facility is housed within a massive, late 19th-century factory shell; we feel the weight of its construction every time we walk past the deep, riveted cast iron columns that support the main vaulted roof, columns that have borne industrial labor for over a century. Our focus is on radical yet historical profiles. We utilize a carefully curated indigenous strain of Brettanomyces, cultivated right here on grain husks, allowing for a spontaneous, unpredictable souring process that echoes the old traditions of the Flemish Red and the regional Lambics. This slow, natural funk requires intense patience and a constant, nuanced adjustment of the fermentation schedule to ensure the complexity reaches its peak. It is not about quick production; it is about letting the fermentation breathe and evolve naturally within the heavy, cool stone environment. Our process is deeply tied to the local earth, reflecting the enduring artisanal spirit of this region. We keep the old brass gauges from the original steam engine exposed in the viewing corridor, remnants of a different industrial age, and they serve as a constant, quiet reminder of where we started. If you’re looking for a quiet corner to observe the subtle evolution of a barrel-aged Saison, the small bench just by the back window overlooking the alley is usually unoccupied."
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_name_local="Le Moulin de Zythos"
|
|
||||||
[2026-04-18 14:50:56.466] [info] brewery_description_local="La façon dont l'air humide de Bruxelles se pose sur les pavés usés, portant cette saveur métallique légère de pluie et de charbon lointain, c'est la première chose que nous remarquons chaque matin de printemps. Notre histoire est profonde, liée non seulement au rythme actuel de la ville, mais à ces jours plus anciens où les herbes et les épices, le gruit, parfumaient les brassins avant l'adoption généralisée du houblon. Nous honorons cette ancienne patience belge dans nos caves. Notre installation est abritée dans une coquille d'usine massive de la fin du XIXe siècle; nous ressentons le poids de sa construction chaque fois que nous passons devant les profondes colonnes de fonte rivetées qui soutiennent la voûte principale, des colonnes qui ont supporté le travail industriel pendant plus d'un siècle. Notre objectif est d'obtenir des profils radicaux mais historiques. Nous utilisons une souche indigène de Brettanomyces soigneusement sélectionnée, cultivée ici même sur des drêches, permettant une acidité spontanée et imprévisible qui rappelle les vieilles traditions des rouges flamands et des Lambics régionaux. Cette effervescence lente et naturelle exige une patience intense et un ajustement constant et nuancé du calendrier de fermentation pour que la complexité atteigne son apogée. Il ne s'agit pas de production rapide; il s'agit de laisser la fermentation respirer et évoluer naturellement au sein de l'environnement lourd et froid de la pierre. Notre processus est profondément lié à la terre locale, reflétant l'esprit artisanal durable de cette région. Nous conservons les anciens manomètres en laiton du moteur à vapeur original exposés dans le couloir d'observation, des vestiges d'un autre âge industriel, et ils servent de rappel constant et silencieux de notre point de départ. Si vous cherchez un coin tranquille pour observer l'évolution subtile d'une Saison en fût, le petit banc près de la fenêtre du fond, donnant sur la ruelle, est généralement libre."
|
|
||||||
[2026-04-18 14:50:56.467] [info] Pipeline executed successfully
|
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
# Testing
|
# Testing
|
||||||
|
|
||||||
This document describes the testing strategy and how to run tests for The
|
This document describes the testing strategy and how to run tests for The Biergarten App.
|
||||||
Biergarten App.
|
|
||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
@@ -10,15 +9,13 @@ The project uses a multi-layered testing approach across backend and frontend:
|
|||||||
- **API.Specs** - BDD integration tests using Reqnroll (Gherkin)
|
- **API.Specs** - BDD integration tests using Reqnroll (Gherkin)
|
||||||
- **Infrastructure.Repository.Tests** - Unit tests for data access layer
|
- **Infrastructure.Repository.Tests** - Unit tests for data access layer
|
||||||
- **Service.Auth.Tests** - Unit tests for authentication business logic
|
- **Service.Auth.Tests** - Unit tests for authentication business logic
|
||||||
- **Storybook Vitest project** - Browser-based interaction tests for shared
|
- **Storybook Vitest project** - Browser-based interaction tests for shared website stories
|
||||||
website stories
|
- **Storybook Playwright suite** - Browser checks against Storybook-rendered components
|
||||||
- **Storybook Playwright suite** - Browser checks against Storybook-rendered
|
|
||||||
components
|
|
||||||
|
|
||||||
## Running Tests with Docker (Recommended)
|
## Running Tests with Docker (Recommended)
|
||||||
|
|
||||||
The easiest way to run all tests is using Docker Compose, which sets up an
|
The easiest way to run all tests is using Docker Compose, which sets up an isolated test
|
||||||
isolated test environment:
|
environment:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker compose -f docker-compose.test.yaml up --abort-on-container-exit
|
docker compose -f docker-compose.test.yaml up --abort-on-container-exit
|
||||||
@@ -101,8 +98,7 @@ npm run test:storybook
|
|||||||
|
|
||||||
**Purpose**:
|
**Purpose**:
|
||||||
|
|
||||||
- Verifies shared stories such as form fields, submit buttons, navbar states,
|
- Verifies shared stories such as form fields, submit buttons, navbar states, toasts, and the theme gallery
|
||||||
toasts, and the theme gallery
|
|
||||||
- Runs in browser mode via Vitest and Storybook integration
|
- Runs in browser mode via Vitest and Storybook integration
|
||||||
|
|
||||||
### Frontend Playwright Storybook Tests
|
### Frontend Playwright Storybook Tests
|
||||||
@@ -117,8 +113,7 @@ npm run test:storybook:playwright
|
|||||||
|
|
||||||
- Storybook dependencies installed
|
- Storybook dependencies installed
|
||||||
- Playwright browser dependencies installed
|
- Playwright browser dependencies installed
|
||||||
- The command will start or reuse the Storybook server defined in
|
- The command will start or reuse the Storybook server defined in `playwright.storybook.config.ts`
|
||||||
`playwright.storybook.config.ts`
|
|
||||||
|
|
||||||
## Test Coverage
|
## Test Coverage
|
||||||
|
|
||||||
@@ -283,8 +278,7 @@ Scenario: User login with valid credentials
|
|||||||
|
|
||||||
## Continuous Integration
|
## Continuous Integration
|
||||||
|
|
||||||
Tests run automatically in CI/CD pipelines using the test Docker Compose
|
Tests run automatically in CI/CD pipelines using the test Docker Compose configuration:
|
||||||
configuration:
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# CI/CD command
|
# CI/CD command
|
||||||
@@ -298,8 +292,7 @@ Exit codes:
|
|||||||
- `0` - All tests passed
|
- `0` - All tests passed
|
||||||
- Non-zero - Test failures occurred
|
- Non-zero - Test failures occurred
|
||||||
|
|
||||||
Frontend UI checks should also be included in CI for the active website
|
Frontend UI checks should also be included in CI for the active website workspace:
|
||||||
workspace:
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd src/Website
|
cd src/Website
|
||||||
@@ -2,14 +2,11 @@
|
|||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
The Core project implements comprehensive JWT token validation across three
|
The Core project implements comprehensive JWT token validation across three token types:
|
||||||
token types:
|
|
||||||
|
|
||||||
- **Access Tokens**: Short-lived (1 hour) tokens for API authentication
|
- **Access Tokens**: Short-lived (1 hour) tokens for API authentication
|
||||||
- **Refresh Tokens**: Long-lived (21 days) tokens for obtaining new access
|
- **Refresh Tokens**: Long-lived (21 days) tokens for obtaining new access tokens
|
||||||
tokens
|
- **Confirmation Tokens**: Short-lived (30 minutes) tokens for email confirmation
|
||||||
- **Confirmation Tokens**: Short-lived (30 minutes) tokens for email
|
|
||||||
confirmation
|
|
||||||
|
|
||||||
## Components
|
## Components
|
||||||
|
|
||||||
@@ -20,13 +17,10 @@ token types:
|
|||||||
Low-level JWT operations.
|
Low-level JWT operations.
|
||||||
|
|
||||||
**Methods:**
|
**Methods:**
|
||||||
|
|
||||||
- `GenerateJwt()` - Creates signed JWT tokens
|
- `GenerateJwt()` - Creates signed JWT tokens
|
||||||
- `ValidateJwtAsync()` - Validates token signature, expiration, and format
|
- `ValidateJwtAsync()` - Validates token signature, expiration, and format
|
||||||
|
|
||||||
**Implementation:**
|
**Implementation:** [JwtInfrastructure.cs](Infrastructure.Jwt/JwtInfrastructure.cs)
|
||||||
[JwtInfrastructure.cs](Infrastructure.Jwt/JwtInfrastructure.cs)
|
|
||||||
|
|
||||||
- Uses Microsoft.IdentityModel.JsonWebTokens.JsonWebTokenHandler
|
- Uses Microsoft.IdentityModel.JsonWebTokens.JsonWebTokenHandler
|
||||||
- Algorithm: HS256 (HMAC-SHA256)
|
- Algorithm: HS256 (HMAC-SHA256)
|
||||||
- Validates token lifetime, signature, and well-formedness
|
- Validates token lifetime, signature, and well-formedness
|
||||||
@@ -38,20 +32,16 @@ Low-level JWT operations.
|
|||||||
High-level token validation with context (token type, user extraction).
|
High-level token validation with context (token type, user extraction).
|
||||||
|
|
||||||
**Methods:**
|
**Methods:**
|
||||||
|
|
||||||
- `ValidateAccessTokenAsync(string token)` - Validates access tokens
|
- `ValidateAccessTokenAsync(string token)` - Validates access tokens
|
||||||
- `ValidateRefreshTokenAsync(string token)` - Validates refresh tokens
|
- `ValidateRefreshTokenAsync(string token)` - Validates refresh tokens
|
||||||
- `ValidateConfirmationTokenAsync(string token)` - Validates confirmation tokens
|
- `ValidateConfirmationTokenAsync(string token)` - Validates confirmation tokens
|
||||||
|
|
||||||
**Returns:** `ValidatedToken` record containing:
|
**Returns:** `ValidatedToken` record containing:
|
||||||
|
|
||||||
- `UserId` (Guid)
|
- `UserId` (Guid)
|
||||||
- `Username` (string)
|
- `Username` (string)
|
||||||
- `Principal` (ClaimsPrincipal) - Full JWT claims
|
- `Principal` (ClaimsPrincipal) - Full JWT claims
|
||||||
|
|
||||||
**Implementation:**
|
**Implementation:** [TokenValidationService.cs](Service.Auth/TokenValidationService.cs)
|
||||||
[TokenValidationService.cs](Service.Auth/TokenValidationService.cs)
|
|
||||||
|
|
||||||
- Reads token secrets from environment variables
|
- Reads token secrets from environment variables
|
||||||
- Extracts and validates claims (Sub, UniqueName)
|
- Extracts and validates claims (Sub, UniqueName)
|
||||||
- Throws `UnauthorizedException` on validation failure
|
- Throws `UnauthorizedException` on validation failure
|
||||||
@@ -61,18 +51,15 @@ High-level token validation with context (token type, user extraction).
|
|||||||
Token generation (existing service extended).
|
Token generation (existing service extended).
|
||||||
|
|
||||||
**Methods:**
|
**Methods:**
|
||||||
|
|
||||||
- `GenerateAccessToken(UserAccount)` - Creates 1-hour access token
|
- `GenerateAccessToken(UserAccount)` - Creates 1-hour access token
|
||||||
- `GenerateRefreshToken(UserAccount)` - Creates 21-day refresh token
|
- `GenerateRefreshToken(UserAccount)` - Creates 21-day refresh token
|
||||||
- `GenerateConfirmationToken(UserAccount)` - Creates 30-minute confirmation
|
- `GenerateConfirmationToken(UserAccount)` - Creates 30-minute confirmation token
|
||||||
token
|
|
||||||
|
|
||||||
### Integration Points
|
### Integration Points
|
||||||
|
|
||||||
#### [ConfirmationService](Service.Auth/IConfirmationService.cs)
|
#### [ConfirmationService](Service.Auth/IConfirmationService.cs)
|
||||||
|
|
||||||
**Flow:**
|
**Flow:**
|
||||||
|
|
||||||
1. Receives confirmation token from user
|
1. Receives confirmation token from user
|
||||||
2. Calls `TokenValidationService.ValidateConfirmationTokenAsync()`
|
2. Calls `TokenValidationService.ValidateConfirmationTokenAsync()`
|
||||||
3. Extracts user ID from validated token
|
3. Extracts user ID from validated token
|
||||||
@@ -82,7 +69,6 @@ Token generation (existing service extended).
|
|||||||
#### [RefreshTokenService](Service.Auth/RefreshTokenService.cs)
|
#### [RefreshTokenService](Service.Auth/RefreshTokenService.cs)
|
||||||
|
|
||||||
**Flow:**
|
**Flow:**
|
||||||
|
|
||||||
1. Receives refresh token from user
|
1. Receives refresh token from user
|
||||||
2. Calls `TokenValidationService.ValidateRefreshTokenAsync()`
|
2. Calls `TokenValidationService.ValidateRefreshTokenAsync()`
|
||||||
3. Retrieves user account via `AuthRepository.GetUserByIdAsync()`
|
3. Retrieves user account via `AuthRepository.GetUserByIdAsync()`
|
||||||
@@ -92,7 +78,6 @@ Token generation (existing service extended).
|
|||||||
#### [AuthController](API.Core/Controllers/AuthController.cs)
|
#### [AuthController](API.Core/Controllers/AuthController.cs)
|
||||||
|
|
||||||
**Endpoints:**
|
**Endpoints:**
|
||||||
|
|
||||||
- `POST /api/auth/register` - Register new user
|
- `POST /api/auth/register` - Register new user
|
||||||
- `POST /api/auth/login` - Authenticate user
|
- `POST /api/auth/login` - Authenticate user
|
||||||
- `POST /api/auth/confirm?token=...` - Confirm email
|
- `POST /api/auth/confirm?token=...` - Confirm email
|
||||||
@@ -103,13 +88,11 @@ Token generation (existing service extended).
|
|||||||
### Token Secrets
|
### Token Secrets
|
||||||
|
|
||||||
Three independent secrets enable:
|
Three independent secrets enable:
|
||||||
|
|
||||||
- **Key rotation** - Rotate each secret type independently
|
- **Key rotation** - Rotate each secret type independently
|
||||||
- **Isolation** - Compromise of one secret doesn't affect others
|
- **Isolation** - Compromise of one secret doesn't affect others
|
||||||
- **Different expiration** - Different token types can expire at different rates
|
- **Different expiration** - Different token types can expire at different rates
|
||||||
|
|
||||||
**Environment Variables:**
|
**Environment Variables:**
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
ACCESS_TOKEN_SECRET=... # Signs 1-hour access tokens
|
ACCESS_TOKEN_SECRET=... # Signs 1-hour access tokens
|
||||||
REFRESH_TOKEN_SECRET=... # Signs 21-day refresh tokens
|
REFRESH_TOKEN_SECRET=... # Signs 21-day refresh tokens
|
||||||
@@ -128,7 +111,6 @@ Each token is validated for:
|
|||||||
### Error Handling
|
### Error Handling
|
||||||
|
|
||||||
Validation failures return HTTP 401 Unauthorized:
|
Validation failures return HTTP 401 Unauthorized:
|
||||||
|
|
||||||
- Invalid signature → "Invalid token"
|
- Invalid signature → "Invalid token"
|
||||||
- Expired token → "Invalid token" (message doesn't reveal reason for security)
|
- Expired token → "Invalid token" (message doesn't reveal reason for security)
|
||||||
- Missing claims → "Invalid token"
|
- Missing claims → "Invalid token"
|
||||||
@@ -167,19 +149,16 @@ Validation failures return HTTP 401 Unauthorized:
|
|||||||
### Unit Tests
|
### Unit Tests
|
||||||
|
|
||||||
**TokenValidationService.test.cs**
|
**TokenValidationService.test.cs**
|
||||||
|
|
||||||
- Happy path: Valid token extraction
|
- Happy path: Valid token extraction
|
||||||
- Error cases: Invalid, expired, malformed tokens
|
- Error cases: Invalid, expired, malformed tokens
|
||||||
- Missing/invalid claims scenarios
|
- Missing/invalid claims scenarios
|
||||||
|
|
||||||
**RefreshTokenService.test.cs**
|
**RefreshTokenService.test.cs**
|
||||||
|
|
||||||
- Successful refresh with valid token
|
- Successful refresh with valid token
|
||||||
- Invalid/expired refresh token rejection
|
- Invalid/expired refresh token rejection
|
||||||
- Non-existent user handling
|
- Non-existent user handling
|
||||||
|
|
||||||
**ConfirmationService.test.cs**
|
**ConfirmationService.test.cs**
|
||||||
|
|
||||||
- Successful confirmation with valid token
|
- Successful confirmation with valid token
|
||||||
- Token validation failures
|
- Token validation failures
|
||||||
- User not found scenarios
|
- User not found scenarios
|
||||||
@@ -187,19 +166,16 @@ Validation failures return HTTP 401 Unauthorized:
|
|||||||
### BDD Tests (Reqnroll)
|
### BDD Tests (Reqnroll)
|
||||||
|
|
||||||
**TokenRefresh.feature**
|
**TokenRefresh.feature**
|
||||||
|
|
||||||
- Successful token refresh
|
- Successful token refresh
|
||||||
- Invalid/expired token rejection
|
- Invalid/expired token rejection
|
||||||
- Missing token validation
|
- Missing token validation
|
||||||
|
|
||||||
**Confirmation.feature**
|
**Confirmation.feature**
|
||||||
|
|
||||||
- Successful email confirmation
|
- Successful email confirmation
|
||||||
- Expired/tampered token rejection
|
- Expired/tampered token rejection
|
||||||
- Missing token validation
|
- Missing token validation
|
||||||
|
|
||||||
**AccessTokenValidation.feature**
|
**AccessTokenValidation.feature**
|
||||||
|
|
||||||
- Protected endpoint access token validation
|
- Protected endpoint access token validation
|
||||||
- Invalid/expired access token rejection
|
- Invalid/expired access token rejection
|
||||||
- Token type mismatch (refresh used as access token)
|
- Token type mismatch (refresh used as access token)
|
||||||
2411
misc/raw-data/beers.csv
Normal file
2411
misc/raw-data/beers.csv
Normal file
File diff suppressed because it is too large
Load Diff
559
misc/raw-data/breweries.csv
Normal file
559
misc/raw-data/breweries.csv
Normal file
@@ -0,0 +1,559 @@
|
|||||||
|
,name,city,state
|
||||||
|
0,NorthGate Brewing ,Minneapolis, MN
|
||||||
|
1,Against the Grain Brewery,Louisville, KY
|
||||||
|
2,Jack's Abby Craft Lagers,Framingham, MA
|
||||||
|
3,Mike Hess Brewing Company,San Diego, CA
|
||||||
|
4,Fort Point Beer Company,San Francisco, CA
|
||||||
|
5,COAST Brewing Company,Charleston, SC
|
||||||
|
6,Great Divide Brewing Company,Denver, CO
|
||||||
|
7,Tapistry Brewing,Bridgman, MI
|
||||||
|
8,Big Lake Brewing,Holland, MI
|
||||||
|
9,The Mitten Brewing Company,Grand Rapids, MI
|
||||||
|
10,Brewery Vivant,Grand Rapids, MI
|
||||||
|
11,Petoskey Brewing,Petoskey, MI
|
||||||
|
12,Blackrocks Brewery,Marquette, MI
|
||||||
|
13,Perrin Brewing Company,Comstock Park, MI
|
||||||
|
14,Witch's Hat Brewing Company,South Lyon, MI
|
||||||
|
15,Founders Brewing Company,Grand Rapids, MI
|
||||||
|
16,Flat 12 Bierwerks,Indianapolis, IN
|
||||||
|
17,Tin Man Brewing Company,Evansville, IN
|
||||||
|
18,Black Acre Brewing Co.,Indianapolis, IN
|
||||||
|
19,Brew Link Brewing,Plainfield, IN
|
||||||
|
20,Bare Hands Brewery,Granger, IN
|
||||||
|
21,Three Pints Brewing,Martinsville, IN
|
||||||
|
22,Four Fathers Brewing ,Valparaiso, IN
|
||||||
|
23,Indiana City Brewing,Indianapolis, IN
|
||||||
|
24,Burn 'Em Brewing,Michigan City, IN
|
||||||
|
25,Sun King Brewing Company,Indianapolis, IN
|
||||||
|
26,Evil Czech Brewery,Mishawaka, IN
|
||||||
|
27,450 North Brewing Company,Columbus, IN
|
||||||
|
28,Taxman Brewing Company,Bargersville, IN
|
||||||
|
29,Cedar Creek Brewery,Seven Points, TX
|
||||||
|
30,SanTan Brewing Company,Chandler, AZ
|
||||||
|
31,Boulevard Brewing Company,Kansas City, MO
|
||||||
|
32,James Page Brewing Company,Stevens Point, WI
|
||||||
|
33,The Dudes' Brewing Company,Torrance, CA
|
||||||
|
34,Ballast Point Brewing Company,San Diego, CA
|
||||||
|
35,Anchor Brewing Company,San Francisco, CA
|
||||||
|
36,Figueroa Mountain Brewing Company,Buellton, CA
|
||||||
|
37,Avery Brewing Company,Boulder, CO
|
||||||
|
38,Twisted X Brewing Company,Dripping Springs, TX
|
||||||
|
39,Gonzo's BiggDogg Brewing,Kalamazoo, MI
|
||||||
|
40,Big Muddy Brewing,Murphysboro, IL
|
||||||
|
41,Lost Nation Brewing,East Fairfield, VT
|
||||||
|
42,Rising Tide Brewing Company,Portland, ME
|
||||||
|
43,Rivertowne Brewing Company,Export, PA
|
||||||
|
44,Revolution Brewing Company,Chicago, IL
|
||||||
|
45,Tallgrass Brewing Company,Manhattan, KS
|
||||||
|
46,Sixpoint Craft Ales,Brooklyn, NY
|
||||||
|
47,White Birch Brewing,Hooksett, NH
|
||||||
|
48,Firestone Walker Brewing Company,Paso Robles, CA
|
||||||
|
49,SweetWater Brewing Company,Atlanta, GA
|
||||||
|
50,Flying Mouse Brewery,Troutville, VA
|
||||||
|
51,Upslope Brewing Company,Boulder, CO
|
||||||
|
52,Pipeworks Brewing Company,Chicago, IL
|
||||||
|
53,Bent Brewstillery,Roseville, MN
|
||||||
|
54,Flesk Brewing Company,Lombard, IL
|
||||||
|
55,Pollyanna Brewing Company,Lemont, IL
|
||||||
|
56,BuckleDown Brewing,Lyons, IL
|
||||||
|
57,Destihl Brewery,Bloomington, IL
|
||||||
|
58,Summit Brewing Company,St. Paul, MN
|
||||||
|
59,Latitude 42 Brewing Company,Portage, MI
|
||||||
|
60,4 Hands Brewing Company,Saint Louis, MO
|
||||||
|
61,Surly Brewing Company,Brooklyn Center, MN
|
||||||
|
62,Against The Grain Brewery,Louisville, KY
|
||||||
|
63,Crazy Mountain Brewing Company,Edwards, CO
|
||||||
|
64,SlapShot Brewing Company,Chicago, IL
|
||||||
|
65,Mikerphone Brewing,Chicago, IL
|
||||||
|
66,Freetail Brewing Company,San Antonio, TX
|
||||||
|
67,3 Daughters Brewing,St Petersburg, FL
|
||||||
|
68,Red Shedman Farm Brewery and Hop...,Mt. Airy, MD
|
||||||
|
69,Appalachian Mountain Brewery,Boone, NC
|
||||||
|
70,Birdsong Brewing Company,Charlotte, NC
|
||||||
|
71,Union Craft Brewing,Baltimore, MD
|
||||||
|
72,Atwater Brewery,Detroit, MI
|
||||||
|
73,Ale Asylum,Madison, WI
|
||||||
|
74,Two Brothers Brewing Company,Warrenville, IL
|
||||||
|
75,Bent Paddle Brewing Company,Duluth, MN
|
||||||
|
76,Bell's Brewery,Kalamazoo, MI
|
||||||
|
77,Blue Owl Brewing,Austin, TX
|
||||||
|
78,Speakasy Ales & Lagers,San Francisco, CA
|
||||||
|
79,Black Tooth Brewing Company,Sheridan, WY
|
||||||
|
80,Hopworks Urban Brewery,Portland, OR
|
||||||
|
81,Epic Brewing,Denver, CO
|
||||||
|
82,New Belgium Brewing Company,Fort Collins, CO
|
||||||
|
83,Sierra Nevada Brewing Company,Chico, CA
|
||||||
|
84,Keweenaw Brewing Company,Houghton, MI
|
||||||
|
85,Brewery Terra Firma,Traverse City, MI
|
||||||
|
86,Grey Sail Brewing Company,Westerly, RI
|
||||||
|
87,Kirkwood Station Brewing Company,Kirkwood, MO
|
||||||
|
88,Goose Island Brewing Company,Chicago, IL
|
||||||
|
89,Broad Brook Brewing LLC,East Windsor, CT
|
||||||
|
90,The Lion Brewery,Wilkes-Barre, PA
|
||||||
|
91,Madtree Brewing Company,Cincinnati, OH
|
||||||
|
92,Jackie O's Pub & Brewery,Athens, OH
|
||||||
|
93,Rhinegeist Brewery,Cincinnati, OH
|
||||||
|
94,Warped Wing Brewing Company,Dayton, OH
|
||||||
|
95,Blackrocks Brewery,Marquette, MA
|
||||||
|
96,Catawba Valley Brewing Company,Morganton, NC
|
||||||
|
97,Tröegs Brewing Company,Hershey, PA
|
||||||
|
98,Mission Brewery,San Diego, CA
|
||||||
|
99,Christian Moerlein Brewing Company,Cincinnati, OH
|
||||||
|
100,West Sixth Brewing,Lexington, KY
|
||||||
|
101,Coastal Extreme Brewing Company,Newport, RI
|
||||||
|
102,King Street Brewing Company,Anchorage, AK
|
||||||
|
103,Beer Works Brewery,Lowell, MA
|
||||||
|
104,Lone Tree Brewing Company,Lone Tree, CO
|
||||||
|
105,Four String Brewing Company,Columbus, OH
|
||||||
|
106,Glabrous Brewing Company,Pineland, ME
|
||||||
|
107,Bonfire Brewing Company,Eagle, CO
|
||||||
|
108,Thomas Hooker Brewing Company,Bloomfield, CT
|
||||||
|
109,"Woodstock Inn, Station & Brewery",North Woodstock, NH
|
||||||
|
110,Renegade Brewing Company,Denver, CO
|
||||||
|
111,Mother Earth Brew Company,Vista, CA
|
||||||
|
112,Black Market Brewing Company,Temecula, CA
|
||||||
|
113,Vault Brewing Company,Yardley, PA
|
||||||
|
114,Jailbreak Brewing Company,Laurel, MD
|
||||||
|
115,Smartmouth Brewing Company,Norfolk, VA
|
||||||
|
116,Base Camp Brewing Co.,Portland, OR
|
||||||
|
117,Alameda Brewing,Portland, OR
|
||||||
|
118,Southern Star Brewing Company,Conroe, TX
|
||||||
|
119,Steamworks Brewing Company,Durango, CO
|
||||||
|
120,Horny Goat Brew Pub,Milwaukee, WI
|
||||||
|
121,Cheboygan Brewing Company,Cheboygan, MI
|
||||||
|
122,Center of the Universe Brewing C...,Ashland, VA
|
||||||
|
123,Ipswich Ale Brewery,Ipswich, MA
|
||||||
|
124,Griffin Claw Brewing Company,Birmingham, MI
|
||||||
|
125,Karbach Brewing Company,Houston, TX
|
||||||
|
126,Uncle Billy's Brewery and Smokeh...,Austin, TX
|
||||||
|
127,Deep Ellum Brewing Company,Dallas, TX
|
||||||
|
128,Real Ale Brewing Company,Blanco, TX
|
||||||
|
129,Straub Brewery,St Mary's, PA
|
||||||
|
130,Shebeen Brewing Company,Wolcott, CT
|
||||||
|
131,Stevens Point Brewery,Stevens Point, WI
|
||||||
|
132,Weston Brewing Company,Weston, MO
|
||||||
|
133,Southern Prohibition Brewing Com...,Hattiesburg, MS
|
||||||
|
134,Minhas Craft Brewery,Monroe, WI
|
||||||
|
135,Pug Ryan's Brewery,Dillon, CO
|
||||||
|
136,Hops & Grains Brewing Company,Austin, TX
|
||||||
|
137,Sietsema Orchards and Cider Mill,Ada, MI
|
||||||
|
138,Summit Brewing Company,St Paul, MN
|
||||||
|
139,Core Brewing & Distilling Company,Springdale, AR
|
||||||
|
140,Independence Brewing Company,Austin, TX
|
||||||
|
141,Cigar City Brewing Company,Tampa, FL
|
||||||
|
142,Third Street Brewhouse,Cold Spring, MN
|
||||||
|
143,Narragansett Brewing Company,Providence, RI
|
||||||
|
144,Grimm Brothers Brewhouse,Loveland, CO
|
||||||
|
145,Cisco Brewers,Nantucket, MA
|
||||||
|
146,Angry Minnow,Hayward, WI
|
||||||
|
147,Platform Beer Company,Cleveland, OH
|
||||||
|
148,Odyssey Beerwerks,Arvada, CO
|
||||||
|
149,Lonerider Brewing Company,Raleigh, NC
|
||||||
|
150,Oakshire Brewing,Eugene, OR
|
||||||
|
151,Fort Pitt Brewing Company,Latrobe, PA
|
||||||
|
152,Tin Roof Brewing Company,Baton Rouge, LA
|
||||||
|
153,Three Creeks Brewing,Sisters, OR
|
||||||
|
154,2 Towns Ciderhouse,Corvallis, OR
|
||||||
|
155,Caldera Brewing Company,Ashland, OR
|
||||||
|
156,Greenbrier Valley Brewing Company,Lewisburg, WV
|
||||||
|
157,Phoenix Ale Brewery,Phoenix, AZ
|
||||||
|
158,Lumberyard Brewing Company,Flagstaff, AZ
|
||||||
|
159,Uinta Brewing Company,Salt Lake City, UT
|
||||||
|
160,Four Peaks Brewing Company,Tempe, AZ
|
||||||
|
161,Martin House Brewing Company,Fort Worth, TX
|
||||||
|
162,Right Brain Brewery,Traverse City, MI
|
||||||
|
163,Sly Fox Brewing Company,Phoenixville, PA
|
||||||
|
164,Round Guys Brewing,Lansdale, PA
|
||||||
|
165,Great Crescent Brewery,Aurora, IN
|
||||||
|
166,Oskar Blues Brewery,Longmont, CO
|
||||||
|
167,Boxcar Brewing Company,West Chester, PA
|
||||||
|
168,High Hops Brewery,Windsor, CO
|
||||||
|
169,Crooked Fence Brewing Company,Garden City, ID
|
||||||
|
170,Everybody's Brewing,White Salmon, WA
|
||||||
|
171,Anderson Valley Brewing Company,Boonville, CA
|
||||||
|
172,Fiddlehead Brewing Company,Shelburne, VT
|
||||||
|
173,Evil Twin Brewing,Brooklyn, NY
|
||||||
|
174,New Orleans Lager & Ale Brewing ...,New Orleans, LA
|
||||||
|
175,Spiteful Brewing Company,Chicago, IL
|
||||||
|
176,Rahr & Sons Brewing Company,Fort Worth, TX
|
||||||
|
177,18th Street Brewery,Gary, IN
|
||||||
|
178,Cambridge Brewing Company,Cambridge, MA
|
||||||
|
179,Carolina Brewery,Pittsboro, NC
|
||||||
|
180,Frog Level Brewing Company,Waynesville, NC
|
||||||
|
181,Wild Wolf Brewing Company,Nellysford, VA
|
||||||
|
182,COOP Ale Works,Oklahoma City, OK
|
||||||
|
183,Seventh Son Brewing Company,Columbus, OH
|
||||||
|
184,Oasis Texas Brewing Company,Austin, TX
|
||||||
|
185,Vander Mill Ciders,Spring Lake, MI
|
||||||
|
186,St. Julian Winery,Paw Paw, MI
|
||||||
|
187,Pedernales Brewing Company,Fredericksburg, TX
|
||||||
|
188,Mother's Brewing,Springfield, MO
|
||||||
|
189,Modern Monks Brewery,Lincoln, NE
|
||||||
|
190,Two Beers Brewing Company,Seattle, WA
|
||||||
|
191,Snake River Brewing Company,Jackson, WY
|
||||||
|
192,Capital Brewery,Middleton, WI
|
||||||
|
193,Anthem Brewing Company,Oklahoma City, OK
|
||||||
|
194,Goodlife Brewing Co.,Bend, OR
|
||||||
|
195,Breakside Brewery,Portland, OR
|
||||||
|
196,Goose Island Brewery Company,Chicago, IL
|
||||||
|
197,Burnside Brewing Co.,Portland, OR
|
||||||
|
198,Hop Valley Brewing Company,Springfield, OR
|
||||||
|
199,Worthy Brewing Company,Bend, OR
|
||||||
|
200,Occidental Brewing Company,Portland, OR
|
||||||
|
201,Fearless Brewing Company,Estacada, OR
|
||||||
|
202,Upland Brewing Company,Bloomington, IN
|
||||||
|
203,Mehana Brewing Co.,Hilo, HI
|
||||||
|
204,Hawai'i Nui Brewing Co.,Hilo, HI
|
||||||
|
205,People's Brewing Company,Lafayette, IN
|
||||||
|
206,Fort George Brewery,Astoria, OR
|
||||||
|
207,Branchline Brewing Company,San Antonio, TX
|
||||||
|
208,Kalona Brewing Company,Kalona, IA
|
||||||
|
209,Modern Times Beer,San Diego, CA
|
||||||
|
210,Temperance Beer Company,Evanston, IL
|
||||||
|
211,Wisconsin Brewing Company,Verona, WI
|
||||||
|
212,Crow Peak Brewing Company,Spearfish, SD
|
||||||
|
213,Grapevine Craft Brewery,Farmers Branch, TX
|
||||||
|
214,Buffalo Bayou Brewing Company,Houston, TX
|
||||||
|
215,Texian Brewing Co.,Richmond, TX
|
||||||
|
216,Orpheus Brewing,Atlanta, GA
|
||||||
|
217,Forgotten Boardwalk,Cherry Hill, NJ
|
||||||
|
218,Laughing Dog Brewing Company,Ponderay, ID
|
||||||
|
219,Bozeman Brewing Company,Bozeman, MT
|
||||||
|
220,Big Choice Brewing,Broomfield, CO
|
||||||
|
221,Big Storm Brewing Company,Odessa, FL
|
||||||
|
222,Carton Brewing Company,Atlantic Highlands, NJ
|
||||||
|
223,Midnight Sun Brewing Company,Anchorage, AK
|
||||||
|
224,Fat Head's Brewery,Middleburg Heights, OH
|
||||||
|
225,Refuge Brewery,Temecula, CA
|
||||||
|
226,Chatham Brewing,Chatham, NY
|
||||||
|
227,DC Brau Brewing Company,Washington, DC
|
||||||
|
228,Geneva Lake Brewing Company,Lake Geneva, WI
|
||||||
|
229,Rochester Mills Brewing Company,Rochester, MI
|
||||||
|
230,Cape Ann Brewing Company,Gloucester, MA
|
||||||
|
231,Borderlands Brewing Company,Tucson, AZ
|
||||||
|
232,College Street Brewhouse and Pub,Lake Havasu City, AZ
|
||||||
|
233,Joseph James Brewing Company,Henderson, NV
|
||||||
|
234,Harpoon Brewery,Boston, MA
|
||||||
|
235,Back East Brewing Company,Bloomfield, CT
|
||||||
|
236,Champion Brewing Company,Charlottesville, VA
|
||||||
|
237,Devil's Backbone Brewing Company,Lexington, VA
|
||||||
|
238,Newburgh Brewing Company,Newburgh, NY
|
||||||
|
239,Wiseacre Brewing Company,Memphis, TN
|
||||||
|
240,Golden Road Brewing,Los Angeles, CA
|
||||||
|
241,New Republic Brewing Company,College Station, TX
|
||||||
|
242,Infamous Brewing Company,Austin, TX
|
||||||
|
243,Two Henrys Brewing Company,Plant City, FL
|
||||||
|
244,Lift Bridge Brewing Company,Stillwater, MN
|
||||||
|
245,Lucky Town Brewing Company,Jackson, MS
|
||||||
|
246,Quest Brewing Company,Greenville, SC
|
||||||
|
247,Creature Comforts,Athens, GA
|
||||||
|
248,Half Full Brewery,Stamford, CT
|
||||||
|
249,Southampton Publick House,Southampton, NY
|
||||||
|
250,Chapman's Brewing,Angola, IN
|
||||||
|
251,Barrio Brewing Company,Tucson, AZ
|
||||||
|
252,Santa Cruz Mountain Brewing,Santa Cruz, CA
|
||||||
|
253,Frankenmuth Brewery,Frankenmuth, MI
|
||||||
|
254,Meckley's Cidery,Somerset Center, MI
|
||||||
|
255,Stillwater Artisanal Ales,Baltimore, MD
|
||||||
|
256,Finch's Beer Company,Chicago, IL
|
||||||
|
257,South Austin Brewery,South Austin, TX
|
||||||
|
258,Bauhaus Brew Labs,Minneapolis, MN
|
||||||
|
259,Ozark Beer Company,Rogers, AR
|
||||||
|
260,Mountain Town Brewing Company ,Mount Pleasant, MI
|
||||||
|
261,Otter Creek Brewing,Waterbury, VT
|
||||||
|
262,The Brewer's Art,Baltimore, MD
|
||||||
|
263,Denver Beer Company,Denver, CO
|
||||||
|
264,Ska Brewing Company,Durango, CO
|
||||||
|
265,Tractor Brewing Company,Albuquerque, NM
|
||||||
|
266,Peak Organic Brewing Company,Portland, ME
|
||||||
|
267,Cape Cod Beer,Hyannis, MA
|
||||||
|
268,Long Trail Brewing Company,Bridgewater Corners, VT
|
||||||
|
269,Great Raft Brewing Company,Shreveport, LA
|
||||||
|
270,Alaskan Brewing Company,Juneau, AK
|
||||||
|
271,Notch Brewing Company,Ipswich, MA
|
||||||
|
272,The Alchemist,Waterbury, VT
|
||||||
|
273,Three Notch'd Brewing Company,Charlottesville, VA
|
||||||
|
274,Portside Brewery,Cleveland, OH
|
||||||
|
275,Otter Creek Brewing,Middlebury, VT
|
||||||
|
276,Montauk Brewing Company,Montauk, NY
|
||||||
|
277,Indeed Brewing Company,Minneapolis, MN
|
||||||
|
278,Berkshire Brewing Company,South Deerfield, MA
|
||||||
|
279,Foolproof Brewing Company,Pawtucket, RI
|
||||||
|
280,Headlands Brewing Company,Mill Valley, CA
|
||||||
|
281,Bolero Snort Brewery,Ridgefield Park, NJ
|
||||||
|
282,Thunderhead Brewing Company,Kearney, NE
|
||||||
|
283,Defiance Brewing Company,Hays, KS
|
||||||
|
284,Milwaukee Brewing Company,Milwaukee, WI
|
||||||
|
285,Catawba Island Brewing,Port Clinton, OH
|
||||||
|
286,Back Forty Beer Company,Gadsden, AL
|
||||||
|
287,Four Corners Brewing Company,Dallas, TX
|
||||||
|
288,Saint Archer Brewery,San Diego, CA
|
||||||
|
289,Rogue Ales,Newport, OR
|
||||||
|
290,Hale's Ales,Seattle, WA
|
||||||
|
291,Tommyknocker Brewery,Idaho Springs, CO
|
||||||
|
292,Baxter Brewing Company,Lewiston, ME
|
||||||
|
293,Northampton Brewery,Northamtpon, MA
|
||||||
|
294,Black Shirt Brewing Company,Denver, CO
|
||||||
|
295,Wachusett Brewing Company,Westminster, MA
|
||||||
|
296,Widmer Brothers Brewing Company,Portland, OR
|
||||||
|
297,Hop Farm Brewing Company,Pittsburgh, PA
|
||||||
|
298,Liquid Hero Brewery,York, PA
|
||||||
|
299,Matt Brewing Company,Utica, NY
|
||||||
|
300,Boston Beer Company,Boston, MA
|
||||||
|
301,Old Forge Brewing Company,Danville, PA
|
||||||
|
302,Utah Brewers Cooperative,Salt Lake City, UT
|
||||||
|
303,Magic Hat Brewing Company,South Burlington, VT
|
||||||
|
304,Blue Hills Brewery,Canton, MA
|
||||||
|
305,Night Shift Brewing,Everett, MA
|
||||||
|
306,Beach Brewing Company,Virginia Beach, VA
|
||||||
|
307,Payette Brewing Company,Garden City, ID
|
||||||
|
308,Brew Bus Brewing,Tampa, FL
|
||||||
|
309,Sockeye Brewing Company,Boise, ID
|
||||||
|
310,Pine Street Brewery,San Francisco, CA
|
||||||
|
311,Dirty Bucket Brewing Company,Woodinville, WA
|
||||||
|
312,Jackalope Brewing Company,Nashville, TN
|
||||||
|
313,Slanted Rock Brewing Company,Meridian, ID
|
||||||
|
314,Piney River Brewing Company,Bucryus, MO
|
||||||
|
315,Cutters Brewing Company,Avon, IN
|
||||||
|
316,Iron Hill Brewery & Restaurant,Wilmington, DE
|
||||||
|
317,Marshall Wharf Brewing Company,Belfast, ME
|
||||||
|
318,Banner Beer Company,Williamsburg, MA
|
||||||
|
319,Dick's Brewing Company,Centralia, WA
|
||||||
|
320,Claremont Craft Ales,Claremont, CA
|
||||||
|
321,Rivertown Brewing Company,Lockland, OH
|
||||||
|
322,Voodoo Brewery,Meadville, PA
|
||||||
|
323,D.L. Geary Brewing Company,Portland, ME
|
||||||
|
324,Pisgah Brewing Company,Black Mountain, NC
|
||||||
|
325,Neshaminy Creek Brewing Company,Croydon, PA
|
||||||
|
326,Morgan Street Brewery,Saint Louis, MO
|
||||||
|
327,Half Acre Beer Company,Chicago, IL
|
||||||
|
328,The Just Beer Project,Burlington, VT
|
||||||
|
329,The Bronx Brewery,Bronx, NY
|
||||||
|
330,Dead Armadillo Craft Brewing,Tulsa, OK
|
||||||
|
331,Catawba Brewing Company,Morganton, NC
|
||||||
|
332,La Cumbre Brewing Company,Albuquerque, NM
|
||||||
|
333,David's Ale Works,Diamond Springs, CA
|
||||||
|
334,The Traveler Beer Company,Burlington, VT
|
||||||
|
335,Fargo Brewing Company,Fargo, ND
|
||||||
|
336,Big Sky Brewing Company,Missoula, MT
|
||||||
|
337,Nebraska Brewing Company,Papillion, NE
|
||||||
|
338,Uncle John's Fruit House Winery,St. John's, MI
|
||||||
|
339,Wormtown Brewery,Worcester, MA
|
||||||
|
340,Due South Brewing Company,Boynton Beach, FL
|
||||||
|
341,Palisade Brewing Company,Palisade, CO
|
||||||
|
342,KelSo Beer Company,Brooklyn, NY
|
||||||
|
343,Hardywood Park Craft Brewery,Richmond, VA
|
||||||
|
344,Wolf Hills Brewing Company,Abingdon, VA
|
||||||
|
345,Lavery Brewing Company,Erie, PA
|
||||||
|
346,Manzanita Brewing Company,Santee, CA
|
||||||
|
347,Fullsteam Brewery,Durham, NC
|
||||||
|
348,Four Horsemen Brewing Company,South Bend, IN
|
||||||
|
349,Hinterland Brewery,Green Bay, WI
|
||||||
|
350,Central Coast Brewing Company,San Luis Obispo, CA
|
||||||
|
351,Westfield River Brewing Company,Westfield, MA
|
||||||
|
352,Elevator Brewing Company,Columbus, OH
|
||||||
|
353,Aslan Brewing Company,Bellingham, WA
|
||||||
|
354,Kulshan Brewery,Bellingham, WA
|
||||||
|
355,Pikes Peak Brewing Company,Monument, CO
|
||||||
|
356,Manayunk Brewing Company,Philadelphia, PA
|
||||||
|
357,Buckeye Brewing,Cleveland, OH
|
||||||
|
358,Daredevil Brewing Company,Shelbyville, IN
|
||||||
|
359,NoDa Brewing Company,Charlotte, NC
|
||||||
|
360,Aviator Brewing Company,Fuquay-Varina, NC
|
||||||
|
361,Wild Onion Brewing Company,Lake Barrington, IL
|
||||||
|
362,Hilliard's Beer,Seattle, WA
|
||||||
|
363,Mikkeller,Pottstown, PA
|
||||||
|
364,Bohemian Brewery,Midvale, UT
|
||||||
|
365,Great River Brewery,Davenport, IA
|
||||||
|
366,Mustang Brewing Company,Mustang, OK
|
||||||
|
367,Airways Brewing Company,Kent, WA
|
||||||
|
368,21st Amendment Brewery,San Francisco, CA
|
||||||
|
369,Eddyline Brewery & Restaurant,Buena Vista, CO
|
||||||
|
370,Pizza Port Brewing Company,Carlsbad, CA
|
||||||
|
371,Sly Fox Brewing Company,Pottstown, PA
|
||||||
|
372,Spring House Brewing Company,Conestoga, PA
|
||||||
|
373,7venth Sun,Dunedin, FL
|
||||||
|
374,Astoria Brewing Company,Astoria, OR
|
||||||
|
375,Maui Brewing Company,Lahaina, HI
|
||||||
|
376,RoughTail Brewing Company,Midwest City, OK
|
||||||
|
377,Lucette Brewing Company,Menominee, WI
|
||||||
|
378,Bold City Brewery,Jacksonville, FL
|
||||||
|
379,Grey Sail Brewing of Rhode Island,Westerly, RI
|
||||||
|
380,Blue Blood Brewing Company,Lincoln, NE
|
||||||
|
381,Swashbuckler Brewing Company,Manheim, PA
|
||||||
|
382,Blue Mountain Brewery,Afton, VA
|
||||||
|
383,Starr Hill Brewery,Crozet, VA
|
||||||
|
384,Westbrook Brewing Company,Mt. Pleasant, SC
|
||||||
|
385,Shipyard Brewing Company,Portland, ME
|
||||||
|
386,Revolution Brewing,Paonia, CO
|
||||||
|
387,Natian Brewery,Portland, OR
|
||||||
|
388,Alltech's Lexington Brewing Company,Lexington, KY
|
||||||
|
389,Oskar Blues Brewery (North Carol...,Brevard, NC
|
||||||
|
390,Orlison Brewing Company,Airway Heights, WA
|
||||||
|
391,Breckenridge Brewery,Denver, CO
|
||||||
|
392,Santa Fe Brewing Company,Santa Fe, NM
|
||||||
|
393,Miami Brewing Company,Miami, FL
|
||||||
|
394,Schilling & Company,Seattle, WA
|
||||||
|
395,Hops & Grain Brewery,Austin, TX
|
||||||
|
396,White Flame Brewing Company,Hudsonville, MI
|
||||||
|
397,Ruhstaller Beer Company,Sacramento, CA
|
||||||
|
398,Saugatuck Brewing Company,Douglas, MI
|
||||||
|
399,Moab Brewery,Moab, UT
|
||||||
|
400,Macon Beer Company,Macon, GA
|
||||||
|
401,Amnesia Brewing Company,Washougal, WA
|
||||||
|
402,Wolverine State Brewing Company,Ann Arbor, MI
|
||||||
|
403,Red Tank Cider Company,Bend, OR
|
||||||
|
404,Cascadia Ciderworks United,Portland, OR
|
||||||
|
405,Fate Brewing Company,Boulder, CO
|
||||||
|
406,Lazy Monk Brewing,Eau Claire, WI
|
||||||
|
407,Bitter Root Brewing,Hamilton, MT
|
||||||
|
408,10 Barrel Brewing Company,Bend, OR
|
||||||
|
409,Tamarack Brewing Company,Lakeside, MT
|
||||||
|
410,New England Brewing Company,Woodbridge, CT
|
||||||
|
411,Seattle Cider Company,Seattle, WA
|
||||||
|
412,Straight to Ale,Huntsville, AL
|
||||||
|
413,Austin Beerworks,Austin, TX
|
||||||
|
414,Blue Mountain Brewery,Arrington, VA
|
||||||
|
415,Coastal Empire Beer Company,Savannah, GA
|
||||||
|
416,Jack's Hard Cider (Hauser Estate...,Biglerville, PA
|
||||||
|
417,Boulder Beer Company,Boulder, CO
|
||||||
|
418,Coalition Brewing Company,Portland, OR
|
||||||
|
419,Sanitas Brewing Company,Boulder, CO
|
||||||
|
420,Gore Range Brewery,Edwards, CO
|
||||||
|
421,Redstone Meadery,Boulder, CO
|
||||||
|
422,Blue Dog Mead,Eugene, OR
|
||||||
|
423,Hess Brewing Company,San Diego, CA
|
||||||
|
424,Wynkoop Brewing Company,Denver, CO
|
||||||
|
425,Ciderboys,Stevens Point, WI
|
||||||
|
426,Armadillo Ale Works,Denton, TX
|
||||||
|
427,Roanoke Railhouse Brewery,Roanoke, VA
|
||||||
|
428,Schlafly Brewing Company,Saint Louis, MO
|
||||||
|
429,Asher Brewing Company,Boulder, CO
|
||||||
|
430,Lost Rhino Brewing Company,Ashburn, VA
|
||||||
|
431,North Country Brewing Company,Slippery Rock, PA
|
||||||
|
432,Seabright Brewery,Santa Cruz, CA
|
||||||
|
433,French Broad Brewery,Asheville, NC
|
||||||
|
434,Angry Orchard Cider Company,Cincinnati, OH
|
||||||
|
435,Two Roads Brewing Company,Stratford, CT
|
||||||
|
436,Southern Oregon Brewing Company,Medford, OR
|
||||||
|
437,Brooklyn Brewery,Brooklyn, NY
|
||||||
|
438,The Right Brain Brewery,Traverse City, MI
|
||||||
|
439,Kona Brewing Company,Kona, HI
|
||||||
|
440,MillKing It Productions,Royal Oak, MI
|
||||||
|
441,Pateros Creek Brewing Company,Fort Collins, CO
|
||||||
|
442,O'Fallon Brewery,O'Fallon, MO
|
||||||
|
443,Marble Brewery,Albuquerque, NM
|
||||||
|
444,Big Wood Brewery,Vadnais Heights, MN
|
||||||
|
445,Howard Brewing Company,Lenoir, NC
|
||||||
|
446,Downeast Cider House,Leominster, MA
|
||||||
|
447,Swamp Head Brewery,Gainesville, FL
|
||||||
|
448,Mavericks Beer Company,Half Moon Bay, CA
|
||||||
|
449,TailGate Beer,San Diego, CA
|
||||||
|
450,Northwest Brewing Company,Pacific, WA
|
||||||
|
451,Dad & Dude's Breweria,Aurora, CO
|
||||||
|
452,Centennial Beer Company,Edwards, CO
|
||||||
|
453,Denali Brewing Company,Talkeetna, AK
|
||||||
|
454,Deschutes Brewery,Bend, OR
|
||||||
|
455,Sunken City Brewing Company,Hardy, VA
|
||||||
|
456,Lucette Brewing Company,Menominie, WI
|
||||||
|
457,The Black Tooth Brewing Company,Sheridan, WY
|
||||||
|
458,Kenai River Brewing Company,Soldotna, AK
|
||||||
|
459,River North Brewery,Denver, CO
|
||||||
|
460,Fremont Brewing Company,Seattle, WA
|
||||||
|
461,Armstrong Brewing Company,South San Francisco, CA
|
||||||
|
462,AC Golden Brewing Company,Golden, CO
|
||||||
|
463,Big Bend Brewing Company,Alpine, TX
|
||||||
|
464,Good Life Brewing Company,Bend, OR
|
||||||
|
465,Engine 15 Brewing,Jacksonville Beach, FL
|
||||||
|
466,Green Room Brewing,Jacksonville, FL
|
||||||
|
467,Brindle Dog Brewing Company,Tampa Bay, FL
|
||||||
|
468,Peace Tree Brewing Company,Knoxville, IA
|
||||||
|
469,Terrapin Brewing Company,Athens, GA
|
||||||
|
470,Pete's Brewing Company,San Antonio, TX
|
||||||
|
471,Okoboji Brewing Company,Spirit Lake, IA
|
||||||
|
472,Crystal Springs Brewing Company,Boulder, CO
|
||||||
|
473,Engine House 9,Tacoma, WA
|
||||||
|
474,Tonka Beer Company,Minnetonka, MN
|
||||||
|
475,Red Hare Brewing Company,Marietta, GA
|
||||||
|
476,Hangar 24 Craft Brewery,Redlands, CA
|
||||||
|
477,Big Elm Brewing,Sheffield, MA
|
||||||
|
478,Good People Brewing Company,Birmingham, AL
|
||||||
|
479,Heavy Seas Beer,Halethorpe, MD
|
||||||
|
480,Telluride Brewing Company,Telluride, CO
|
||||||
|
481,7 Seas Brewing Company,Gig Harbor, WA
|
||||||
|
482,Confluence Brewing Company,Des Moines, IA
|
||||||
|
483,Bale Breaker Brewing Company,Yakima, WA
|
||||||
|
484,The Manhattan Brewing Company,New York, NY
|
||||||
|
485,MacTarnahans Brewing Company,Portland, OR
|
||||||
|
486,Stillmank Beer Company,Green Bay, WI
|
||||||
|
487,Redhook Brewery,Woodinville, WA
|
||||||
|
488,Dock Street Brewery,Philadelphia, PA
|
||||||
|
489,Blue Point Brewing Company,Patchogue, NY
|
||||||
|
490,Tampa Bay Brewing Company,Tampa, FL
|
||||||
|
491,Devil's Canyon Brewery,Belmont, CA
|
||||||
|
492,Stone Coast Brewing Company,Portland, ME
|
||||||
|
493,Broken Tooth Brewing Company,Anchorage, AK
|
||||||
|
494,Seven Brides Brewery,Silverton, OR
|
||||||
|
495,Newburyport Brewing Company,Newburyport, MA
|
||||||
|
496,Dry Dock Brewing Company,Aurora, CO
|
||||||
|
497,Cans Bar and Canteen,Charlotte, NC
|
||||||
|
498,Sprecher Brewing Company,Glendale, WI
|
||||||
|
499,Wildwood Brewing Company,Stevensville, MT
|
||||||
|
500,High Noon Saloon And Brewery,Leavenworth, KS
|
||||||
|
501,Woodchuck Hard Cider,Middlebury, VT
|
||||||
|
502,Sea Dog Brewing Company,Portland, ME
|
||||||
|
503,Oskar Blues Brewery,Lyons, CO
|
||||||
|
504,Carolina Beer & Beverage,Mooresville, NC
|
||||||
|
505,Krebs Brewing Company (Pete's Pl...,Krebs, OK
|
||||||
|
506,Warbird Brewing Company,Fort Wayne, IN
|
||||||
|
507,Mudshark Brewing Company,Lake Havasu City, AZ
|
||||||
|
508,Spilker Ales,Cortland, NE
|
||||||
|
509,Wingman Brewers,Tacoma, WA
|
||||||
|
510,Kettle House Brewing Company,Missoula, MT
|
||||||
|
511,Sherwood Forest Brewers,Marlborough, MA
|
||||||
|
512,Cottrell Brewing,Pawcatuck, CT
|
||||||
|
513,Arctic Craft Brewery,Colorado Springs, CO
|
||||||
|
514,Monkey Paw Pub & Brewery,San Diego, CA
|
||||||
|
515,Crabtree Brewing Company,Greeley, CO
|
||||||
|
516,Emerald City Beer Company,Seattle, WA
|
||||||
|
517,Butcher's Brewing,Carlsbad, CA
|
||||||
|
518,New South Brewing Company,Myrtle Beach, SC
|
||||||
|
519,Big River Brewing Company,Chattanooga, TN
|
||||||
|
520,Twisted Pine Brewing Company,Boulder, CO
|
||||||
|
521,Flying Dog Brewery,Frederick, MD
|
||||||
|
522,Uncommon Brewers,Santa Cruz, CA
|
||||||
|
523,Aspen Brewing Company,Aspen, CO
|
||||||
|
524,Triangle Brewing Company,Durham, NC
|
||||||
|
525,Bomb Beer Company,New York, NY
|
||||||
|
526,Churchkey Can Company,Seattle, WA
|
||||||
|
527,Intuition Ale Works,Jacksonville, FL
|
||||||
|
528,Asheville Brewing Company,Asheville, NC
|
||||||
|
529,Northwoods Brewpub,Eau Claire, WI
|
||||||
|
530,Buckbean Brewing Company,Reno, NV
|
||||||
|
531,Dolores River Brewery,Dolores, CO
|
||||||
|
532,Flat Rock Brewing Company,Smithton, PA
|
||||||
|
533,Abita Brewing Company,Abita Springs, LA
|
||||||
|
534,Mammoth Brewing Company,Mammoth Lakes, CA
|
||||||
|
535,Harvest Moon Brewing Company,Belt, MT
|
||||||
|
536,Grand Canyon Brewing Company,Williams, AZ
|
||||||
|
537,Lewis and Clark Brewing Company,Helena, MT
|
||||||
|
538,Dundee Brewing Company,Rochester, NY
|
||||||
|
539,Twin Lakes Brewing Company,Greenville, DE
|
||||||
|
540,Mother Earth Brewing Company,Kinston, NC
|
||||||
|
541,Arcadia Brewing Company,Battle Creek, MI
|
||||||
|
542,Angry Minnow Brewing Company,Hayward, WI
|
||||||
|
543,Great Northern Brewing Company,Whitefish, MT
|
||||||
|
544,Pyramid Breweries,Seattle, WA
|
||||||
|
545,Lancaster Brewing Company,Lancaster, PA
|
||||||
|
546,Upstate Brewing Company,Elmira, NY
|
||||||
|
547,Moat Mountain Smoke House & Brew...,North Conway, NH
|
||||||
|
548,Prescott Brewing Company,Prescott, AZ
|
||||||
|
549,Mogollon Brewing Company,Flagstaff, AZ
|
||||||
|
550,Wind River Brewing Company,Pinedale, WY
|
||||||
|
551,Silverton Brewery,Silverton, CO
|
||||||
|
552,Mickey Finn's Brewery,Libertyville, IL
|
||||||
|
553,Covington Brewhouse,Covington, LA
|
||||||
|
554,Dave's Brewfarm,Wilson, WI
|
||||||
|
555,Ukiah Brewing Company,Ukiah, CA
|
||||||
|
556,Butternuts Beer and Ale,Garrattsville, NY
|
||||||
|
557,Sleeping Lady Brewing Company,Anchorage, AK
|
||||||
|
162686
misc/raw-data/breweries.json
Normal file
162686
misc/raw-data/breweries.json
Normal file
File diff suppressed because it is too large
Load Diff
578
misc/raw-data/ontariobreweries.json
Normal file
578
misc/raw-data/ontariobreweries.json
Normal file
@@ -0,0 +1,578 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"text": "100 Acre Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/100-acre-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "All My Friends Beer Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/all-my-friends-beer-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "All or Nothing Brewhouse",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/all-or-nothing-brewhouse/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Anderson Craft Ales",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/anderson-craft-ales/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Badlands Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/badlands-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Bancroft Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/bancroft-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Banded Goose Brewing Comany",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/banded-goose-brewing-comany/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Beau’s Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/beaus-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "BeerLab! London",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/beerlab-london/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Bellwoods Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/bellwoods-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Bench Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/bench-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Beyond The Pale Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/beyond-the-pale-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Bicycle Craft Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/bicycle-craft-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Big Rig Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/big-rig-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Big Rock Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/big-rock-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Black Gold Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/black-gold-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Black Oak Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/black-oak-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Block 3 Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/block-3-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Blood Brothers Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/blood-brothers-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Bobcaygeon Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/bobcaygeon-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Boshkung Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/boshkung-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Brauwerk Hoffman – Rockland",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/brauwerk-hoffman-rockland/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Bridge Masters Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/bridge-masters-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Broadhead Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/broadhead-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Broken Rail Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/broken-rail-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Burdock Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/burdock-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "C’est What Durham Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/cest-what-durham-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Calabogie Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/calabogie-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Cameron’s Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/camerons-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Canvas Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/canvas-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Caps Off Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/caps-off-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Century Barn Brewing & Beverage Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/century-barn-brewing-and-beverage-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Chronicle Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/chronicle-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Clifford Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/clifford-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Cold Bear Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/cold-bear-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Collective Arts Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/collective-arts-brewing-ltd/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Common Good Beer Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/common-good-beer-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Couchiching Craft Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/couchiching-craft-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Cowbell Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/cowbell-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Cured Craft Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/cured-craft-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Daft Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/daft-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Dog House Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/dog-house-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Dominion City Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/dominion-city-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Eastbound Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/eastbound-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Equals Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/equals-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Fairweather Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/fairweather-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Farm League Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/farm-league-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Fixed Gear Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/fixed-gear-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Flying Monkeys Craft Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/flying-monkeys-craft-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Focal Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/focal-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Foundry Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/foundry-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Four Fathers Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/four-fathers-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Frank Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/frank-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Freddy’s",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/freddys/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Full Beard Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/1068-2/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Furnace Room Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/furnace-room-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Gateway City Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/gateway-city-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Glasstown Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/glasstown-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Godspeed Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/godspeed-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Goldenfield Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/goldenfield-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Grand River Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/grand-river-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Granite Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/granite-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Great Lakes Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/great-lakes-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Haliburton Highlands Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/haliburton-highlands-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Imperial City Brew House",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/imperial-city-brew-house/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Indie Ale House",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/indie-ale-house/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Jobsite Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/jobsite-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Kichesippi Beer Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/kichesippi-beer-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Kick and Push Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/kick-and-push-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Lake of Bays Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/lake-of-bays-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Lake Of The Woods Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/lake-of-the-woods-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Left Field Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/left-field-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Lightcaster Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/lightcaster-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "MacKinnon Brothers Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/mackinnon-brothers-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Maclean’s Ales Inc.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/macleans-ales-inc/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Magnotta Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/magnotta-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Market Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/market-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Mascot Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/mascot-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Matron Fine Beer",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/matron-fine-beer/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Meyers Creek Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/meyers-creek-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Midtown Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/midtown-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Miski Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/miski-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Muddy York Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/muddy-york-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Muskoka Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/muskoka-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Natterjack Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/natterjack-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Newark Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/newark-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Niagara Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/niagara-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Niagara College Teaching Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/niagara-college-teaching-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Niagara Oast House Brewers",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/niagara-oast-house-brewers/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Nickel Brook Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/nickel-brook-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Northern Superior Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/northern-superior-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Old Credit Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/old-credit-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Old Flame Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/1239-2/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Orléans Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/orleans-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Overflow Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/overflow-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Parsons Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/parsons-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Perth Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/perth-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Prince Eddy’s Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/prince-eddys-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Quayle’s Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/quayles-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Quetico Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/quetico-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Railway City Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/railway-city-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Ramblin’ Road Brewery Farm",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/ramblin-road-brewery-farm/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Red Barn Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/red-barn-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Refined Fool Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/refined-fool-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Rouge River Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/rouge-river-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Royal City Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/royal-city-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Sassy Britches Brewing Co Ltd",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/sassy-britches-brewing-co-ltd/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Sawdust City Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/sawdust-city-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Shawn & Ed Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/shawn-ed-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Silversmith Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/silversmith-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Slake Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/slake-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Sleeping Giant Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/sleeping-giant-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Something in the Water Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/something-in-the-water-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Sonnen Hill Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/sonnen-hill-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Sons of Kent Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/sons-of-kent-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Spark Beer",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/spark-beer/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Split Rail Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/split-rail-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Stack Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/stack-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Steam Whistle Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/steam-whistle-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Steel Wheel Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/steel-wheel-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Stonehooker Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/stonehooker-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Stonepicker Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/stonepicker-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Stray Dog Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/stray-dog-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "The Exchange Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/the-exchange-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "The Grove Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/the-grove-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "The Second Wedge Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/the-second-wedge-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Thornbury Village Craft Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/thornbury-village-craft-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Three Sheets Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/three-sheets-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Tooth and Nail Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/tooth-and-nail-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Torched Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/torched-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Town Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/town-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Trestle Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/trestle-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "True History Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/true-history-brewing/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Upper Thames Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/upper-thames-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Vimy Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/vimy-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Walkerville Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/walkerville-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Wave Maker Craft Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/wave-maker-craft-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Wellington Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/wellington-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Whiprsnapr Brewing Co.",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/whiprsnapr-brewing-co/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Whiskeyjack Beer Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/whiskeyjack-beer-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Whitewater Brewing Company",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/whitewater-brewing-company/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Willibald Farm Distillery & Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/willibald-farm-distillery-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Windmill Brewery",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/windmill-brewery/"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"text": "Wishbone Brewing",
|
||||||
|
"href": "https://ontariocraftbrewers.com/brewery-profile/wishbone-brewing/"
|
||||||
|
}
|
||||||
|
]
|
||||||
3
pipeline/.gitignore
vendored
Normal file
3
pipeline/.gitignore
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
dist
|
||||||
|
build
|
||||||
|
data
|
||||||
113
pipeline/CMakeLists.txt
Normal file
113
pipeline/CMakeLists.txt
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
cmake_minimum_required(VERSION 3.20)
|
||||||
|
project(biergarten-pipeline VERSION 0.1.0 LANGUAGES CXX)
|
||||||
|
|
||||||
|
cmake_policy(SET CMP0167 NEW)
|
||||||
|
|
||||||
|
set(CMAKE_CXX_STANDARD 23)
|
||||||
|
set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
||||||
|
set(CMAKE_CXX_EXTENSIONS OFF)
|
||||||
|
|
||||||
|
find_package(CURL REQUIRED)
|
||||||
|
find_package(Boost REQUIRED COMPONENTS unit_test_framework)
|
||||||
|
find_package(SQLite3 REQUIRED)
|
||||||
|
|
||||||
|
include(FetchContent)
|
||||||
|
|
||||||
|
FetchContent_Declare(
|
||||||
|
nlohmann_json
|
||||||
|
GIT_REPOSITORY https://github.com/nlohmann/json.git
|
||||||
|
GIT_TAG v3.11.3
|
||||||
|
)
|
||||||
|
FetchContent_MakeAvailable(nlohmann_json)
|
||||||
|
|
||||||
|
# TODO: Integrate real llama.cpp when generator is ready to use actual models
|
||||||
|
# For now, using mocked brewery generation in generator.cpp
|
||||||
|
|
||||||
|
# SQLite for in-memory database
|
||||||
|
find_package(SQLite3 REQUIRED)
|
||||||
|
|
||||||
|
file(GLOB_RECURSE SOURCES CONFIGURE_DEPENDS
|
||||||
|
src/*.cpp
|
||||||
|
)
|
||||||
|
|
||||||
|
add_executable(biergarten-pipeline ${SOURCES})
|
||||||
|
|
||||||
|
target_include_directories(biergarten-pipeline
|
||||||
|
PRIVATE
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/includes
|
||||||
|
)
|
||||||
|
|
||||||
|
target_link_libraries(biergarten-pipeline
|
||||||
|
PRIVATE
|
||||||
|
CURL::libcurl
|
||||||
|
nlohmann_json::nlohmann_json
|
||||||
|
Boost::unit_test_framework
|
||||||
|
SQLite::SQLite3
|
||||||
|
)
|
||||||
|
|
||||||
|
target_compile_options(biergarten-pipeline PRIVATE
|
||||||
|
$<$<CXX_COMPILER_ID:GNU,Clang>:
|
||||||
|
-Wall
|
||||||
|
-Wextra
|
||||||
|
-Wpedantic
|
||||||
|
-Wshadow
|
||||||
|
-Wconversion
|
||||||
|
-Wsign-conversion
|
||||||
|
>
|
||||||
|
$<$<CXX_COMPILER_ID:MSVC>:
|
||||||
|
/W4
|
||||||
|
/WX
|
||||||
|
>
|
||||||
|
)
|
||||||
|
|
||||||
|
add_custom_command(TARGET biergarten-pipeline POST_BUILD
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E make_directory
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/output
|
||||||
|
COMMENT "Creating output/ directory for seed SQL files"
|
||||||
|
)
|
||||||
|
|
||||||
|
find_program(VALGRIND valgrind)
|
||||||
|
if(VALGRIND)
|
||||||
|
add_custom_target(memcheck
|
||||||
|
COMMAND ${VALGRIND}
|
||||||
|
--leak-check=full
|
||||||
|
--error-exitcode=1
|
||||||
|
$<TARGET_FILE:biergarten-pipeline> --help
|
||||||
|
DEPENDS biergarten-pipeline
|
||||||
|
COMMENT "Running Valgrind memcheck"
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
include(CTest)
|
||||||
|
|
||||||
|
if(BUILD_TESTING)
|
||||||
|
find_package(Boost REQUIRED COMPONENTS unit_test_framework)
|
||||||
|
|
||||||
|
file(GLOB_RECURSE TEST_SOURCES CONFIGURE_DEPENDS
|
||||||
|
tests/*.cpp
|
||||||
|
tests/*.cc
|
||||||
|
tests/*.cxx
|
||||||
|
)
|
||||||
|
|
||||||
|
if(TEST_SOURCES)
|
||||||
|
add_executable(biergarten-pipeline-tests ${TEST_SOURCES})
|
||||||
|
|
||||||
|
target_include_directories(biergarten-pipeline-tests
|
||||||
|
PRIVATE
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/include
|
||||||
|
)
|
||||||
|
|
||||||
|
target_link_libraries(biergarten-pipeline-tests
|
||||||
|
PRIVATE
|
||||||
|
Boost::unit_test_framework
|
||||||
|
CURL::libcurl
|
||||||
|
nlohmann_json::nlohmann_json
|
||||||
|
llama
|
||||||
|
)
|
||||||
|
|
||||||
|
add_test(
|
||||||
|
NAME biergarten-pipeline-tests
|
||||||
|
COMMAND biergarten-pipeline-tests
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
1
pipeline/README.md
Normal file
1
pipeline/README.md
Normal file
@@ -0,0 +1 @@
|
|||||||
|
|
||||||
111
pipeline/includes/data_downloader.h
Normal file
111
pipeline/includes/data_downloader.h
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
/**
|
||||||
|
* @file data_downloader.h
|
||||||
|
* @brief Download geographic data from GitHub repositories using libcurl.
|
||||||
|
*
|
||||||
|
* Provides functionality to fetch JSON data from GitHub using libcurl, with
|
||||||
|
* support for commit-based versioning to ensure reproducible builds. Downloads
|
||||||
|
* are cached to avoid repeated network requests.
|
||||||
|
*
|
||||||
|
* Example usage:
|
||||||
|
* @code
|
||||||
|
* DataDownloader downloader;
|
||||||
|
* std::string jsonPath = downloader.DownloadCountriesDatabase(
|
||||||
|
* "/tmp/countries-data.json", // local cache path
|
||||||
|
* "c5eb7772" // optional commit hash or HEAD
|
||||||
|
* );
|
||||||
|
* // Now use jsonPath with JsonLoader::LoadWorldCities(jsonPath, db)
|
||||||
|
* @endcode
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef DATA_DOWNLOADER_H
|
||||||
|
#define DATA_DOWNLOADER_H
|
||||||
|
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @class DataDownloader
|
||||||
|
* @brief Manages downloading and caching of geographic data from GitHub.
|
||||||
|
*
|
||||||
|
* This class encapsulates libcurl networking operations for reproducible
|
||||||
|
* data fetching. All methods are non-blocking and synchronous.
|
||||||
|
*
|
||||||
|
* @note Requires libcurl to be available at runtime.
|
||||||
|
* @note GitHub raw content CDN is used for efficient downloads.
|
||||||
|
*/
|
||||||
|
class DataDownloader {
|
||||||
|
public:
|
||||||
|
/**
|
||||||
|
* @brief Default constructor.
|
||||||
|
*
|
||||||
|
* Initializes the downloader without any specific state. The downloader
|
||||||
|
* is ready to use immediately.
|
||||||
|
*/
|
||||||
|
DataDownloader();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Destructor.
|
||||||
|
*
|
||||||
|
* Cleans up any resources. No explicit cleanup needed beyond destruction.
|
||||||
|
*/
|
||||||
|
~DataDownloader();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Download the countries+states+cities JSON database from GitHub.
|
||||||
|
*
|
||||||
|
* Downloads the geographic data from the
|
||||||
|
* dr5hn/countries-states-cities-database repository. If the file already
|
||||||
|
* exists at cachePath, it is used directly without downloading again.
|
||||||
|
*
|
||||||
|
* The download URL format is:
|
||||||
|
* @verbatim
|
||||||
|
* https://raw.githubusercontent.com/dr5hn/countries-states-cities-database/
|
||||||
|
* {commit}/json/countries+states+cities.json
|
||||||
|
* @endverbatim
|
||||||
|
*
|
||||||
|
* @param cachePath Local filesystem path where the JSON file should be
|
||||||
|
* stored. If the file already exists, download is skipped.
|
||||||
|
* @param commit Git commit hash or branch name (default: "c5eb7772").
|
||||||
|
* Examples: "HEAD", "main", "c5eb7772",
|
||||||
|
* "c5eb7772225f6b1802a54f39adb8c73464a85be1a"
|
||||||
|
*
|
||||||
|
* @return The file path where JSON was saved (same as cachePath).
|
||||||
|
*
|
||||||
|
* @throws std::runtime_error if:
|
||||||
|
* - Network download fails
|
||||||
|
* - File cannot be written to cachePath
|
||||||
|
* - Commit hash is invalid (404 on GitHub)
|
||||||
|
*
|
||||||
|
* Example with default commit (stable v2026-03-28):
|
||||||
|
* @code
|
||||||
|
* std::string path =
|
||||||
|
* downloader.DownloadCountriesDatabase("/tmp/data.json");
|
||||||
|
* @endcode
|
||||||
|
*
|
||||||
|
* Example with custom commit:
|
||||||
|
* @code
|
||||||
|
* std::string path = downloader.DownloadCountriesDatabase(
|
||||||
|
* "/tmp/data.json",
|
||||||
|
* "main" // Download latest from main branch
|
||||||
|
* );
|
||||||
|
* @endcode
|
||||||
|
*/
|
||||||
|
std::string DownloadCountriesDatabase(
|
||||||
|
const std::string &cachePath,
|
||||||
|
const std::string &commit = "c5eb7772" // Stable commit: 2026-03-28 export
|
||||||
|
);
|
||||||
|
|
||||||
|
private:
|
||||||
|
/**
|
||||||
|
* @brief Check if a file already exists at the given path.
|
||||||
|
*
|
||||||
|
* Used internally to implement cache-hit logic. No download occurs if
|
||||||
|
* the file already exists.
|
||||||
|
*
|
||||||
|
* @param filePath Path to check.
|
||||||
|
* @return True if file exists and is readable, false otherwise.
|
||||||
|
*/
|
||||||
|
bool FileExists(const std::string &filePath) const;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // DATA_DOWNLOADER_H
|
||||||
102
pipeline/includes/database.h
Normal file
102
pipeline/includes/database.h
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
#include <sqlite3.h>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
/// @struct Country
|
||||||
|
/// @brief Represents a country with geographic identifiers
|
||||||
|
struct Country {
|
||||||
|
int id;
|
||||||
|
std::string name;
|
||||||
|
std::string iso2; ///< 2-letter ISO code (e.g., "US", "CA")
|
||||||
|
std::string iso3; ///< 3-letter ISO code (e.g., "USA", "CAN")
|
||||||
|
};
|
||||||
|
|
||||||
|
/// @struct State
|
||||||
|
/// @brief Represents a state or province with geographic identifiers
|
||||||
|
struct State {
|
||||||
|
int id;
|
||||||
|
std::string name;
|
||||||
|
std::string iso2; ///< 2-letter state code (e.g., "CA", "ON")
|
||||||
|
int countryId;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @class SqliteDatabase
|
||||||
|
* @brief Thread-safe in-memory SQLite database wrapper for geographic data
|
||||||
|
*
|
||||||
|
* Manages a local in-memory SQLite database with countries, states, and cities.
|
||||||
|
* All write operations are serialized via mutex to enable safe concurrent
|
||||||
|
* access from multiple threads. Uses INSERT OR IGNORE for idempotent
|
||||||
|
* operations.
|
||||||
|
*
|
||||||
|
* Schema Relationships:
|
||||||
|
* countries (id, name, iso2, iso3)
|
||||||
|
* ↓ (one-to-many)
|
||||||
|
* states (id, country_id, name, iso2)
|
||||||
|
* ↓ (one-to-many)
|
||||||
|
* cities (id, state_id, country_id, name, latitude, longitude)
|
||||||
|
*/
|
||||||
|
class SqliteDatabase {
|
||||||
|
private:
|
||||||
|
sqlite3 *db = nullptr; ///< SQLite database connection handle
|
||||||
|
std::mutex dbMutex; ///< Protects all database operations from race conditions
|
||||||
|
|
||||||
|
/// @brief Creates the schema with three related tables and foreign keys
|
||||||
|
void InitializeSchema();
|
||||||
|
|
||||||
|
public:
|
||||||
|
/// @brief Destructor: safely closes the database connection
|
||||||
|
~SqliteDatabase();
|
||||||
|
|
||||||
|
/// @brief Opens an in-memory SQLite database and initializes the schema
|
||||||
|
void Initialize();
|
||||||
|
|
||||||
|
/// @brief Inserts a country record
|
||||||
|
/// @param id Unique country identifier
|
||||||
|
/// @param name Country name
|
||||||
|
/// @param iso2 2-letter ISO country code
|
||||||
|
/// @param iso3 3-letter ISO country code
|
||||||
|
/// @note Thread-safe: uses mutex lock. Idempotent: INSERT OR IGNORE prevents
|
||||||
|
/// duplicates
|
||||||
|
void InsertCountry(int id, const std::string &name, const std::string &iso2,
|
||||||
|
const std::string &iso3);
|
||||||
|
|
||||||
|
/// @brief Inserts a state/province record
|
||||||
|
/// @param id Unique state identifier
|
||||||
|
/// @param countryId Foreign key reference to parent country
|
||||||
|
/// @param name State/province name
|
||||||
|
/// @param iso2 2-letter state code (e.g., "CA", "ON")
|
||||||
|
/// @note Thread-safe and idempotent via mutex and INSERT OR IGNORE
|
||||||
|
void InsertState(int id, int countryId, const std::string &name,
|
||||||
|
const std::string &iso2);
|
||||||
|
|
||||||
|
/// @brief Inserts a city record with geographic coordinates
|
||||||
|
/// @param id Unique city identifier
|
||||||
|
/// @param stateId Foreign key reference to parent state
|
||||||
|
/// @param countryId Foreign key reference to parent country
|
||||||
|
/// @param name City name
|
||||||
|
/// @param latitude Geographic latitude coordinate (WGS84)
|
||||||
|
/// @param longitude Geographic longitude coordinate (WGS84)
|
||||||
|
/// @note Thread-safe and idempotent. Called by multithreaded JSON loader.
|
||||||
|
void InsertCity(int id, int stateId, int countryId, const std::string &name,
|
||||||
|
double latitude, double longitude);
|
||||||
|
|
||||||
|
/// @brief Queries all cities from the database
|
||||||
|
/// @return Vector of (city_id, city_name) pairs sorted alphabetically
|
||||||
|
std::vector<std::pair<int, std::string>> QueryCities();
|
||||||
|
|
||||||
|
/// @brief Queries all countries from the database with ISO codes
|
||||||
|
/// @param limit Maximum number of records to return (0 = all)
|
||||||
|
/// @return Vector of Country structs (includes id, name, iso2, iso3) sorted
|
||||||
|
/// alphabetically
|
||||||
|
std::vector<Country> QueryCountries(int limit = 0);
|
||||||
|
|
||||||
|
/// @brief Queries all states from the database with ISO codes
|
||||||
|
/// @param limit Maximum number of records to return (0 = all)
|
||||||
|
/// @return Vector of State structs (includes id, name, iso2, countryId)
|
||||||
|
/// sorted alphabetically
|
||||||
|
std::vector<State> QueryStates(int limit = 0);
|
||||||
|
};
|
||||||
59
pipeline/includes/generator.h
Normal file
59
pipeline/includes/generator.h
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @class LlamaBreweryGenerator
|
||||||
|
* @brief Generates brewery names and descriptions for cities
|
||||||
|
*
|
||||||
|
* Currently provides a deterministic mock implementation that generates
|
||||||
|
* brewery names and descriptions based on city name hashing.
|
||||||
|
*
|
||||||
|
* Design Pattern: Strategy pattern ready for swapping real llama.cpp
|
||||||
|
* implementation later. The LoadModel() and GenerateBrewery() interface
|
||||||
|
* will remain the same once actual LM inference is integrated.
|
||||||
|
*
|
||||||
|
* Mock Implementation: Uses std::hash to deterministically map city names
|
||||||
|
* to brewery templates, ensuring reproducible results for testing.
|
||||||
|
*/
|
||||||
|
class LlamaBreweryGenerator {
|
||||||
|
private:
|
||||||
|
/// Adjectives for brewery names (e.g., "Craft", "Heritage", etc.)
|
||||||
|
const std::vector<std::string> breweryAdjectives = {
|
||||||
|
"Craft", "Heritage", "Local", "Artisan",
|
||||||
|
"Pioneer", "Golden", "Modern", "Classic"};
|
||||||
|
|
||||||
|
/// Nouns for brewery names (e.g., "Brewing Co.", "Brewery", etc.)
|
||||||
|
const std::vector<std::string> breweryNouns = {
|
||||||
|
"Brewing Co.", "Brewery", "Bier Haus", "Taproom",
|
||||||
|
"Works", "House", "Fermentery", "Ale Co."};
|
||||||
|
|
||||||
|
/// Pre-written brewery descriptions (currently hand-crafted)
|
||||||
|
const std::vector<std::string> descriptions = {
|
||||||
|
"Handcrafted pale ales and seasonal IPAs with local ingredients.",
|
||||||
|
"Traditional lagers and experimental sours in small batches.",
|
||||||
|
"Award-winning stouts and wildly hoppy blonde ales.",
|
||||||
|
"Craft brewery specializing in Belgian-style triples and dark porters.",
|
||||||
|
"Modern brewery blending tradition with bold experimental flavors."};
|
||||||
|
|
||||||
|
public:
|
||||||
|
/// @struct Brewery
|
||||||
|
/// @brief Output structure for generated brewery data
|
||||||
|
struct Brewery {
|
||||||
|
std::string name; ///< Generated brewery name (e.g., "Craft Brewing Co.")
|
||||||
|
std::string description; ///< Short description of brewery style/offerings
|
||||||
|
};
|
||||||
|
|
||||||
|
/// @brief Loads a language model (currently mocked)
|
||||||
|
/// @param modelPath Path to GGUF model file (not used in mock)
|
||||||
|
/// @note In real implementation, loads llama.cpp model into memory
|
||||||
|
void LoadModel(const std::string &modelPath);
|
||||||
|
|
||||||
|
/// @brief Generates a brewery name and description for a city
|
||||||
|
/// @param cityName City name to generate brewery for
|
||||||
|
/// @param seed Integer seed (used for deterministic output in mock)
|
||||||
|
/// @return Brewery struct with name and description
|
||||||
|
/// @note Deterministic: same cityName+seed always produces same brewery
|
||||||
|
Brewery GenerateBrewery(const std::string &cityName, int seed);
|
||||||
|
};
|
||||||
85
pipeline/includes/json_loader.h
Normal file
85
pipeline/includes/json_loader.h
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "database.h"
|
||||||
|
#include <nlohmann/json.hpp>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
using json = nlohmann::json;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @class JsonLoader
|
||||||
|
* @brief Loads world geographic data from JSON file into SQLite database
|
||||||
|
*
|
||||||
|
* Handles parsing and population of world cities, states, and countries from
|
||||||
|
* a structured JSON source file. The loader uses parallel threads to chunk
|
||||||
|
* the city records and maximize database insertion throughput.
|
||||||
|
*
|
||||||
|
* Input Format (JSON Structure):
|
||||||
|
* @code
|
||||||
|
* {
|
||||||
|
* "countries": [
|
||||||
|
* {"id": 1, "name": "Canada", "iso2": "CA", "iso3": "CAN"},
|
||||||
|
* ...
|
||||||
|
* ],
|
||||||
|
* "states": [
|
||||||
|
* {"id": 1, "country_id": 1, "name": "Ontario", "iso2": "ON"},
|
||||||
|
* ...
|
||||||
|
* ],
|
||||||
|
* "cities": [
|
||||||
|
* {"id": 1, "state_id": 1, "country_id": 1, "name": "Toronto",
|
||||||
|
* "latitude": 43.6532, "longitude": -79.3832},
|
||||||
|
* ...
|
||||||
|
* ]
|
||||||
|
* }
|
||||||
|
* @endcode
|
||||||
|
*
|
||||||
|
* Performance Characteristics:
|
||||||
|
* - Reads entire JSON file into memory (nlohmann/json parser)
|
||||||
|
* - Iterates through countries: typically 200+ records
|
||||||
|
* - Iterates through states: typically 3000+ records
|
||||||
|
* - Iterates through cities: typically 50,000+ records (MAJOR DATASET)
|
||||||
|
* - Uses multithreading to chunk city insertion across threads
|
||||||
|
* - Thread pool size defaults to number of CPU cores
|
||||||
|
*
|
||||||
|
* Multithreading Strategy:
|
||||||
|
* - Divides cities into N chunks (N = CPU core count)
|
||||||
|
* - Each thread processes one chunk sequentially
|
||||||
|
* - Database has mutex protection for thread-safe concurrent access
|
||||||
|
* - Allows safe parallel writing to same SQLite database
|
||||||
|
*
|
||||||
|
* Example Usage:
|
||||||
|
* @code
|
||||||
|
* SqliteDatabase db;
|
||||||
|
* db.Initialize();
|
||||||
|
* JsonLoader::LoadWorldCities("../data/world_city_data.json", db);
|
||||||
|
* // Database now contains all countries, states, and cities
|
||||||
|
* @endcode
|
||||||
|
*/
|
||||||
|
class JsonLoader {
|
||||||
|
public:
|
||||||
|
/// @brief Loads world geographic data from JSON and populates database
|
||||||
|
///
|
||||||
|
/// Process:
|
||||||
|
/// 1. Reads and parses entire JSON file
|
||||||
|
/// 2. Inserts all countries into database (typically 200-250 records)
|
||||||
|
/// 3. Inserts all states/provinces (typically 3000+ records)
|
||||||
|
/// 4. Spawns worker threads to insert cities (typically 50,000+ records)
|
||||||
|
/// 5. Waits for all threads to complete
|
||||||
|
/// 6. Prints statistics about loaded data
|
||||||
|
///
|
||||||
|
/// @param jsonPath Filesystem path to world_city_data.json
|
||||||
|
/// @param db Reference to initialized SqliteDatabase to populate
|
||||||
|
///
|
||||||
|
/// @throws std::runtime_error if JSON file cannot be read or parsed
|
||||||
|
/// @throws std::runtime_error if database insertion fails
|
||||||
|
///
|
||||||
|
/// Output Examples:
|
||||||
|
/// @code
|
||||||
|
/// Loading JSON: ../data/world_city_data.json
|
||||||
|
/// Loaded countries: 250
|
||||||
|
/// Loaded states: 3500
|
||||||
|
/// Loaded cities: 52000
|
||||||
|
/// ✓ World city data loaded successfully
|
||||||
|
/// @endcode
|
||||||
|
static void LoadWorldCities(const std::string &jsonPath, SqliteDatabase &db);
|
||||||
|
};
|
||||||
163
pipeline/src/data_downloader.cpp
Normal file
163
pipeline/src/data_downloader.cpp
Normal file
@@ -0,0 +1,163 @@
|
|||||||
|
/**
|
||||||
|
* @file data_downloader.cpp
|
||||||
|
* @brief Implementation of DataDownloader using libcurl for HTTP downloads.
|
||||||
|
*
|
||||||
|
* Provides robust downloading with proper error handling, timeout management,
|
||||||
|
* and local caching to avoid repeated network calls. Uses GitHub's raw content
|
||||||
|
* CDN for reliable high-bandwidth downloads.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "data_downloader.h"
|
||||||
|
#include <cstdio>
|
||||||
|
#include <curl/curl.h>
|
||||||
|
#include <fstream>
|
||||||
|
#include <iostream>
|
||||||
|
#include <sstream>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Callback function for libcurl to write downloaded content to file.
|
||||||
|
*
|
||||||
|
* This callback is invoked repeatedly by curl as data arrives over the network.
|
||||||
|
* Each invocation contains a chunk of the response body. The function writes
|
||||||
|
* the content to the output file stream.
|
||||||
|
*
|
||||||
|
* @param contents Pointer to buffer containing data chunk.
|
||||||
|
* @param size Element size (always 1 for text).
|
||||||
|
* @param nmemb Number of elements in chunk.
|
||||||
|
* @param userp Opaque pointer to std::ofstream (FILE*).
|
||||||
|
*
|
||||||
|
* @return Total bytes written. Must match (size * nmemb) for success;
|
||||||
|
* returning less signals an error to curl.
|
||||||
|
*
|
||||||
|
* @note libcurl requires this signature: (char* ptr, size_t size, size_t nmemb,
|
||||||
|
* void* userp)
|
||||||
|
*/
|
||||||
|
static size_t WriteCallback(void *contents, size_t size, size_t nmemb,
|
||||||
|
void *userp) {
|
||||||
|
// Calculate total bytes in this chunk
|
||||||
|
size_t realsize = size * nmemb;
|
||||||
|
|
||||||
|
// Cast userp back to ofstream
|
||||||
|
std::ofstream *outFile = static_cast<std::ofstream *>(userp);
|
||||||
|
|
||||||
|
// Write to file
|
||||||
|
outFile->write(static_cast<char *>(contents), realsize);
|
||||||
|
|
||||||
|
// Return actual bytes written (success = requested amount)
|
||||||
|
return realsize;
|
||||||
|
}
|
||||||
|
|
||||||
|
DataDownloader::DataDownloader() {
|
||||||
|
// curl_global_init is called by user or external subsystem in a thread-safe
|
||||||
|
// manner. Not calling it here to avoid multiple initialization in
|
||||||
|
// multi-downloader scenarios.
|
||||||
|
}
|
||||||
|
|
||||||
|
DataDownloader::~DataDownloader() {
|
||||||
|
// No explicit cleanup needed; curl_global_cleanup managed externally.
|
||||||
|
}
|
||||||
|
|
||||||
|
bool DataDownloader::FileExists(const std::string &filePath) const {
|
||||||
|
// Use POSIX stat() to check file existence without opening it
|
||||||
|
struct stat buffer;
|
||||||
|
return (stat(filePath.c_str(), &buffer) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string
|
||||||
|
DataDownloader::DownloadCountriesDatabase(const std::string &cachePath,
|
||||||
|
const std::string &commit) {
|
||||||
|
// Check if file already cached locally
|
||||||
|
if (FileExists(cachePath)) {
|
||||||
|
std::cout << "[DataDownloader] Cache hit: " << cachePath << std::endl;
|
||||||
|
return cachePath;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Construct download URL
|
||||||
|
// Full commit hash is accepted, but only first 7 chars (short hash) are
|
||||||
|
// needed
|
||||||
|
std::string shortCommit = commit;
|
||||||
|
if (commit.length() > 7) {
|
||||||
|
shortCommit = commit.substr(0, 7);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string url = "https://raw.githubusercontent.com/dr5hn/"
|
||||||
|
"countries-states-cities-database/" +
|
||||||
|
shortCommit + "/json/countries+states+cities.json";
|
||||||
|
|
||||||
|
std::cout << "[DataDownloader] Downloading: " << url << std::endl;
|
||||||
|
|
||||||
|
// Initialize curl handle
|
||||||
|
CURL *curl = curl_easy_init();
|
||||||
|
if (!curl) {
|
||||||
|
throw std::runtime_error("[DataDownloader] Failed to initialize libcurl");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Open output file for writing (binary mode to preserve exact bytes)
|
||||||
|
std::ofstream outFile(cachePath, std::ios::binary);
|
||||||
|
if (!outFile.is_open()) {
|
||||||
|
curl_easy_cleanup(curl);
|
||||||
|
throw std::runtime_error("[DataDownloader] Cannot open file for writing: " +
|
||||||
|
cachePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Configure curl for download
|
||||||
|
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
|
||||||
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, static_cast<void *>(&outFile));
|
||||||
|
|
||||||
|
// Set reasonable timeout (30 seconds for initial connection, 300s for
|
||||||
|
// transfer)
|
||||||
|
curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 30L);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_TIMEOUT, 300L);
|
||||||
|
|
||||||
|
// Follow redirects (CDN may redirect)
|
||||||
|
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_MAXREDIRS, 5L);
|
||||||
|
|
||||||
|
// Use gzip compression if server supports it
|
||||||
|
curl_easy_setopt(curl, CURLOPT_ACCEPT_ENCODING, "gzip");
|
||||||
|
|
||||||
|
// Set user agent to identify the application
|
||||||
|
curl_easy_setopt(curl, CURLOPT_USERAGENT, "biergarten-pipeline/0.1.0");
|
||||||
|
|
||||||
|
// Perform the download
|
||||||
|
CURLcode res = curl_easy_perform(curl);
|
||||||
|
outFile.close();
|
||||||
|
|
||||||
|
// Check for curl errors
|
||||||
|
if (res != CURLE_OK) {
|
||||||
|
curl_easy_cleanup(curl);
|
||||||
|
|
||||||
|
// Remove partially downloaded file
|
||||||
|
std::remove(cachePath.c_str());
|
||||||
|
|
||||||
|
std::string error = std::string("[DataDownloader] Download failed: ") +
|
||||||
|
curl_easy_strerror(res);
|
||||||
|
throw std::runtime_error(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check HTTP response code
|
||||||
|
long httpCode = 0;
|
||||||
|
curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &httpCode);
|
||||||
|
curl_easy_cleanup(curl);
|
||||||
|
|
||||||
|
if (httpCode != 200) {
|
||||||
|
// Remove partially downloaded or error file
|
||||||
|
std::remove(cachePath.c_str());
|
||||||
|
|
||||||
|
std::stringstream ss;
|
||||||
|
ss << "[DataDownloader] HTTP error " << httpCode
|
||||||
|
<< " (commit: " << shortCommit << ")";
|
||||||
|
throw std::runtime_error(ss.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get file size for diagnostics
|
||||||
|
std::ifstream fileCheck(cachePath, std::ios::binary | std::ios::ate);
|
||||||
|
std::streamsize size = fileCheck.tellg();
|
||||||
|
fileCheck.close();
|
||||||
|
|
||||||
|
std::cout << "[DataDownloader] ✓ Download complete: " << cachePath << " ("
|
||||||
|
<< (size / (1024.0 * 1024.0)) << " MB)" << std::endl;
|
||||||
|
return cachePath;
|
||||||
|
}
|
||||||
229
pipeline/src/database.cpp
Normal file
229
pipeline/src/database.cpp
Normal file
@@ -0,0 +1,229 @@
|
|||||||
|
#include "database.h"
|
||||||
|
#include <iostream>
|
||||||
|
#include <stdexcept>
|
||||||
|
|
||||||
|
void SqliteDatabase::InitializeSchema() {
|
||||||
|
std::lock_guard<std::mutex> lock(dbMutex);
|
||||||
|
|
||||||
|
const char *schema = R"(
|
||||||
|
CREATE TABLE IF NOT EXISTS countries (
|
||||||
|
id INTEGER PRIMARY KEY,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
iso2 TEXT,
|
||||||
|
iso3 TEXT
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS states (
|
||||||
|
id INTEGER PRIMARY KEY,
|
||||||
|
country_id INTEGER NOT NULL,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
iso2 TEXT,
|
||||||
|
FOREIGN KEY(country_id) REFERENCES countries(id)
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS cities (
|
||||||
|
id INTEGER PRIMARY KEY,
|
||||||
|
state_id INTEGER NOT NULL,
|
||||||
|
country_id INTEGER NOT NULL,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
latitude REAL,
|
||||||
|
longitude REAL,
|
||||||
|
FOREIGN KEY(state_id) REFERENCES states(id),
|
||||||
|
FOREIGN KEY(country_id) REFERENCES countries(id)
|
||||||
|
);
|
||||||
|
)";
|
||||||
|
|
||||||
|
char *errMsg = nullptr;
|
||||||
|
int rc = sqlite3_exec(db, schema, nullptr, nullptr, &errMsg);
|
||||||
|
if (rc != SQLITE_OK) {
|
||||||
|
std::string error = errMsg ? std::string(errMsg) : "Unknown error";
|
||||||
|
sqlite3_free(errMsg);
|
||||||
|
throw std::runtime_error("Failed to create schema: " + error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SqliteDatabase::~SqliteDatabase() {
|
||||||
|
if (db) {
|
||||||
|
sqlite3_close(db);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SqliteDatabase::Initialize() {
|
||||||
|
int rc = sqlite3_open(":memory:", &db);
|
||||||
|
if (rc) {
|
||||||
|
throw std::runtime_error("Failed to create in-memory SQLite database");
|
||||||
|
}
|
||||||
|
std::cout << "✓ In-memory SQLite database created\n";
|
||||||
|
InitializeSchema();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SqliteDatabase::InsertCountry(int id, const std::string &name,
|
||||||
|
const std::string &iso2,
|
||||||
|
const std::string &iso3) {
|
||||||
|
std::lock_guard<std::mutex> lock(dbMutex);
|
||||||
|
|
||||||
|
const char *query = R"(
|
||||||
|
INSERT OR IGNORE INTO countries (id, name, iso2, iso3)
|
||||||
|
VALUES (?, ?, ?, ?)
|
||||||
|
)";
|
||||||
|
|
||||||
|
sqlite3_stmt *stmt;
|
||||||
|
int rc = sqlite3_prepare_v2(db, query, -1, &stmt, nullptr);
|
||||||
|
if (rc != SQLITE_OK)
|
||||||
|
throw std::runtime_error("Failed to prepare country insert");
|
||||||
|
|
||||||
|
sqlite3_bind_int(stmt, 1, id);
|
||||||
|
sqlite3_bind_text(stmt, 2, name.c_str(), -1, SQLITE_STATIC);
|
||||||
|
sqlite3_bind_text(stmt, 3, iso2.c_str(), -1, SQLITE_STATIC);
|
||||||
|
sqlite3_bind_text(stmt, 4, iso3.c_str(), -1, SQLITE_STATIC);
|
||||||
|
|
||||||
|
if (sqlite3_step(stmt) != SQLITE_DONE) {
|
||||||
|
throw std::runtime_error("Failed to insert country");
|
||||||
|
}
|
||||||
|
sqlite3_finalize(stmt);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SqliteDatabase::InsertState(int id, int countryId, const std::string &name,
|
||||||
|
const std::string &iso2) {
|
||||||
|
std::lock_guard<std::mutex> lock(dbMutex);
|
||||||
|
|
||||||
|
const char *query = R"(
|
||||||
|
INSERT OR IGNORE INTO states (id, country_id, name, iso2)
|
||||||
|
VALUES (?, ?, ?, ?)
|
||||||
|
)";
|
||||||
|
|
||||||
|
sqlite3_stmt *stmt;
|
||||||
|
int rc = sqlite3_prepare_v2(db, query, -1, &stmt, nullptr);
|
||||||
|
if (rc != SQLITE_OK)
|
||||||
|
throw std::runtime_error("Failed to prepare state insert");
|
||||||
|
|
||||||
|
sqlite3_bind_int(stmt, 1, id);
|
||||||
|
sqlite3_bind_int(stmt, 2, countryId);
|
||||||
|
sqlite3_bind_text(stmt, 3, name.c_str(), -1, SQLITE_STATIC);
|
||||||
|
sqlite3_bind_text(stmt, 4, iso2.c_str(), -1, SQLITE_STATIC);
|
||||||
|
|
||||||
|
if (sqlite3_step(stmt) != SQLITE_DONE) {
|
||||||
|
throw std::runtime_error("Failed to insert state");
|
||||||
|
}
|
||||||
|
sqlite3_finalize(stmt);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SqliteDatabase::InsertCity(int id, int stateId, int countryId,
|
||||||
|
const std::string &name, double latitude,
|
||||||
|
double longitude) {
|
||||||
|
std::lock_guard<std::mutex> lock(dbMutex);
|
||||||
|
|
||||||
|
const char *query = R"(
|
||||||
|
INSERT OR IGNORE INTO cities (id, state_id, country_id, name, latitude, longitude)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?)
|
||||||
|
)";
|
||||||
|
|
||||||
|
sqlite3_stmt *stmt;
|
||||||
|
int rc = sqlite3_prepare_v2(db, query, -1, &stmt, nullptr);
|
||||||
|
if (rc != SQLITE_OK)
|
||||||
|
throw std::runtime_error("Failed to prepare city insert");
|
||||||
|
|
||||||
|
sqlite3_bind_int(stmt, 1, id);
|
||||||
|
sqlite3_bind_int(stmt, 2, stateId);
|
||||||
|
sqlite3_bind_int(stmt, 3, countryId);
|
||||||
|
sqlite3_bind_text(stmt, 4, name.c_str(), -1, SQLITE_STATIC);
|
||||||
|
sqlite3_bind_double(stmt, 5, latitude);
|
||||||
|
sqlite3_bind_double(stmt, 6, longitude);
|
||||||
|
|
||||||
|
if (sqlite3_step(stmt) != SQLITE_DONE) {
|
||||||
|
throw std::runtime_error("Failed to insert city");
|
||||||
|
}
|
||||||
|
sqlite3_finalize(stmt);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::pair<int, std::string>> SqliteDatabase::QueryCities() {
|
||||||
|
std::lock_guard<std::mutex> lock(dbMutex);
|
||||||
|
|
||||||
|
std::vector<std::pair<int, std::string>> cities;
|
||||||
|
sqlite3_stmt *stmt = nullptr;
|
||||||
|
|
||||||
|
const char *query = "SELECT id, name FROM cities ORDER BY name";
|
||||||
|
int rc = sqlite3_prepare_v2(db, query, -1, &stmt, nullptr);
|
||||||
|
|
||||||
|
if (rc != SQLITE_OK) {
|
||||||
|
throw std::runtime_error("Failed to prepare query");
|
||||||
|
}
|
||||||
|
|
||||||
|
while (sqlite3_step(stmt) == SQLITE_ROW) {
|
||||||
|
int id = sqlite3_column_int(stmt, 0);
|
||||||
|
const char *name =
|
||||||
|
reinterpret_cast<const char *>(sqlite3_column_text(stmt, 1));
|
||||||
|
cities.push_back({id, name ? std::string(name) : ""});
|
||||||
|
}
|
||||||
|
|
||||||
|
sqlite3_finalize(stmt);
|
||||||
|
return cities;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<Country> SqliteDatabase::QueryCountries(int limit) {
|
||||||
|
std::lock_guard<std::mutex> lock(dbMutex);
|
||||||
|
|
||||||
|
std::vector<Country> countries;
|
||||||
|
sqlite3_stmt *stmt = nullptr;
|
||||||
|
|
||||||
|
std::string query =
|
||||||
|
"SELECT id, name, iso2, iso3 FROM countries ORDER BY name";
|
||||||
|
if (limit > 0) {
|
||||||
|
query += " LIMIT " + std::to_string(limit);
|
||||||
|
}
|
||||||
|
|
||||||
|
int rc = sqlite3_prepare_v2(db, query.c_str(), -1, &stmt, nullptr);
|
||||||
|
|
||||||
|
if (rc != SQLITE_OK) {
|
||||||
|
throw std::runtime_error("Failed to prepare countries query");
|
||||||
|
}
|
||||||
|
|
||||||
|
while (sqlite3_step(stmt) == SQLITE_ROW) {
|
||||||
|
int id = sqlite3_column_int(stmt, 0);
|
||||||
|
const char *name =
|
||||||
|
reinterpret_cast<const char *>(sqlite3_column_text(stmt, 1));
|
||||||
|
const char *iso2 =
|
||||||
|
reinterpret_cast<const char *>(sqlite3_column_text(stmt, 2));
|
||||||
|
const char *iso3 =
|
||||||
|
reinterpret_cast<const char *>(sqlite3_column_text(stmt, 3));
|
||||||
|
countries.push_back({id, name ? std::string(name) : "",
|
||||||
|
iso2 ? std::string(iso2) : "",
|
||||||
|
iso3 ? std::string(iso3) : ""});
|
||||||
|
}
|
||||||
|
|
||||||
|
sqlite3_finalize(stmt);
|
||||||
|
return countries;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<State> SqliteDatabase::QueryStates(int limit) {
|
||||||
|
std::lock_guard<std::mutex> lock(dbMutex);
|
||||||
|
|
||||||
|
std::vector<State> states;
|
||||||
|
sqlite3_stmt *stmt = nullptr;
|
||||||
|
|
||||||
|
std::string query =
|
||||||
|
"SELECT id, name, iso2, country_id FROM states ORDER BY name";
|
||||||
|
if (limit > 0) {
|
||||||
|
query += " LIMIT " + std::to_string(limit);
|
||||||
|
}
|
||||||
|
|
||||||
|
int rc = sqlite3_prepare_v2(db, query.c_str(), -1, &stmt, nullptr);
|
||||||
|
|
||||||
|
if (rc != SQLITE_OK) {
|
||||||
|
throw std::runtime_error("Failed to prepare states query");
|
||||||
|
}
|
||||||
|
|
||||||
|
while (sqlite3_step(stmt) == SQLITE_ROW) {
|
||||||
|
int id = sqlite3_column_int(stmt, 0);
|
||||||
|
const char *name =
|
||||||
|
reinterpret_cast<const char *>(sqlite3_column_text(stmt, 1));
|
||||||
|
const char *iso2 =
|
||||||
|
reinterpret_cast<const char *>(sqlite3_column_text(stmt, 2));
|
||||||
|
int countryId = sqlite3_column_int(stmt, 3);
|
||||||
|
states.push_back({id, name ? std::string(name) : "",
|
||||||
|
iso2 ? std::string(iso2) : "", countryId});
|
||||||
|
}
|
||||||
|
|
||||||
|
sqlite3_finalize(stmt);
|
||||||
|
return states;
|
||||||
|
}
|
||||||
81
pipeline/src/generator.cpp
Normal file
81
pipeline/src/generator.cpp
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
#include "generator.h"
|
||||||
|
#include <functional>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Initializes the brewery generator by loading a language model
|
||||||
|
*
|
||||||
|
* Current Implementation (Mock):
|
||||||
|
* - Outputs informational messages about model initialization
|
||||||
|
* - Does not load actual llama.cpp model yet
|
||||||
|
* - Serves as interface definition for future real implementation
|
||||||
|
*
|
||||||
|
* Future Implementation:
|
||||||
|
* - Will load a GGUF-format LLM model file using llama.cpp
|
||||||
|
* - Will initialize CPU/GPU inference context
|
||||||
|
* - Will cache model weights for repeated brewery generation
|
||||||
|
*
|
||||||
|
* @param modelPath Path to GGUF model file (e.g., "models/llama-7b.gguf")
|
||||||
|
*
|
||||||
|
* Example output:
|
||||||
|
* @code
|
||||||
|
* [Mock] Initialized llama model: models/llama-7b.gguf
|
||||||
|
* ✓ Model ready
|
||||||
|
* @endcode
|
||||||
|
*/
|
||||||
|
void LlamaBreweryGenerator::LoadModel(const std::string &modelPath) {
|
||||||
|
std::cout << " [Mock] Initialized llama model: " << modelPath << "\n";
|
||||||
|
std::cout << " ✓ Model ready\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Generates a brewery name and description for a city using
|
||||||
|
* deterministic hashing
|
||||||
|
*
|
||||||
|
* Algorithm:
|
||||||
|
* 1. Combines city name with seed to create unique hash input
|
||||||
|
* 2. Uses std::hash<std::string> to compute deterministic hash value
|
||||||
|
* 3. Uses modulo arithmetic to map hash to template arrays:
|
||||||
|
* - name: adjective[hash % 8] + noun[(hash/7) % 8]
|
||||||
|
* - description: descriptions[(hash/13) % 5]
|
||||||
|
* 4. Returns Brewery struct with generated name and description
|
||||||
|
*
|
||||||
|
* Determinism:
|
||||||
|
* - Same cityName + seed ALWAYS produces same result
|
||||||
|
* - Enables reproducible testing and consistent brewery assignments
|
||||||
|
* - Hash distribution spreads city names across template combinations
|
||||||
|
*
|
||||||
|
* Example:
|
||||||
|
* @code
|
||||||
|
* auto gen = LlamaBreweryGenerator();
|
||||||
|
* auto brewery = gen.GenerateBrewery("Toronto", 1);
|
||||||
|
* // Always produces same brewery for same city/seed
|
||||||
|
* assert(gen.GenerateBrewery("Toronto", 1).name == brewery.name);
|
||||||
|
* @endcode
|
||||||
|
*
|
||||||
|
* @param cityName The city to generate a brewery for
|
||||||
|
* @param seed An integer seed for deterministic variation (usually 0 or row ID)
|
||||||
|
* @return Brewery struct containing:
|
||||||
|
* - name: Combined adjective + noun (e.g., "Craft Brewing Co.")
|
||||||
|
* - description: Pre-written description matching brewery style
|
||||||
|
*
|
||||||
|
* @note Future: Replace hashing with actual LLM inference
|
||||||
|
* Interface will remain identical for smooth migration
|
||||||
|
*/
|
||||||
|
LlamaBreweryGenerator::Brewery
|
||||||
|
LlamaBreweryGenerator::GenerateBrewery(const std::string &cityName, int seed) {
|
||||||
|
// Deterministic mock generation based on city name and seed
|
||||||
|
// Combines city name with seed to ensure different results for same city
|
||||||
|
// with different seed values (useful for generating multiple breweries per
|
||||||
|
// city)
|
||||||
|
size_t nameHash = std::hash<std::string>{}(cityName + std::to_string(seed));
|
||||||
|
|
||||||
|
Brewery result;
|
||||||
|
// Select adjective and noun using hash modulo
|
||||||
|
// Divided by 7 and 13 to ensure different modulo results from same hash
|
||||||
|
result.name = breweryAdjectives[nameHash % breweryAdjectives.size()] + " " +
|
||||||
|
breweryNouns[(nameHash / 7) % breweryNouns.size()];
|
||||||
|
result.description = descriptions[(nameHash / 13) % descriptions.size()];
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
222
pipeline/src/json_loader.cpp
Normal file
222
pipeline/src/json_loader.cpp
Normal file
@@ -0,0 +1,222 @@
|
|||||||
|
#include "json_loader.h"
|
||||||
|
#include <fstream>
|
||||||
|
#include <iostream>
|
||||||
|
#include <mutex>
|
||||||
|
#include <thread>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Loads world geographic data from JSON file into SQLite database
|
||||||
|
*
|
||||||
|
* This function implements a hierarchical multithreaded loading strategy:
|
||||||
|
*
|
||||||
|
* THREADING ARCHITECTURE:
|
||||||
|
* ┌─────────────────────────────────────────────────────────────────┐
|
||||||
|
* │ Main Thread: Parse JSON (45 MB) │
|
||||||
|
* └────────────────────┬────────────────────────────────────────────┘
|
||||||
|
* │
|
||||||
|
* ┌─────────────┴──────────────┬──────────────┐
|
||||||
|
* ▼ ▼ ▼
|
||||||
|
* Country Thread 0 Country Thread 1 ... Thread N
|
||||||
|
* ├─ Insert Country ├─ Insert Country └─ Insert Country
|
||||||
|
* │
|
||||||
|
* ├─ State Thread A ├─ State Thread C
|
||||||
|
* │ ├─ Insert State │ ├─ Insert State
|
||||||
|
* │ ├─ Insert 100 cities │ └─ Insert 150 cities
|
||||||
|
* │ └─ +stats └─ +stats
|
||||||
|
* │
|
||||||
|
* └─ State Thread B
|
||||||
|
* ├─ Insert State
|
||||||
|
* ├─ Insert 200 cities
|
||||||
|
* └─ +stats
|
||||||
|
*
|
||||||
|
* THREADING DETAILS:
|
||||||
|
* - Countries loop: divided among CPU_CORE_COUNT threads
|
||||||
|
* - Each country: states processed in dedicated threads (nested parallelism)
|
||||||
|
* - Each state: cities inserted sequentially (within thread)
|
||||||
|
* - All writes protected by mutex in SqliteDatabase
|
||||||
|
* - Processing stats (city count) synchronized with mutex
|
||||||
|
*
|
||||||
|
* INPUT JSON STRUCTURE:
|
||||||
|
* The JSON file contains three main arrays:
|
||||||
|
*
|
||||||
|
* 1. Countries (~250 records):
|
||||||
|
* { id: int, name: string, iso2: string, iso3: string }
|
||||||
|
*
|
||||||
|
* 2. States/Provinces (~3500 records):
|
||||||
|
* { id: int, country_id: int, name: string, iso2: string }
|
||||||
|
*
|
||||||
|
* 3. Cities (~50000 records):
|
||||||
|
* { id: int, state_id: int, country_id: int, name: string,
|
||||||
|
* latitude: double, longitude: double }
|
||||||
|
*
|
||||||
|
* PERFORMANCE:
|
||||||
|
* - JSON parsing: Single-threaded, happens once at start
|
||||||
|
* - Country insertion: Parallelized across CPU cores
|
||||||
|
* - State insertion: Parallelized within each country via nested threads
|
||||||
|
* - City insertion: Sequential within each state (reduces serialization)
|
||||||
|
* - Total expected runtime: 2-5 seconds for 50k cities on modern CPU
|
||||||
|
*
|
||||||
|
* ERROR HANDLING:
|
||||||
|
* - Missing JSON file: throws std::runtime_error
|
||||||
|
* - Invalid JSON: throws nlohmann::json::parse_error
|
||||||
|
* - Bad city records: silently skipped (try-catch within loop)
|
||||||
|
* - Database errors: re-thrown from db.Insert*() calls
|
||||||
|
*
|
||||||
|
* STATISTICS:
|
||||||
|
* Prints progress messages showing:
|
||||||
|
* - Number of countries loaded
|
||||||
|
* - Number of worker threads created
|
||||||
|
* - Total cities inserted into database
|
||||||
|
*
|
||||||
|
* @param jsonPath Path to JSON file (typically: ../data/world_city_data.json)
|
||||||
|
* @param db Reference to initialized SqliteDatabase to populate
|
||||||
|
*/
|
||||||
|
void JsonLoader::LoadWorldCities(const std::string &jsonPath,
|
||||||
|
SqliteDatabase &db) {
|
||||||
|
std::cout << "\nLoading " << jsonPath << " (45 MB)...\n";
|
||||||
|
|
||||||
|
// Open and read JSON file from disk
|
||||||
|
std::ifstream jsonFile(jsonPath);
|
||||||
|
if (!jsonFile.is_open()) {
|
||||||
|
throw std::runtime_error("Failed to open JSON file: " + jsonPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse entire JSON into memory (nlohmann/json library)
|
||||||
|
json data;
|
||||||
|
try {
|
||||||
|
jsonFile >> data;
|
||||||
|
} catch (const std::exception &e) {
|
||||||
|
throw std::runtime_error("JSON parse error: " + std::string(e.what()));
|
||||||
|
}
|
||||||
|
jsonFile.close();
|
||||||
|
|
||||||
|
// DEBUG: Check JSON structure
|
||||||
|
if (!data.is_array()) {
|
||||||
|
std::cerr << "[DEBUG] JSON root is not an array. Type: " << data.type_name()
|
||||||
|
<< std::endl;
|
||||||
|
if (data.is_object()) {
|
||||||
|
std::cerr << "[DEBUG] JSON root is object with keys: ";
|
||||||
|
for (auto &[key, val] : data.items()) {
|
||||||
|
std::cerr << key << " ";
|
||||||
|
}
|
||||||
|
std::cerr << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "✓ Loaded " << data.size()
|
||||||
|
<< " records (expecting countries array)\n";
|
||||||
|
|
||||||
|
if (data.size() == 0) {
|
||||||
|
throw std::runtime_error("JSON file appears to be empty or malformed. "
|
||||||
|
"Check download succeeded.");
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "Processing countries with multithreading...\n";
|
||||||
|
|
||||||
|
// Determine optimal thread count based on CPU cores
|
||||||
|
unsigned int numThreads = std::thread::hardware_concurrency();
|
||||||
|
std::cout << " Using " << numThreads << " threads\n\n";
|
||||||
|
|
||||||
|
// Shared counter for statistics (protected by mutex)
|
||||||
|
int processedCities = 0;
|
||||||
|
std::mutex statsMutex;
|
||||||
|
|
||||||
|
// Spawn threads to process countries in parallel
|
||||||
|
std::vector<std::thread> countryThreads;
|
||||||
|
const size_t countriesPerThread = (data.size() + numThreads - 1) / numThreads;
|
||||||
|
|
||||||
|
for (size_t t = 0; t < numThreads; ++t) {
|
||||||
|
countryThreads.push_back(std::thread([&, t]() {
|
||||||
|
// Each thread processes a range of countries
|
||||||
|
size_t start = t * countriesPerThread;
|
||||||
|
size_t end = std::min((t + 1) * countriesPerThread, data.size());
|
||||||
|
|
||||||
|
for (size_t i = start; i < end; ++i) {
|
||||||
|
const auto &country = data[i];
|
||||||
|
int countryId = country["id"];
|
||||||
|
std::string countryName = country["name"];
|
||||||
|
std::string iso2 = country.value("iso2", "");
|
||||||
|
std::string iso3 = country.value("iso3", "");
|
||||||
|
|
||||||
|
// Insert country record
|
||||||
|
db.InsertCountry(countryId, countryName, iso2, iso3);
|
||||||
|
|
||||||
|
// Process states within this country
|
||||||
|
if (country.contains("states") && country["states"].is_array()) {
|
||||||
|
const auto &states = country["states"];
|
||||||
|
|
||||||
|
// Spawn threads to process states in parallel
|
||||||
|
// This creates nested parallelism: country threads spawn state
|
||||||
|
// threads
|
||||||
|
std::vector<std::thread> stateThreads;
|
||||||
|
|
||||||
|
for (size_t s = 0; s < states.size(); ++s) {
|
||||||
|
stateThreads.push_back(std::thread([&, s, countryId]() {
|
||||||
|
const auto &state = states[s];
|
||||||
|
int stateId = state["id"];
|
||||||
|
std::string stateName = state["name"];
|
||||||
|
std::string stateIso2 = state.value("iso2", "");
|
||||||
|
|
||||||
|
// Insert state record
|
||||||
|
db.InsertState(stateId, countryId, stateName, stateIso2);
|
||||||
|
|
||||||
|
// Process cities for this state
|
||||||
|
if (state.contains("cities") && state["cities"].is_array()) {
|
||||||
|
// Cities within a state are processed sequentially
|
||||||
|
// (within the state thread - reduces context switching)
|
||||||
|
for (const auto &city : state["cities"]) {
|
||||||
|
try {
|
||||||
|
int cityId = city["id"].get<int>();
|
||||||
|
std::string cityName = city["name"];
|
||||||
|
|
||||||
|
// Parse latitude and longitude as strings first (they're
|
||||||
|
// stored as strings in JSON), then convert to double
|
||||||
|
double lat = 0.0;
|
||||||
|
double lng = 0.0;
|
||||||
|
if (city.contains("latitude")) {
|
||||||
|
lat = std::stod(city["latitude"].get<std::string>());
|
||||||
|
}
|
||||||
|
if (city.contains("longitude")) {
|
||||||
|
lng = std::stod(city["longitude"].get<std::string>());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Insert city record to database
|
||||||
|
// Database has mutex protection for thread-safe access
|
||||||
|
db.InsertCity(cityId, stateId, countryId, cityName, lat,
|
||||||
|
lng);
|
||||||
|
|
||||||
|
// Update shared statistics counter (protected by mutex)
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(statsMutex);
|
||||||
|
processedCities++;
|
||||||
|
}
|
||||||
|
} catch (const std::exception &e) {
|
||||||
|
// Silently skip malformed city entries
|
||||||
|
// Example: missing required fields, invalid coordinates
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait for all state threads to complete
|
||||||
|
// Important: don't proceed to next country until states are done
|
||||||
|
for (auto &t : stateThreads) {
|
||||||
|
if (t.joinable())
|
||||||
|
t.join();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait for all country threads to complete
|
||||||
|
// This blocks until all nested state/city insertions are done
|
||||||
|
for (auto &t : countryThreads) {
|
||||||
|
if (t.joinable())
|
||||||
|
t.join();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "✓ Loaded " << processedCities << " cities into database\n\n";
|
||||||
|
}
|
||||||
154
pipeline/src/main.cpp
Normal file
154
pipeline/src/main.cpp
Normal file
@@ -0,0 +1,154 @@
|
|||||||
|
/**
|
||||||
|
* @file main.cpp
|
||||||
|
* @brief Entry point for the brewery data pipeline
|
||||||
|
*
|
||||||
|
* Pipeline Overview:
|
||||||
|
* This is the main data processing pipeline that:
|
||||||
|
* 1. Initializes an in-memory SQLite database
|
||||||
|
* 2. Loads world city data from a JSON file (50k+ cities)
|
||||||
|
* 3. Initializes the brewery generation system (currently mocked)
|
||||||
|
* 4. Demonstrates brewery generation for sample cities
|
||||||
|
*
|
||||||
|
* Architecture:
|
||||||
|
* ┌─────────────┐
|
||||||
|
* │ JSON File │ (world_city_data.json - 50k+ cities)
|
||||||
|
* └──────┬──────┘
|
||||||
|
* │
|
||||||
|
* ▼
|
||||||
|
* ┌─────────────────────┐
|
||||||
|
* │ JsonLoader::Load │ Parse and validate JSON
|
||||||
|
* └──────┬──────────────┘
|
||||||
|
* │
|
||||||
|
* ▼
|
||||||
|
* ┌─────────────────────┐
|
||||||
|
* │ SQLite Database │ Store cities in-memory
|
||||||
|
* └──────┬──────────────┘
|
||||||
|
* │
|
||||||
|
* ▼
|
||||||
|
* ┌─────────────────────┐
|
||||||
|
* │ BreweryGenerator │ Mock generation (hash-based)
|
||||||
|
* │ .GenerateBrewery() │ Future: LLM-based generation
|
||||||
|
* └─────────────────────┘
|
||||||
|
*
|
||||||
|
* Command Line Arguments:
|
||||||
|
* - argv[1]: Path to GGUF model file (default: ./model.gguf)
|
||||||
|
* - argv[2]: Path to cache directory for JSON downloads (default: /tmp)
|
||||||
|
* - argv[3]: Git commit hash for reproducible data version (default: c5eb7772)
|
||||||
|
*
|
||||||
|
* The pipeline automatically downloads the geographic data from GitHub on first
|
||||||
|
* run and caches it locally to avoid repeated network calls.
|
||||||
|
*
|
||||||
|
* Example Usage - Auto-download (stable 2026-03-28 build):
|
||||||
|
* @code
|
||||||
|
* ./brewery-pipeline ./llama-7b.gguf
|
||||||
|
* @endcode
|
||||||
|
*
|
||||||
|
* Example Usage - Custom commit:
|
||||||
|
* @code
|
||||||
|
* ./brewery-pipeline ./llama-7b.gguf /tmp main
|
||||||
|
* @endcode
|
||||||
|
*
|
||||||
|
* Exit Codes:
|
||||||
|
* - 0: Pipeline completed successfully
|
||||||
|
* - 1: Pipeline failed (exception caught)
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "data_downloader.h"
|
||||||
|
#include "database.h"
|
||||||
|
#include "generator.h"
|
||||||
|
#include "json_loader.h"
|
||||||
|
#include <curl/curl.h>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
int main(int argc, char *argv[]) {
|
||||||
|
try {
|
||||||
|
// Initialize libcurl globally (thread-safe mode)
|
||||||
|
curl_global_init(CURL_GLOBAL_DEFAULT);
|
||||||
|
|
||||||
|
// Parse command-line arguments
|
||||||
|
std::string modelPath = argc > 1 ? argv[1] : "./model.gguf";
|
||||||
|
std::string cacheDir = argc > 2 ? argv[2] : "/tmp";
|
||||||
|
std::string commit =
|
||||||
|
argc > 3 ? argv[3] : "c5eb7772"; // Default: stable 2026-03-28
|
||||||
|
|
||||||
|
// Construct cache path for downloaded JSON
|
||||||
|
std::string jsonPath = cacheDir + "/countries+states+cities.json";
|
||||||
|
|
||||||
|
// Step 0: Download geographic data from GitHub (cached locally)
|
||||||
|
// On first run, downloads 45MB JSON. On subsequent runs, uses cached file.
|
||||||
|
// Commit hash allows pinning to specific data versions for reproducibility.
|
||||||
|
std::cout << "\n[Pipeline] Downloading geographic data from GitHub...\n";
|
||||||
|
DataDownloader downloader;
|
||||||
|
downloader.DownloadCountriesDatabase(jsonPath, commit);
|
||||||
|
|
||||||
|
SqliteDatabase db;
|
||||||
|
|
||||||
|
// Step 1: Initialize empty in-memory database
|
||||||
|
std::cout << "Initializing in-memory SQLite database...\n";
|
||||||
|
db.Initialize();
|
||||||
|
|
||||||
|
// Step 2: Load world city data from JSON file
|
||||||
|
// This populates the database with ~50k city records
|
||||||
|
// Each record includes: city name, country, latitude, longitude, population
|
||||||
|
JsonLoader::LoadWorldCities(jsonPath, db);
|
||||||
|
|
||||||
|
// Step 3: Initialize brewery generator
|
||||||
|
// Current: Mock implementation using deterministic hashing
|
||||||
|
// Future: LLM-based generation with llama.cpp
|
||||||
|
std::cout << "Initializing brewery generator...\n";
|
||||||
|
LlamaBreweryGenerator generator;
|
||||||
|
generator.LoadModel(modelPath);
|
||||||
|
|
||||||
|
// Step 4: Query geographic data from database
|
||||||
|
std::cout << "\n=== GEOGRAPHIC DATA OVERVIEW ===\n";
|
||||||
|
|
||||||
|
auto countries = db.QueryCountries(50);
|
||||||
|
auto states = db.QueryStates(50);
|
||||||
|
auto cities = db.QueryCities();
|
||||||
|
|
||||||
|
std::cout << "\nTotal records loaded:";
|
||||||
|
std::cout << "\n Countries: " << db.QueryCountries(0).size();
|
||||||
|
std::cout << "\n States: " << db.QueryStates(0).size();
|
||||||
|
std::cout << "\n Cities: " << cities.size() << "\n";
|
||||||
|
|
||||||
|
// Display 50 countries
|
||||||
|
std::cout << "\n--- 50 COUNTRIES ---\n";
|
||||||
|
for (size_t i = 0; i < countries.size(); i++) {
|
||||||
|
std::cout << (i + 1) << ". " << countries[i].iso2 << " ("
|
||||||
|
<< countries[i].iso3 << ") " << countries[i].name << "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Display 50 states
|
||||||
|
std::cout << "\n--- 50 STATES ---\n";
|
||||||
|
for (size_t i = 0; i < states.size(); i++) {
|
||||||
|
std::cout << (i + 1) << ". " << states[i].iso2 << ": " << states[i].name
|
||||||
|
<< "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Display 50 cities
|
||||||
|
std::cout << "\n--- 50 CITIES ---\n";
|
||||||
|
for (size_t i = 0; i < std::min(size_t(50), cities.size()); i++) {
|
||||||
|
std::cout << (i + 1) << ". " << cities[i].second << "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step 5: Demonstrate brewery generation on sample cities
|
||||||
|
std::cout << "\n=== SAMPLE BREWERY GENERATION ===\n\n";
|
||||||
|
for (size_t i = 0; i < std::min(size_t(5), cities.size()); i++) {
|
||||||
|
const auto &[cityId, cityName] = cities[i];
|
||||||
|
auto brewery = generator.GenerateBrewery(cityName, i);
|
||||||
|
std::cout << " " << cityName << ": " << brewery.name << "\n";
|
||||||
|
std::cout << " → " << brewery.description << "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "\n✓ Pipeline completed successfully\n";
|
||||||
|
|
||||||
|
// Cleanup
|
||||||
|
curl_global_cleanup();
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
} catch (const std::exception &e) {
|
||||||
|
std::cerr << "✗ Pipeline failed: " << e.what() << "\n";
|
||||||
|
curl_global_cleanup();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -31,7 +31,6 @@
|
|||||||
<ProjectReference Include="..\..\Infrastructure\Infrastructure.Repository\Infrastructure.Repository.csproj" />
|
<ProjectReference Include="..\..\Infrastructure\Infrastructure.Repository\Infrastructure.Repository.csproj" />
|
||||||
<ProjectReference Include="..\..\Infrastructure\Infrastructure.Jwt\Infrastructure.Jwt.csproj" />
|
<ProjectReference Include="..\..\Infrastructure\Infrastructure.Jwt\Infrastructure.Jwt.csproj" />
|
||||||
<ProjectReference Include="..\..\Service\Service.Auth\Service.Auth.csproj" />
|
<ProjectReference Include="..\..\Service\Service.Auth\Service.Auth.csproj" />
|
||||||
<ProjectReference Include="..\..\Service\Service.Breweries\Service.Breweries.csproj" />
|
|
||||||
<ProjectReference Include="..\..\Service\Service.UserManagement\Service.UserManagement.csproj" />
|
<ProjectReference Include="..\..\Service\Service.UserManagement\Service.UserManagement.csproj" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
@@ -86,13 +86,6 @@ namespace API.Core.Controllers
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpPost("confirm/resend")]
|
|
||||||
public async Task<ActionResult> ResendConfirmation([FromQuery] Guid userId)
|
|
||||||
{
|
|
||||||
await confirmationService.ResendConfirmationEmailAsync(userId);
|
|
||||||
return Ok(new ResponseBody { Message = "confirmation email has been resent" });
|
|
||||||
}
|
|
||||||
|
|
||||||
[AllowAnonymous]
|
[AllowAnonymous]
|
||||||
[HttpPost("refresh")]
|
[HttpPost("refresh")]
|
||||||
public async Task<ActionResult> Refresh(
|
public async Task<ActionResult> Refresh(
|
||||||
@@ -1,15 +1,20 @@
|
|||||||
using API.Core;
|
using API.Core;
|
||||||
using API.Core.Authentication;
|
using API.Core.Authentication;
|
||||||
|
using API.Core.Contracts.Common;
|
||||||
|
using Domain.Exceptions;
|
||||||
using FluentValidation;
|
using FluentValidation;
|
||||||
using FluentValidation.AspNetCore;
|
using FluentValidation.AspNetCore;
|
||||||
using Infrastructure.Email;
|
using Infrastructure.Email;
|
||||||
|
using Infrastructure.Email.Templates;
|
||||||
using Infrastructure.Email.Templates.Rendering;
|
using Infrastructure.Email.Templates.Rendering;
|
||||||
using Infrastructure.Jwt;
|
using Infrastructure.Jwt;
|
||||||
using Infrastructure.PasswordHashing;
|
using Infrastructure.PasswordHashing;
|
||||||
using Infrastructure.Repository.Auth;
|
using Infrastructure.Repository.Auth;
|
||||||
using Infrastructure.Repository.Sql;
|
using Infrastructure.Repository.Sql;
|
||||||
using Infrastructure.Repository.UserAccount;
|
using Infrastructure.Repository.UserAccount;
|
||||||
using Infrastructure.Repository.Breweries;
|
using Microsoft.AspNetCore.Authentication;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using Microsoft.AspNetCore.Mvc.Filters;
|
||||||
using Service.Auth;
|
using Service.Auth;
|
||||||
using Service.Emails;
|
using Service.Emails;
|
||||||
using Service.UserManagement.User;
|
using Service.UserManagement.User;
|
||||||
@@ -50,7 +55,6 @@ builder.Services.AddSingleton<
|
|||||||
|
|
||||||
builder.Services.AddScoped<IUserAccountRepository, UserAccountRepository>();
|
builder.Services.AddScoped<IUserAccountRepository, UserAccountRepository>();
|
||||||
builder.Services.AddScoped<IAuthRepository, AuthRepository>();
|
builder.Services.AddScoped<IAuthRepository, AuthRepository>();
|
||||||
builder.Services.AddScoped<IBreweryRepository, BreweryRepository>();
|
|
||||||
|
|
||||||
builder.Services.AddScoped<IUserService, UserService>();
|
builder.Services.AddScoped<IUserService, UserService>();
|
||||||
builder.Services.AddScoped<ILoginService, LoginService>();
|
builder.Services.AddScoped<ILoginService, LoginService>();
|
||||||
@@ -26,7 +26,6 @@
|
|||||||
<Project Path="Service/Service.Auth.Tests/Service.Auth.Tests.csproj" />
|
<Project Path="Service/Service.Auth.Tests/Service.Auth.Tests.csproj" />
|
||||||
<Project Path="Service/Service.Emails/Service.Emails.csproj" />
|
<Project Path="Service/Service.Emails/Service.Emails.csproj" />
|
||||||
<Project Path="Service/Service.UserManagement/Service.UserManagement.csproj" />
|
<Project Path="Service/Service.UserManagement/Service.UserManagement.csproj" />
|
||||||
<Project Path="Service/Service.Auth/Service.Auth.csproj" />
|
<Project Path="Service\Service.Auth\Service.Auth.csproj" />
|
||||||
<Project Path="Service/Service.Breweries/Service.Breweries.csproj" />
|
|
||||||
</Folder>
|
</Folder>
|
||||||
</Solution>
|
</Solution>
|
||||||
@@ -37,7 +37,7 @@ CREATE TABLE dbo.UserAccount
|
|||||||
|
|
||||||
UpdatedAt DATETIME,
|
UpdatedAt DATETIME,
|
||||||
|
|
||||||
DateOfBirth DATE NOT NULL,
|
DateOfBirth DATETIME NOT NULL,
|
||||||
|
|
||||||
Timer ROWVERSION,
|
Timer ROWVERSION,
|
||||||
|
|
||||||
@@ -49,6 +49,7 @@ CREATE TABLE dbo.UserAccount
|
|||||||
|
|
||||||
CONSTRAINT AK_Email
|
CONSTRAINT AK_Email
|
||||||
UNIQUE (Email)
|
UNIQUE (Email)
|
||||||
|
|
||||||
);
|
);
|
||||||
|
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
@@ -108,7 +109,7 @@ CREATE TABLE UserAvatar -- delete avatar photo when user account is deleted
|
|||||||
|
|
||||||
CONSTRAINT AK_UserAvatar_UserAccountID
|
CONSTRAINT AK_UserAvatar_UserAccountID
|
||||||
UNIQUE (UserAccountID)
|
UNIQUE (UserAccountID)
|
||||||
);
|
)
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_UserAvatar_UserAccount
|
CREATE NONCLUSTERED INDEX IX_UserAvatar_UserAccount
|
||||||
ON UserAvatar(UserAccountID);
|
ON UserAvatar(UserAccountID);
|
||||||
@@ -124,7 +125,8 @@ CREATE TABLE UserVerification -- delete verification data when user account is d
|
|||||||
UserAccountID UNIQUEIDENTIFIER NOT NULL,
|
UserAccountID UNIQUEIDENTIFIER NOT NULL,
|
||||||
|
|
||||||
VerificationDateTime DATETIME NOT NULL
|
VerificationDateTime DATETIME NOT NULL
|
||||||
CONSTRAINT DF_VerificationDateTime DEFAULT GETDATE(),
|
CONSTRAINT DF_VerificationDateTime
|
||||||
|
DEFAULT GETDATE(),
|
||||||
|
|
||||||
Timer ROWVERSION,
|
Timer ROWVERSION,
|
||||||
|
|
||||||
@@ -153,13 +155,13 @@ CREATE TABLE UserCredential -- delete credentials when user account is deleted
|
|||||||
|
|
||||||
UserAccountID UNIQUEIDENTIFIER NOT NULL,
|
UserAccountID UNIQUEIDENTIFIER NOT NULL,
|
||||||
|
|
||||||
CreatedAt DATETIME NOT NULL
|
CreatedAt DATETIME
|
||||||
CONSTRAINT DF_UserCredential_CreatedAt DEFAULT GETDATE(),
|
CONSTRAINT DF_UserCredential_CreatedAt DEFAULT GETDATE() NOT NULL,
|
||||||
|
|
||||||
Expiry DATETIME NOT NULL
|
Expiry DATETIME
|
||||||
CONSTRAINT DF_UserCredential_Expiry DEFAULT DATEADD(DAY, 90, GETDATE()),
|
CONSTRAINT DF_UserCredential_Expiry DEFAULT DATEADD(DAY, 90, GETDATE()) NOT NULL,
|
||||||
|
|
||||||
Hash NVARCHAR(256) NOT NULL,
|
Hash NVARCHAR(MAX) NOT NULL,
|
||||||
-- uses argon2
|
-- uses argon2
|
||||||
|
|
||||||
IsRevoked BIT NOT NULL
|
IsRevoked BIT NOT NULL
|
||||||
@@ -175,16 +177,12 @@ CREATE TABLE UserCredential -- delete credentials when user account is deleted
|
|||||||
CONSTRAINT FK_UserCredential_UserAccount
|
CONSTRAINT FK_UserCredential_UserAccount
|
||||||
FOREIGN KEY (UserAccountID)
|
FOREIGN KEY (UserAccountID)
|
||||||
REFERENCES UserAccount(UserAccountID)
|
REFERENCES UserAccount(UserAccountID)
|
||||||
ON DELETE CASCADE
|
ON DELETE CASCADE,
|
||||||
);
|
);
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_UserCredential_UserAccount
|
CREATE NONCLUSTERED INDEX IX_UserCredential_UserAccount
|
||||||
ON UserCredential(UserAccountID);
|
ON UserCredential(UserAccountID);
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_UserCredential_Account_Active
|
|
||||||
ON UserCredential(UserAccountID, IsRevoked, Expiry)
|
|
||||||
INCLUDE (Hash);
|
|
||||||
|
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -197,8 +195,8 @@ CREATE TABLE UserFollow
|
|||||||
|
|
||||||
FollowingID UNIQUEIDENTIFIER NOT NULL,
|
FollowingID UNIQUEIDENTIFIER NOT NULL,
|
||||||
|
|
||||||
CreatedAt DATETIME NOT NULL
|
CreatedAt DATETIME
|
||||||
CONSTRAINT DF_UserFollow_CreatedAt DEFAULT GETDATE(),
|
CONSTRAINT DF_UserFollow_CreatedAt DEFAULT GETDATE() NOT NULL,
|
||||||
|
|
||||||
Timer ROWVERSION,
|
Timer ROWVERSION,
|
||||||
|
|
||||||
@@ -207,13 +205,11 @@ CREATE TABLE UserFollow
|
|||||||
|
|
||||||
CONSTRAINT FK_UserFollow_UserAccount
|
CONSTRAINT FK_UserFollow_UserAccount
|
||||||
FOREIGN KEY (UserAccountID)
|
FOREIGN KEY (UserAccountID)
|
||||||
REFERENCES UserAccount(UserAccountID)
|
REFERENCES UserAccount(UserAccountID),
|
||||||
ON DELETE NO ACTION,
|
|
||||||
|
|
||||||
CONSTRAINT FK_UserFollow_UserAccountFollowing
|
CONSTRAINT FK_UserFollow_UserAccountFollowing
|
||||||
FOREIGN KEY (FollowingID)
|
FOREIGN KEY (FollowingID)
|
||||||
REFERENCES UserAccount(UserAccountID)
|
REFERENCES UserAccount(UserAccountID),
|
||||||
ON DELETE NO ACTION,
|
|
||||||
|
|
||||||
CONSTRAINT CK_CannotFollowOwnAccount
|
CONSTRAINT CK_CannotFollowOwnAccount
|
||||||
CHECK (UserAccountID != FollowingID)
|
CHECK (UserAccountID != FollowingID)
|
||||||
@@ -225,6 +221,7 @@ CREATE NONCLUSTERED INDEX IX_UserFollow_UserAccount_FollowingID
|
|||||||
CREATE NONCLUSTERED INDEX IX_UserFollow_FollowingID_UserAccount
|
CREATE NONCLUSTERED INDEX IX_UserFollow_FollowingID_UserAccount
|
||||||
ON UserFollow(FollowingID, UserAccountID);
|
ON UserFollow(FollowingID, UserAccountID);
|
||||||
|
|
||||||
|
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -302,6 +299,7 @@ CREATE TABLE City
|
|||||||
CREATE NONCLUSTERED INDEX IX_City_StateProvince
|
CREATE NONCLUSTERED INDEX IX_City_StateProvince
|
||||||
ON City(StateProvinceID);
|
ON City(StateProvinceID);
|
||||||
|
|
||||||
|
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -310,8 +308,6 @@ CREATE TABLE BreweryPost -- A user cannot be deleted if they have a post
|
|||||||
BreweryPostID UNIQUEIDENTIFIER
|
BreweryPostID UNIQUEIDENTIFIER
|
||||||
CONSTRAINT DF_BreweryPostID DEFAULT NEWID(),
|
CONSTRAINT DF_BreweryPostID DEFAULT NEWID(),
|
||||||
|
|
||||||
BreweryName NVARCHAR(256) NOT NULL,
|
|
||||||
|
|
||||||
PostedByID UNIQUEIDENTIFIER NOT NULL,
|
PostedByID UNIQUEIDENTIFIER NOT NULL,
|
||||||
|
|
||||||
Description NVARCHAR(512) NOT NULL,
|
Description NVARCHAR(512) NOT NULL,
|
||||||
@@ -329,15 +325,15 @@ CREATE TABLE BreweryPost -- A user cannot be deleted if they have a post
|
|||||||
CONSTRAINT FK_BreweryPost_UserAccount
|
CONSTRAINT FK_BreweryPost_UserAccount
|
||||||
FOREIGN KEY (PostedByID)
|
FOREIGN KEY (PostedByID)
|
||||||
REFERENCES UserAccount(UserAccountID)
|
REFERENCES UserAccount(UserAccountID)
|
||||||
ON DELETE NO ACTION
|
ON DELETE NO ACTION,
|
||||||
);
|
|
||||||
|
)
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_BreweryPost_PostedByID
|
CREATE NONCLUSTERED INDEX IX_BreweryPost_PostedByID
|
||||||
ON BreweryPost(PostedByID);
|
ON BreweryPost(PostedByID);
|
||||||
|
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
|
|
||||||
CREATE TABLE BreweryPostLocation
|
CREATE TABLE BreweryPostLocation
|
||||||
(
|
(
|
||||||
BreweryPostLocationID UNIQUEIDENTIFIER
|
BreweryPostLocationID UNIQUEIDENTIFIER
|
||||||
@@ -353,7 +349,7 @@ CREATE TABLE BreweryPostLocation
|
|||||||
|
|
||||||
CityID UNIQUEIDENTIFIER NOT NULL,
|
CityID UNIQUEIDENTIFIER NOT NULL,
|
||||||
|
|
||||||
Coordinates GEOGRAPHY NULL,
|
Coordinates GEOGRAPHY NOT NULL,
|
||||||
|
|
||||||
Timer ROWVERSION,
|
Timer ROWVERSION,
|
||||||
|
|
||||||
@@ -366,11 +362,7 @@ CREATE TABLE BreweryPostLocation
|
|||||||
CONSTRAINT FK_BreweryPostLocation_BreweryPost
|
CONSTRAINT FK_BreweryPostLocation_BreweryPost
|
||||||
FOREIGN KEY (BreweryPostID)
|
FOREIGN KEY (BreweryPostID)
|
||||||
REFERENCES BreweryPost(BreweryPostID)
|
REFERENCES BreweryPost(BreweryPostID)
|
||||||
ON DELETE CASCADE,
|
ON DELETE CASCADE
|
||||||
|
|
||||||
CONSTRAINT FK_BreweryPostLocation_City
|
|
||||||
FOREIGN KEY (CityID)
|
|
||||||
REFERENCES City(CityID)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_BreweryPostLocation_BreweryPost
|
CREATE NONCLUSTERED INDEX IX_BreweryPostLocation_BreweryPost
|
||||||
@@ -379,18 +371,6 @@ CREATE NONCLUSTERED INDEX IX_BreweryPostLocation_BreweryPost
|
|||||||
CREATE NONCLUSTERED INDEX IX_BreweryPostLocation_City
|
CREATE NONCLUSTERED INDEX IX_BreweryPostLocation_City
|
||||||
ON BreweryPostLocation(CityID);
|
ON BreweryPostLocation(CityID);
|
||||||
|
|
||||||
-- To assess when the time comes:
|
|
||||||
|
|
||||||
-- This would allow for efficient spatial queries to find breweries within a certain distance of a location, but it adds overhead to insert/update operations.
|
|
||||||
|
|
||||||
-- CREATE SPATIAL INDEX SIDX_BreweryPostLocation_Coordinates
|
|
||||||
-- ON BreweryPostLocation(Coordinates)
|
|
||||||
-- USING GEOGRAPHY_GRID
|
|
||||||
-- WITH (
|
|
||||||
-- GRIDS = (LEVEL_1 = MEDIUM, LEVEL_2 = MEDIUM, LEVEL_3 = MEDIUM, LEVEL_4 = MEDIUM),
|
|
||||||
-- CELLS_PER_OBJECT = 16
|
|
||||||
-- );
|
|
||||||
|
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -423,14 +403,13 @@ CREATE TABLE BreweryPostPhoto -- All photos linked to a post are deleted if the
|
|||||||
);
|
);
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_BreweryPostPhoto_Photo_BreweryPost
|
CREATE NONCLUSTERED INDEX IX_BreweryPostPhoto_Photo_BreweryPost
|
||||||
ON BreweryPostPhoto(PhotoID, BreweryPostID);
|
ON BreweryPostPhoto(PhotoID, BreweryPostID);
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_BreweryPostPhoto_BreweryPost_Photo
|
CREATE NONCLUSTERED INDEX IX_BreweryPostPhoto_BreweryPost_Photo
|
||||||
ON BreweryPostPhoto(BreweryPostID, PhotoID);
|
ON BreweryPostPhoto(BreweryPostID, PhotoID);
|
||||||
|
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
|
|
||||||
CREATE TABLE BeerStyle
|
CREATE TABLE BeerStyle
|
||||||
(
|
(
|
||||||
BeerStyleID UNIQUEIDENTIFIER
|
BeerStyleID UNIQUEIDENTIFIER
|
||||||
@@ -465,7 +444,7 @@ CREATE TABLE BeerPost
|
|||||||
-- Alcohol By Volume (typically 0-67%)
|
-- Alcohol By Volume (typically 0-67%)
|
||||||
|
|
||||||
IBU INT NOT NULL,
|
IBU INT NOT NULL,
|
||||||
-- International Bitterness Units (typically 0-120)
|
-- International Bitterness Units (typically 0-100)
|
||||||
|
|
||||||
PostedByID UNIQUEIDENTIFIER NOT NULL,
|
PostedByID UNIQUEIDENTIFIER NOT NULL,
|
||||||
|
|
||||||
@@ -485,8 +464,7 @@ CREATE TABLE BeerPost
|
|||||||
|
|
||||||
CONSTRAINT FK_BeerPost_PostedBy
|
CONSTRAINT FK_BeerPost_PostedBy
|
||||||
FOREIGN KEY (PostedByID)
|
FOREIGN KEY (PostedByID)
|
||||||
REFERENCES UserAccount(UserAccountID)
|
REFERENCES UserAccount(UserAccountID),
|
||||||
ON DELETE NO ACTION,
|
|
||||||
|
|
||||||
CONSTRAINT FK_BeerPost_BeerStyle
|
CONSTRAINT FK_BeerPost_BeerStyle
|
||||||
FOREIGN KEY (BeerStyleID)
|
FOREIGN KEY (BeerStyleID)
|
||||||
@@ -544,10 +522,10 @@ CREATE TABLE BeerPostPhoto -- All photos linked to a beer post are deleted if th
|
|||||||
);
|
);
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_BeerPostPhoto_Photo_BeerPost
|
CREATE NONCLUSTERED INDEX IX_BeerPostPhoto_Photo_BeerPost
|
||||||
ON BeerPostPhoto(PhotoID, BeerPostID);
|
ON BeerPostPhoto(PhotoID, BeerPostID);
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_BeerPostPhoto_BeerPost_Photo
|
CREATE NONCLUSTERED INDEX IX_BeerPostPhoto_BeerPost_Photo
|
||||||
ON BeerPostPhoto(BeerPostID, PhotoID);
|
ON BeerPostPhoto(BeerPostID, PhotoID);
|
||||||
|
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
@@ -561,35 +539,17 @@ CREATE TABLE BeerPostComment
|
|||||||
|
|
||||||
BeerPostID UNIQUEIDENTIFIER NOT NULL,
|
BeerPostID UNIQUEIDENTIFIER NOT NULL,
|
||||||
|
|
||||||
CommentedByID UNIQUEIDENTIFIER NOT NULL,
|
|
||||||
|
|
||||||
Rating INT NOT NULL,
|
Rating INT NOT NULL,
|
||||||
|
|
||||||
CreatedAt DATETIME NOT NULL
|
|
||||||
CONSTRAINT DF_BeerPostComment_CreatedAt DEFAULT GETDATE(),
|
|
||||||
|
|
||||||
UpdatedAt DATETIME NULL,
|
|
||||||
|
|
||||||
Timer ROWVERSION,
|
Timer ROWVERSION,
|
||||||
|
|
||||||
CONSTRAINT PK_BeerPostComment
|
CONSTRAINT PK_BeerPostComment
|
||||||
PRIMARY KEY (BeerPostCommentID),
|
PRIMARY KEY (BeerPostCommentID),
|
||||||
|
|
||||||
CONSTRAINT FK_BeerPostComment_BeerPost
|
CONSTRAINT FK_BeerPostComment_BeerPost
|
||||||
FOREIGN KEY (BeerPostID)
|
FOREIGN KEY (BeerPostID) REFERENCES BeerPost(BeerPostID)
|
||||||
REFERENCES BeerPost(BeerPostID),
|
)
|
||||||
|
|
||||||
CONSTRAINT FK_BeerPostComment_UserAccount
|
|
||||||
FOREIGN KEY (CommentedByID)
|
|
||||||
REFERENCES UserAccount(UserAccountID)
|
|
||||||
ON DELETE NO ACTION,
|
|
||||||
|
|
||||||
CONSTRAINT CHK_BeerPostComment_Rating
|
|
||||||
CHECK (Rating BETWEEN 1 AND 5)
|
|
||||||
);
|
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_BeerPostComment_BeerPost
|
CREATE NONCLUSTERED INDEX IX_BeerPostComment_BeerPost
|
||||||
ON BeerPostComment(BeerPostID);
|
ON BeerPostComment(BeerPostID)
|
||||||
|
|
||||||
CREATE NONCLUSTERED INDEX IX_BeerPostComment_CommentedBy
|
|
||||||
ON BeerPostComment(CommentedByID);
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user