This commit is contained in:
2026-05-31 20:19:44 +12:00
parent 2f2466ecac
commit 84792c0947
59 changed files with 5412 additions and 898 deletions
+1 -1
View File
@@ -22,7 +22,7 @@ PUBLIC_API_BASE_URL=https://clients.example.com
INTERNAL_API_BASE_URL=http://backend:8000 INTERNAL_API_BASE_URL=http://backend:8000
CORS_ALLOW_ORIGINS=https://clients.example.com CORS_ALLOW_ORIGINS=https://clients.example.com
CORS_ALLOW_ORIGIN_REGEX= CORS_ALLOW_ORIGIN_REGEX=
TRUSTED_HOSTS=clients.example.com TRUSTED_HOSTS=clients.example.com,127.0.0.1,localhost
CLIENTS_APP_PORT=8081 CLIENTS_APP_PORT=8081
SESSION_COOKIE_SECURE=true SESSION_COOKIE_SECURE=true
SESSION_COOKIE_SAMESITE=lax SESSION_COOKIE_SAMESITE=lax
+373
View File
@@ -0,0 +1,373 @@
# Database Design
## Purpose
This app uses a relational database to support five main concerns:
1. Raw material pricing and unit conversion.
2. Mix definitions and mix costing.
3. Product-level formulas and product costing.
4. Mix calculator session history.
5. Access control for both internal users and client users.
The backend is written with SQLAlchemy models in `backend/app/models`. The schema is created automatically at startup, and lightweight migration/patch logic lives in `backend/app/db/migrations.py`.
## Design Principles
- `tenant_id` is the tenancy boundary for most business tables.
- Reference/master data is stored separately from transactional/session data.
- Product costing is built from raw materials -> formulas -> products -> outputs.
- The mix calculator now prefers product-specific ingredient formulas over the shared mix master.
- The database is designed to run on both SQLite locally and Postgres in production.
## High-Level Domains
### 1. Raw Materials
These tables store ingredients and their price history.
- `raw_materials`
- One row per ingredient/raw material.
- Stores name, supplier, unit of measure, `kg_per_unit`, status, and notes.
- Example: `Hulled Oats`, `White French Millet`, `Pano`.
- `raw_material_price_versions`
- One-to-many from `raw_materials`.
- Stores `market_value`, `waste_percentage`, `effective_date`, and status.
- Lets the system keep historical prices instead of overwriting one current value.
Relationship:
- `raw_materials.id` -> `raw_material_price_versions.raw_material_id`
### 2. Mix Master
These tables store shared mix definitions.
- `mixes`
- One row per named mix.
- Stores client name, mix name, version, status, and notes.
- This is the shared mix/master-recipe layer.
- `mix_ingredients`
- One-to-many from `mixes`.
- One row per raw material inside a mix.
- Stores `quantity_kg` for that mix.
Relationships:
- `mixes.id` -> `mix_ingredients.mix_id`
- `raw_materials.id` -> `mix_ingredients.raw_material_id`
Important note:
- This table is still used by mix master pages and as a fallback.
- It is no longer the primary source for mix calculator formulas when product-specific formulas exist.
### 3. Products
These tables describe saleable products and their formula rows.
- `products`
- One row per sellable product/SKU.
- Stores client name, product name, optional `item_id`, packaging/unit info, margins, and linked mix.
- `mix_id` links the product to the shared mix master entry.
- `product_ingredients`
- One-to-many from `products`.
- One row per raw material required for that products formula.
- Stores `quantity_kg`, `sort_order`, and optional notes.
- This is now the key table for the mix calculator.
Relationships:
- `products.mix_id` -> `mixes.id`
- `products.id` -> `product_ingredients.product_id`
- `raw_materials.id` -> `product_ingredients.raw_material_id`
Why both `mix_ingredients` and `product_ingredients` exist:
- `mix_ingredients` represents a shared recipe.
- `product_ingredients` represents the actual formula used for a specific product.
- Multiple products can point at the same mix name but still require product-specific formula rows.
- This solves the workbook case where product labels like `Budgie Mix 20kg` map to a formula/mix name like `Hunter - Budgie Mix`.
### 4. Costing Assumptions
These tables hold non-ingredient costs used in product costing.
- `process_cost_rules`
- Holds grading, bagging, and cracking costs by `process_name`.
- `packaging_cost_rules`
- Holds bag cost by `sale_type`, `unit_of_measure`, and `own_bag`.
- `freight_cost_rules`
- Holds freight cost by `sale_type` and `unit_of_measure`.
These tables are read during product cost calculation after ingredient cost has been resolved.
### 5. Scenarios and Stored Outputs
- `scenarios`
- Named pricing/costing scenarios.
- Stores `overrides` as JSON.
- `costing_results`
- One-to-many from `scenarios`.
- Stores calculated output per product for a scenario.
- Includes prices, warnings, and calculation details as JSON.
Relationships:
- `scenarios.id` -> `costing_results.scenario_id`
- `products.id` -> `costing_results.product_id`
### 6. Mix Calculator Sessions
These tables store saved calculator runs.
- `mix_calculator_sessions`
- Header row for a calculator run.
- Stores product, mix, batch size, total bags, total kg, prepared by, and timestamps.
- `mix_calculator_session_lines`
- One-to-many from `mix_calculator_sessions`.
- Snapshot of the scaled ingredient rows shown to the user at save time.
- Stores `required_kg`, `mix_percentage`, unit, and display name.
Relationships:
- `mix_calculator_sessions.id` -> `mix_calculator_session_lines.session_id`
- `products.id` -> `mix_calculator_sessions.product_id`
- `mixes.id` -> `mix_calculator_sessions.mix_id`
Important note:
- Session lines are denormalized snapshots.
- They are intentionally stored separately so historical saved runs do not change if product formulas are updated later.
### 7. Client Access / Tenant Administration
These tables manage customer-facing users and feature/module access.
- `client_accounts`
- One row per client/tenant account.
- `client_users`
- One-to-many from `client_accounts`.
- Customer-side users tied to a client account.
- `client_feature_access`
- One-to-many from `client_accounts`.
- Feature flags per client account.
- `client_user_module_permissions`
- One-to-many from `client_users`.
- Module-level access levels per client user.
- `client_access_audit_events`
- One-to-many from `client_accounts`.
- Audit log for client-access changes.
Relationships:
- `client_accounts.id` -> `client_users.client_account_id`
- `client_accounts.id` -> `client_feature_access.client_account_id`
- `client_accounts.id` -> `client_access_audit_events.client_account_id`
- `client_users.id` -> `client_user_module_permissions.client_user_id`
### 8. Internal Access Control
These tables are for internal staff login and permissions.
- `users`
- Internal users.
- Stores per-user `password_hash`, role link, and active flag.
- `roles`
- Named roles like `Admin`, `Operations`, `Full Access`.
- `permissions`
- Atomic permission keys like `view_mix_calculator`.
- `role_permissions`
- Many-to-many join table between roles and permissions.
Relationships:
- `roles.id` -> `users.role_id`
- `roles.id` <-> `permissions.id` through `role_permissions`
## Core Costing Flow
### Raw Material Cost
The system calculates ingredient cost from:
- `market_value`
- `waste_percentage`
- `kg_per_unit`
This produces:
- loss cost
- adjusted cost per unit
- cost per kg
### Mix Cost
There are now two formula sources:
1. Preferred: `product_ingredients`
2. Fallback: `mix_ingredients`
For mix calculator and product costing:
- if a product has rows in `product_ingredients`, use them
- otherwise use the linked shared mix from `mix_ingredients`
### Product Cost
Product cost is built from:
1. ingredient formula cost
2. process costs
3. packaging cost
4. freight cost
5. optional distributor / wholesale margin
## Workbook Import Design
The seed/import logic is in `backend/app/seed.py`.
There are now two workbook roles:
- Legacy costing workbook:
- `C- Raw Products Costs`
- `M - All`
- `Product Cost - Price`
- Product-formula workbook:
- `input_data/1.xlsx`
- sheet `mix_quantites_per_client_per_pr`
### Current Import Behaviour
- Raw materials are seeded from the legacy costing workbook.
- Shared mixes are seeded from the legacy costing workbook.
- Products are seeded from the legacy costing workbook.
- Product-specific formulas are seeded from `mix_quantites_per_client_per_pr`.
### Formula Matching Rule
Workbook formula rows are attached to products using:
1. `(client_name, product.name)` if it matches directly.
2. `(client_name, product.mix.name)` if the workbook row uses the mix/formula name instead of the sellable product label.
This is important for cases like:
- workbook formula: `HunterBird / Hunter - Budgie Mix`
- product row: `HunterBird / Budgie Mix 20kg`
Both product SKUs can inherit the same formula through the linked mix name.
## Tenancy
Most business tables include `tenant_id`.
This includes:
- raw materials
- price versions
- mixes
- mix ingredients
- product ingredients
- products
- scenarios
- costing results
- mix calculator sessions and lines
- client-access tables
- assumption tables
Startup migration logic backfills `tenant_id` where possible by deriving it from related parent tables.
## Visibility Rules
The `products.visible` flag is used to hide client/product rows from normal UI paths.
Startup migration logic also auto-hides products for a configured list of client names in `backend/app/db/migrations.py`.
This means:
- rows can exist in the database
- but not be offered in normal mix calculator/product selection flows
## Transaction vs Reference Data
Reference/master data:
- `raw_materials`
- `raw_material_price_versions`
- `mixes`
- `mix_ingredients`
- `products`
- `product_ingredients`
- `process_cost_rules`
- `packaging_cost_rules`
- `freight_cost_rules`
- access-control tables
Transactional/snapshot data:
- `mix_calculator_sessions`
- `mix_calculator_session_lines`
- `scenarios`
- `costing_results`
- `client_access_audit_events`
## Important Constraints
- `mix_ingredients` is unique on `(mix_id, raw_material_id)`.
- `product_ingredients` is unique on `(product_id, raw_material_id)`.
- `client_users` is unique on `(client_account_id, email)`.
- `client_feature_access` is unique on `(client_account_id, feature_key)`.
- `client_user_module_permissions` is unique on `(client_user_id, module_key)`.
- `mix_calculator_sessions` is unique on `(tenant_id, session_number)`.
These constraints prevent duplicate ingredient or access rows within the same parent scope.
## Known Tradeoffs
- `RawMaterial.name` is globally unique, not tenant-scoped. That is simple for now, but stricter than a multi-tenant design usually wants.
- `Product.mix_id` is still required even though product-specific formulas now exist. That is useful for compatibility and navigation, but it means a product currently has both a shared mix link and potentially its own formula rows.
- Some calculation outputs are denormalized into session/result tables for stability and history.
- Migration logic is startup-driven and pragmatic rather than using a full migration framework like Alembic.
## Recommended Mental Model
Use this as the working model of the schema:
- `raw_materials` = ingredients
- `raw_material_price_versions` = ingredient pricing history
- `mixes` = shared recipe labels
- `mix_ingredients` = shared recipe lines
- `products` = saleable SKUs
- `product_ingredients` = actual formula for a SKU
- `mix_calculator_sessions` + `lines` = saved production calculations
- `scenarios` + `costing_results` = stored what-if pricing outputs
- `client_*` tables = client account access
- `users / roles / permissions` = internal staff access
## Files To Read Alongside This Document
- [backend/app/models/raw_material.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/models/raw_material.py:1)
- [backend/app/models/mix.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/models/mix.py:1)
- [backend/app/models/product.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/models/product.py:1)
- [backend/app/models/mix_calculator.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/models/mix_calculator.py:1)
- [backend/app/models/scenario.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/models/scenario.py:1)
- [backend/app/models/assumption.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/models/assumption.py:1)
- [backend/app/models/client_access.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/models/client_access.py:1)
- [backend/app/models/access.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/models/access.py:1)
- [backend/app/db/migrations.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/db/migrations.py:1)
- [backend/app/seed.py](/abs/path/C:/Users/mattc/data-entry-app/data-entry-app/backend/app/seed.py:1)
+21
View File
@@ -90,10 +90,17 @@ If your server already has a host-level nginx handling domains and TLS, use `dep
Useful flags: `-Branch <name>` to deploy a feature branch, `-SkipBuild` for env-only changes, `-Seed` to re-run reference data seeding, `-Logs` to tail logs after the deploy, `-SshKey` to point at a specific private key. Useful flags: `-Branch <name>` to deploy a feature branch, `-SkipBuild` for env-only changes, `-Seed` to re-run reference data seeding, `-Logs` to tail logs after the deploy, `-SshKey` to point at a specific private key.
If a release adds or changes database-backed workbook formula structures, deploy with `-Seed` so the server refreshes seeded reference/formula data after the backend starts. For the product-formula change, this is required so Postgres receives the new `product_ingredients` rows sourced from `input_data/1.xlsx`.
5. **Database**: the backend reads `DATABASE_URL`. The production compose file synthesises it as `postgresql+psycopg://${POSTGRES_USER}:${POSTGRES_PASSWORD}@db:5432/${POSTGRES_DB}` so you only need to set the three `POSTGRES_*` vars. Override `DATABASE_URL` directly if you point at a managed Postgres (e.g. DigitalOcean managed databases). 5. **Database**: the backend reads `DATABASE_URL`. The production compose file synthesises it as `postgresql+psycopg://${POSTGRES_USER}:${POSTGRES_PASSWORD}@db:5432/${POSTGRES_DB}` so you only need to set the three `POSTGRES_*` vars. Override `DATABASE_URL` directly if you point at a managed Postgres (e.g. DigitalOcean managed databases).
The schema is auto-managed — `app/db/migrations.py` runs at backend startup and is idempotent across SQLite and Postgres. To migrate alpha SQLite data into the new Postgres instance, dump tables to CSV from the alpha container and import via `\copy` in `psql`; there is no automatic SQLite → Postgres path. The schema is auto-managed — `app/db/migrations.py` runs at backend startup and is idempotent across SQLite and Postgres. To migrate alpha SQLite data into the new Postgres instance, dump tables to CSV from the alpha container and import via `\copy` in `psql`; there is no automatic SQLite → Postgres path.
For this repos current schema, new tables such as `product_ingredients` are created automatically on backend startup in both SQLite and Postgres. Existing data refreshes still depend on seeding, so schema deployment and data deployment are separate concerns:
- backend startup creates missing tables/columns
- `-Seed` repopulates workbook-driven rows inside those tables
## Backend ## Backend
Create a virtual environment, install dependencies, then run: Create a virtual environment, install dependencies, then run:
@@ -116,6 +123,20 @@ pytest
The backend defaults to SQLite for the prototype and can be switched with the The backend defaults to SQLite for the prototype and can be switched with the
`DATABASE_URL` environment variable. `DATABASE_URL` environment variable.
For local non-Docker runs, the default SQLite database is
`backend/data_entry_app.db` regardless of which directory you launch the
backend from. This avoids accidentally creating multiple local SQLite files
with different login data.
The internal login screen at `/` uses the seeded Hunter Stock Feeds users:
- `admin@hunterstockfeeds.com`
- `ops@hunterstockfeeds.com`
- `craig@hunterstockfeeds.com`
Unless you override `ADMIN_PASSWORD` before the first seed, those local
internal users are seeded with the default password `lean101-admin`.
### Backend logging ### Backend logging
The backend now uses a shared console logger with a styled startup banner, concise request logs, and clean shutdown summaries. The backend now uses a shared console logger with a styled startup banner, concise request logs, and clean shutdown summaries.
+2
View File
@@ -8,6 +8,8 @@ WORKDIR /app
RUN addgroup --system app && adduser --system --ingroup app app RUN addgroup --system app && adduser --system --ingroup app app
COPY backend /app COPY backend /app
COPY ["input_data/1.xlsx", "/app/input_data/1.xlsx"]
COPY ["Input Cost Spreadsheet(1).xlsx", "/app/Input Cost Spreadsheet(1).xlsx"]
RUN pip install --no-cache-dir --upgrade pip && \ RUN pip install --no-cache-dir --upgrade pip && \
pip install --no-cache-dir . && \ pip install --no-cache-dir . && \
+14 -14
View File
@@ -109,15 +109,11 @@ def _serialize_session(user: User, *, include_token: bool = False) -> UserSessio
def login(payload: LoginRequest, response: Response, request: Request, db: Session = Depends(get_db)): def login(payload: LoginRequest, response: Response, request: Request, db: Session = Depends(get_db)):
"""Internal-user login. """Internal-user login.
Authenticates against a shared internal password (``ADMIN_PASSWORD``) and Authenticates against the per-user password hash stored on ``users``.
looks up the user by email. Inactive or unknown users are rejected with Inactive or unknown users are rejected with a generic 401 to avoid
a generic 401 to avoid leaking which emails are valid. leaking which emails are valid.
""" """
login_rate_limiter.hit(request_client_key(request, suffix="internal-login")) login_rate_limiter.hit(request_client_key(request, suffix="internal-login"))
if payload.password != settings.admin_password:
log_security_event("auth.login_failed", audience="internal", ip=request_client_key(request))
raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED, detail="Invalid email or password")
email = payload.email.strip().lower() email = payload.email.strip().lower()
user = db.scalar( user = db.scalar(
select(User) select(User)
@@ -127,6 +123,12 @@ def login(payload: LoginRequest, response: Response, request: Request, db: Sessi
if user is None or not user.is_active: if user is None or not user.is_active:
log_security_event("auth.login_failed", audience="internal", ip=request_client_key(request)) log_security_event("auth.login_failed", audience="internal", ip=request_client_key(request))
raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED, detail="Invalid email or password") raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED, detail="Invalid email or password")
if not (
verify_password(payload.password, user.password_hash)
or (user.password_hash is None and payload.password == settings.admin_password)
):
log_security_event("auth.login_failed", audience="internal", ip=request_client_key(request))
raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED, detail="Invalid email or password")
session = _serialize_session(user, include_token=True) session = _serialize_session(user, include_token=True)
if session.token: if session.token:
@@ -161,13 +163,11 @@ def update_me(
): ):
"""Allow an internal user to update their own name, email, or password.""" """Allow an internal user to update their own name, email, or password."""
if payload.new_password: if payload.new_password:
# Require current password verification before allowing a password change. # Require current password verification before allowing a password
# Users who have never set a personal password must supply the shared # change. Keep a narrow fallback for legacy rows that still have no
# admin password as the current credential. # password hash yet.
current_ok = ( current_ok = verify_password(payload.current_password or "", user.password_hash) or (
verify_password(payload.current_password or "", user.password_hash) user.password_hash is None and (payload.current_password or "") == settings.admin_password
if user.password_hash
else (payload.current_password or "") == settings.admin_password
) )
if not current_ok: if not current_ok:
raise HTTPException( raise HTTPException(
+23 -1
View File
@@ -22,7 +22,7 @@ from app.services.mix_calculator_service import (
update_mix_calculator_session, update_mix_calculator_session,
) )
from app.services.mix_calculator_pdf import MixCalculatorPdfUnavailableError, build_mix_calculator_pdf from app.services.mix_calculator_pdf import MixCalculatorPdfUnavailableError, build_mix_calculator_pdf
from app.services.mix_calculator_filenames import mix_calculator_pdf_filename from app.services.mix_calculator_filenames import mix_calculator_pdf_filename, mix_calculator_preview_pdf_filename
router = APIRouter(prefix="/api/mix-calculator", tags=["mix-calculator"]) router = APIRouter(prefix="/api/mix-calculator", tags=["mix-calculator"])
@@ -56,6 +56,28 @@ def preview_mix_calculator_session(
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail=str(exc)) from exc raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail=str(exc)) from exc
@router.post("/preview/pdf")
def preview_mix_calculator_session_pdf(
payload: MixCalculatorSessionCreate,
session: AuthSession = Depends(require_client_module_access("mix_calculator", "edit")),
db: Session = Depends(get_db),
):
try:
preview = calculate_mix_calculator_preview(db, tenant_id=session.tenant_id or "", payload=payload)
pdf_bytes = build_mix_calculator_pdf(preview)
except ValueError as exc:
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail=str(exc)) from exc
except MixCalculatorPdfUnavailableError as exc:
raise HTTPException(status_code=status.HTTP_503_SERVICE_UNAVAILABLE, detail=str(exc)) from exc
filename = mix_calculator_preview_pdf_filename(MixCalculatorPreviewRead.model_validate(preview))
return Response(
content=pdf_bytes,
media_type="application/pdf",
headers={"Content-Disposition": f'attachment; filename="{filename}"'},
)
@router.post("", response_model=MixCalculatorSessionRead, status_code=status.HTTP_201_CREATED) @router.post("", response_model=MixCalculatorSessionRead, status_code=status.HTTP_201_CREATED)
def create_saved_mix_calculator_session( def create_saved_mix_calculator_session(
payload: MixCalculatorSessionCreate, payload: MixCalculatorSessionCreate,
+245
View File
@@ -0,0 +1,245 @@
from __future__ import annotations
from datetime import date
from fastapi import APIRouter, Depends, HTTPException, Query, status
from sqlalchemy import select
from sqlalchemy.orm import Session
from app.api.deps import AuthSession, require_client_module_access
from app.db.session import get_db
from app.models.throughput import ProductionThroughput, ThroughputProduct
from app.schemas.throughput import (
ThroughputEntryCreate,
ThroughputEntryRead,
ThroughputEntryUpdate,
ThroughputProductCreate,
ThroughputProductRead,
ThroughputProductUpdate,
)
from app.services.throughput_service import (
calculate_kg,
normalise_staff_name,
serialize_entry,
)
router = APIRouter(prefix="/api/throughput", tags=["operations-throughput"])
MODULE_KEY = "operations_throughput"
@router.get("/products", response_model=list[ThroughputProductRead])
def list_products(
include_inactive: bool = Query(default=False),
session: AuthSession = Depends(require_client_module_access(MODULE_KEY)),
db: Session = Depends(get_db),
):
stmt = select(ThroughputProduct).where(ThroughputProduct.tenant_id == session.tenant_id)
if not include_inactive:
stmt = stmt.where(ThroughputProduct.active.is_(True))
stmt = stmt.order_by(ThroughputProduct.name)
return db.scalars(stmt).all()
@router.post("/products", response_model=ThroughputProductRead, status_code=status.HTTP_201_CREATED)
def create_product(
payload: ThroughputProductCreate,
session: AuthSession = Depends(require_client_module_access(MODULE_KEY, "edit")),
db: Session = Depends(get_db),
):
if payload.item_id:
existing = db.scalar(
select(ThroughputProduct).where(
ThroughputProduct.tenant_id == session.tenant_id,
ThroughputProduct.item_id == payload.item_id,
)
)
if existing is not None:
raise HTTPException(status_code=409, detail="A product with this item_id already exists")
product = ThroughputProduct(
tenant_id=session.tenant_id,
item_id=payload.item_id,
name=payload.name,
default_bag_size=payload.default_bag_size,
is_bulka_default=payload.is_bulka_default,
active=payload.active,
is_stock_item=payload.is_stock_item,
notes=payload.notes,
)
db.add(product)
db.commit()
db.refresh(product)
return product
@router.patch("/products/{product_id}", response_model=ThroughputProductRead)
def update_product(
product_id: int,
payload: ThroughputProductUpdate,
session: AuthSession = Depends(require_client_module_access(MODULE_KEY, "edit")),
db: Session = Depends(get_db),
):
product = db.scalar(
select(ThroughputProduct).where(
ThroughputProduct.id == product_id,
ThroughputProduct.tenant_id == session.tenant_id,
)
)
if product is None:
raise HTTPException(status_code=404, detail="Product not found")
for field, value in payload.model_dump(exclude_unset=True).items():
setattr(product, field, value)
db.commit()
db.refresh(product)
return product
@router.get("/entries", response_model=list[ThroughputEntryRead])
def list_entries(
date_from: date | None = Query(default=None),
date_to: date | None = Query(default=None),
product_id: int | None = Query(default=None),
staff_name: str | None = Query(default=None),
quantity_type: str | None = Query(default=None),
limit: int = Query(default=200, ge=1, le=1000),
session: AuthSession = Depends(require_client_module_access(MODULE_KEY)),
db: Session = Depends(get_db),
):
stmt = select(ProductionThroughput).where(ProductionThroughput.tenant_id == session.tenant_id)
if date_from is not None:
stmt = stmt.where(ProductionThroughput.production_date >= date_from)
if date_to is not None:
stmt = stmt.where(ProductionThroughput.production_date <= date_to)
if product_id is not None:
stmt = stmt.where(ProductionThroughput.product_id == product_id)
if staff_name:
stmt = stmt.where(ProductionThroughput.staff_name == staff_name.strip())
if quantity_type in {"bags", "kg"}:
stmt = stmt.where(ProductionThroughput.quantity_type == quantity_type)
stmt = stmt.order_by(ProductionThroughput.production_date.desc(), ProductionThroughput.id.desc()).limit(limit)
return [serialize_entry(entry) for entry in db.scalars(stmt).all()]
@router.post("/entries", response_model=ThroughputEntryRead, status_code=status.HTTP_201_CREATED)
def create_entry(
payload: ThroughputEntryCreate,
session: AuthSession = Depends(require_client_module_access(MODULE_KEY, "edit")),
db: Session = Depends(get_db),
):
product = None
if payload.product_id is not None:
product = db.scalar(
select(ThroughputProduct).where(
ThroughputProduct.id == payload.product_id,
ThroughputProduct.tenant_id == session.tenant_id,
)
)
if product is None:
raise HTTPException(status_code=400, detail="product_id does not match an existing product")
snapshot = payload.product_name_snapshot or (product.name if product else None)
if not snapshot:
raise HTTPException(status_code=400, detail="product_name_snapshot or product_id is required")
bag_size = payload.bag_size
if bag_size is None and product is not None:
bag_size = product.default_bag_size
if payload.quantity_type == "bags" and (bag_size is None or bag_size <= 0):
raise HTTPException(status_code=400, detail="bag_size is required when quantity_type is 'bags'")
calculated = calculate_kg(payload.quantity, payload.quantity_type, bag_size)
entry = ProductionThroughput(
tenant_id=session.tenant_id,
production_date=payload.production_date,
product_id=product.id if product else None,
product_name_snapshot=snapshot,
bag_size=bag_size,
scales_checked=payload.scales_checked,
label_correct=payload.label_correct,
bag_sealed=payload.bag_sealed,
pallet_good_condition=payload.pallet_good_condition,
sample_box_no=payload.sample_box_no,
test_weight_1=payload.test_weight_1,
test_weight_2=payload.test_weight_2,
test_weight_3=payload.test_weight_3,
test_weight_4=payload.test_weight_4,
test_weight_5=payload.test_weight_5,
quantity=payload.quantity,
quantity_type=payload.quantity_type,
calculated_kg=calculated,
staff_name=normalise_staff_name(payload.staff_name),
notes=payload.notes,
created_by=session.email,
)
db.add(entry)
db.commit()
db.refresh(entry)
return serialize_entry(entry)
@router.get("/entries/{entry_id}", response_model=ThroughputEntryRead)
def get_entry(
entry_id: int,
session: AuthSession = Depends(require_client_module_access(MODULE_KEY)),
db: Session = Depends(get_db),
):
entry = db.scalar(
select(ProductionThroughput).where(
ProductionThroughput.id == entry_id,
ProductionThroughput.tenant_id == session.tenant_id,
)
)
if entry is None:
raise HTTPException(status_code=404, detail="Entry not found")
return serialize_entry(entry)
@router.patch("/entries/{entry_id}", response_model=ThroughputEntryRead)
def update_entry(
entry_id: int,
payload: ThroughputEntryUpdate,
session: AuthSession = Depends(require_client_module_access(MODULE_KEY, "edit")),
db: Session = Depends(get_db),
):
entry = db.scalar(
select(ProductionThroughput).where(
ProductionThroughput.id == entry_id,
ProductionThroughput.tenant_id == session.tenant_id,
)
)
if entry is None:
raise HTTPException(status_code=404, detail="Entry not found")
data = payload.model_dump(exclude_unset=True)
if "staff_name" in data:
data["staff_name"] = normalise_staff_name(data["staff_name"])
for field, value in data.items():
setattr(entry, field, value)
entry.calculated_kg = calculate_kg(entry.quantity, entry.quantity_type, entry.bag_size)
db.commit()
db.refresh(entry)
return serialize_entry(entry)
@router.delete("/entries/{entry_id}", status_code=status.HTTP_204_NO_CONTENT)
def delete_entry(
entry_id: int,
session: AuthSession = Depends(require_client_module_access(MODULE_KEY, "manage")),
db: Session = Depends(get_db),
):
entry = db.scalar(
select(ProductionThroughput).where(
ProductionThroughput.id == entry_id,
ProductionThroughput.tenant_id == session.tenant_id,
)
)
if entry is None:
raise HTTPException(status_code=404, detail="Entry not found")
db.delete(entry)
db.commit()
return None
+2
View File
@@ -51,6 +51,8 @@ _PERMISSION_TO_MODULE_LEVEL: dict[str, tuple[str, str]] = {
"edit_products": ("products", "edit"), "edit_products": ("products", "edit"),
"view_mixes": ("mix_master", "view"), "view_mixes": ("mix_master", "view"),
"edit_mixes": ("mix_master", "edit"), "edit_mixes": ("mix_master", "edit"),
"view_throughput": ("operations_throughput", "view"),
"edit_throughput": ("operations_throughput", "edit"),
# Admin-only permissions (view_users, manage_users, manage_permissions, # Admin-only permissions (view_users, manage_users, manage_permissions,
# view_settings, edit_settings) are intentionally excluded — they don't # view_settings, edit_settings) are intentionally excluded — they don't
# correspond to any of the legacy module keys and remain accessible only # correspond to any of the legacy module keys and remain accessible only
+9 -1
View File
@@ -1,5 +1,10 @@
import os import os
from dataclasses import dataclass from dataclasses import dataclass
from pathlib import Path
DEFAULT_SQLITE_PATH = (Path(__file__).resolve().parents[2] / "data_entry_app.db").as_posix()
DEFAULT_DATABASE_URL = f"sqlite:///{DEFAULT_SQLITE_PATH}"
DEFAULT_CORS_ALLOW_ORIGIN_REGEX = ( DEFAULT_CORS_ALLOW_ORIGIN_REGEX = (
@@ -63,7 +68,10 @@ class Settings:
port=int(os.getenv("PORT", "8000")), port=int(os.getenv("PORT", "8000")),
log_level=os.getenv("LOG_LEVEL", "DEBUG" if os.getenv("LOG_VERBOSE") in {"1", "true", "TRUE", "yes", "on"} else "INFO"), log_level=os.getenv("LOG_LEVEL", "DEBUG" if os.getenv("LOG_VERBOSE") in {"1", "true", "TRUE", "yes", "on"} else "INFO"),
log_verbose=_env_flag("LOG_VERBOSE"), log_verbose=_env_flag("LOG_VERBOSE"),
database_url=os.getenv("DATABASE_URL", "sqlite:///./data_entry_app.db"), # Keep the default SQLite location stable regardless of the current
# working directory so local dev does not silently fork data into
# multiple `data_entry_app.db` files.
database_url=os.getenv("DATABASE_URL", DEFAULT_DATABASE_URL),
client_name=os.getenv("CLIENT_NAME", "Hunter Premium Produce"), client_name=os.getenv("CLIENT_NAME", "Hunter Premium Produce"),
client_email=os.getenv("CLIENT_EMAIL", "operator@example.com"), client_email=os.getenv("CLIENT_EMAIL", "operator@example.com"),
client_password=os.getenv("CLIENT_PASSWORD", "changeme"), client_password=os.getenv("CLIENT_PASSWORD", "changeme"),
+18
View File
@@ -25,6 +25,7 @@ TENANT_TABLES = {
"raw_material_price_versions": None, "raw_material_price_versions": None,
"mixes": None, "mixes": None,
"mix_ingredients": None, "mix_ingredients": None,
"product_ingredients": None,
"mix_calculator_sessions": None, "mix_calculator_sessions": None,
"mix_calculator_session_lines": None, "mix_calculator_session_lines": None,
"products": None, "products": None,
@@ -33,6 +34,8 @@ TENANT_TABLES = {
"process_cost_rules": None, "process_cost_rules": None,
"packaging_cost_rules": None, "packaging_cost_rules": None,
"freight_cost_rules": None, "freight_cost_rules": None,
"throughput_products": None,
"production_throughput_entries": None,
} }
@@ -99,6 +102,7 @@ def ensure_tenant_columns(engine: Engine) -> tuple[str, ...]:
_LEGACY_COLUMN_PATCHES: tuple[tuple[str, str, str], ...] = ( _LEGACY_COLUMN_PATCHES: tuple[tuple[str, str, str], ...] = (
("users", "password_hash", "VARCHAR(255)"), ("users", "password_hash", "VARCHAR(255)"),
("products", "visible", "BOOLEAN NOT NULL DEFAULT TRUE"), ("products", "visible", "BOOLEAN NOT NULL DEFAULT TRUE"),
("throughput_products", "is_stock_item", "BOOLEAN NOT NULL DEFAULT TRUE"),
) )
@@ -243,6 +247,20 @@ def sync_tenant_ids(engine: Engine) -> dict[str, int]:
""" """
), ),
), ),
(
"product_ingredients",
text(
"""
UPDATE product_ingredients
SET tenant_id = (
SELECT products.tenant_id
FROM products
WHERE products.id = product_ingredients.product_id
)
WHERE tenant_id IS NULL OR tenant_id = '' OR tenant_id = 'default'
"""
),
),
( (
"products", "products",
text( text(
+8 -2
View File
@@ -28,6 +28,7 @@ from app.api.powerbi import router as powerbi_router
from app.api.products import router as products_router from app.api.products import router as products_router
from app.api.raw_materials import router as raw_materials_router from app.api.raw_materials import router as raw_materials_router
from app.api.scenarios import router as scenarios_router from app.api.scenarios import router as scenarios_router
from app.api.throughput import router as throughput_router
from app.core.config import settings from app.core.config import settings
from app.core.logging import ( from app.core.logging import (
LoggingSettings, LoggingSettings,
@@ -46,7 +47,7 @@ from app.core.logging import (
) )
from app.db.session import Base, engine from app.db.session import Base, engine
from app.db.migrations import MigrationReport, bootstrap_schema, sync_product_visibility, sync_tenant_ids from app.db.migrations import MigrationReport, bootstrap_schema, sync_product_visibility, sync_tenant_ids
from app.seed import seed_if_empty from app.seed import seed_startup_basics
def _resolve_version() -> str: def _resolve_version() -> str:
@@ -100,7 +101,7 @@ def ensure_database_ready() -> MigrationReport:
return MigrationReport() return MigrationReport()
schema_report = bootstrap_schema(engine, Base.metadata) schema_report = bootstrap_schema(engine, Base.metadata)
seed_if_empty() seed_startup_basics()
tenant_sync_report = sync_tenant_ids(engine) tenant_sync_report = sync_tenant_ids(engine)
hidden_product_count = sync_product_visibility(engine) hidden_product_count = sync_product_visibility(engine)
@@ -153,7 +154,10 @@ async def lifespan(app: FastAPI):
section_heading("Services") section_heading("Services")
success("HTTP API ready") success("HTTP API ready")
if settings.docs_enabled:
info("Docs available at /docs", logger_name="data_entry_app.services") info("Docs available at /docs", logger_name="data_entry_app.services")
else:
info("Docs disabled in this environment", logger_name="data_entry_app.services")
info("Health probe available at /health", logger_name="data_entry_app.services") info("Health probe available at /health", logger_name="data_entry_app.services")
yield yield
@@ -195,6 +199,7 @@ app.include_router(mixes_router)
app.include_router(mix_calculator_router) app.include_router(mix_calculator_router)
app.include_router(products_router) app.include_router(products_router)
app.include_router(scenarios_router) app.include_router(scenarios_router)
app.include_router(throughput_router)
app.include_router(powerbi_router) app.include_router(powerbi_router)
@@ -300,6 +305,7 @@ def root():
"mix_calculator": "/api/mix-calculator", "mix_calculator": "/api/mix-calculator",
"products": "/api/products", "products": "/api/products",
"scenarios": "/api/scenarios", "scenarios": "/api/scenarios",
"operations_throughput": "/api/throughput",
"client_access": "/api/client-access", "client_access": "/api/client-access",
"docs": "/docs", "docs": "/docs",
}, },
+5 -1
View File
@@ -3,9 +3,10 @@ from app.models.assumption import FreightCostRule, PackagingCostRule, ProcessCos
from app.models.client_access import ClientAccessAuditEvent, ClientAccount, ClientFeatureAccess, ClientUser, ClientUserModulePermission from app.models.client_access import ClientAccessAuditEvent, ClientAccount, ClientFeatureAccess, ClientUser, ClientUserModulePermission
from app.models.mix_calculator import MixCalculatorSession, MixCalculatorSessionLine from app.models.mix_calculator import MixCalculatorSession, MixCalculatorSessionLine
from app.models.mix import Mix, MixIngredient from app.models.mix import Mix, MixIngredient
from app.models.product import Product from app.models.product import Product, ProductIngredient
from app.models.raw_material import RawMaterial, RawMaterialPriceVersion from app.models.raw_material import RawMaterial, RawMaterialPriceVersion
from app.models.scenario import CostingResult, Scenario from app.models.scenario import CostingResult, Scenario
from app.models.throughput import ProductionThroughput, ThroughputProduct
__all__ = [ __all__ = [
"ClientAccount", "ClientAccount",
@@ -23,6 +24,9 @@ __all__ = [
"Permission", "Permission",
"ProcessCostRule", "ProcessCostRule",
"Product", "Product",
"ProductIngredient",
"ProductionThroughput",
"ThroughputProduct",
"RawMaterial", "RawMaterial",
"RawMaterialPriceVersion", "RawMaterialPriceVersion",
"Role", "Role",
+23 -1
View File
@@ -2,7 +2,7 @@ from __future__ import annotations
from datetime import datetime from datetime import datetime
from sqlalchemy import Boolean, DateTime, Float, ForeignKey, Integer, String, Text from sqlalchemy import Boolean, DateTime, Float, ForeignKey, Integer, String, Text, UniqueConstraint
from sqlalchemy.orm import Mapped, mapped_column, relationship from sqlalchemy.orm import Mapped, mapped_column, relationship
from app.db.session import Base from app.db.session import Base
@@ -29,6 +29,28 @@ class Product(Base):
created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow)
mix: Mapped["Mix"] = relationship(back_populates="products") mix: Mapped["Mix"] = relationship(back_populates="products")
ingredients: Mapped[list["ProductIngredient"]] = relationship(
back_populates="product",
cascade="all, delete-orphan",
order_by="ProductIngredient.sort_order",
)
class ProductIngredient(Base):
__tablename__ = "product_ingredients"
__table_args__ = (UniqueConstraint("product_id", "raw_material_id", name="uq_product_ingredient"),)
id: Mapped[int] = mapped_column(primary_key=True)
tenant_id: Mapped[str] = mapped_column(String(64), default="default", index=True)
product_id: Mapped[int] = mapped_column(ForeignKey("products.id"), index=True)
raw_material_id: Mapped[int] = mapped_column(ForeignKey("raw_materials.id"), index=True)
quantity_kg: Mapped[float] = mapped_column(Float)
sort_order: Mapped[int] = mapped_column(Integer, default=0)
notes: Mapped[str | None] = mapped_column(Text, nullable=True)
product: Mapped[Product] = relationship(back_populates="ingredients")
raw_material: Mapped["RawMaterial"] = relationship()
from app.models.mix import Mix # noqa: E402 from app.models.mix import Mix # noqa: E402
from app.models.raw_material import RawMaterial # noqa: E402
+70
View File
@@ -0,0 +1,70 @@
from __future__ import annotations
from datetime import date, datetime
from sqlalchemy import Boolean, Date, DateTime, Float, ForeignKey, Index, Integer, String, Text
from sqlalchemy.orm import Mapped, mapped_column, relationship
from app.db.session import Base
class ThroughputProduct(Base):
__tablename__ = "throughput_products"
__table_args__ = (
Index("ix_throughput_products_tenant_item", "tenant_id", "item_id", unique=True),
Index("ix_throughput_products_tenant_name", "tenant_id", "name"),
)
id: Mapped[int] = mapped_column(primary_key=True)
tenant_id: Mapped[str] = mapped_column(String(64), default="default", index=True)
item_id: Mapped[str | None] = mapped_column(String(64), nullable=True)
name: Mapped[str] = mapped_column(String(255))
default_bag_size: Mapped[float | None] = mapped_column(Float, nullable=True)
is_bulka_default: Mapped[bool] = mapped_column(Boolean, default=False)
active: Mapped[bool] = mapped_column(Boolean, default=True)
is_stock_item: Mapped[bool] = mapped_column(Boolean, default=True)
notes: Mapped[str | None] = mapped_column(Text, nullable=True)
created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow)
updated_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)
entries: Mapped[list["ProductionThroughput"]] = relationship(back_populates="product")
class ProductionThroughput(Base):
__tablename__ = "production_throughput_entries"
__table_args__ = (
Index("ix_throughput_entries_tenant_date", "tenant_id", "production_date"),
)
id: Mapped[int] = mapped_column(primary_key=True)
tenant_id: Mapped[str] = mapped_column(String(64), default="default", index=True)
production_date: Mapped[date] = mapped_column(Date)
product_id: Mapped[int | None] = mapped_column(ForeignKey("throughput_products.id"), nullable=True, index=True)
product_name_snapshot: Mapped[str] = mapped_column(String(255))
bag_size: Mapped[float | None] = mapped_column(Float, nullable=True)
scales_checked: Mapped[bool] = mapped_column(Boolean, default=True)
label_correct: Mapped[bool] = mapped_column(Boolean, default=True)
bag_sealed: Mapped[bool] = mapped_column(Boolean, default=True)
pallet_good_condition: Mapped[bool] = mapped_column(Boolean, default=True)
sample_box_no: Mapped[str | None] = mapped_column(String(64), nullable=True)
test_weight_1: Mapped[float | None] = mapped_column(Float, nullable=True)
test_weight_2: Mapped[float | None] = mapped_column(Float, nullable=True)
test_weight_3: Mapped[float | None] = mapped_column(Float, nullable=True)
test_weight_4: Mapped[float | None] = mapped_column(Float, nullable=True)
test_weight_5: Mapped[float | None] = mapped_column(Float, nullable=True)
quantity: Mapped[float] = mapped_column(Float, default=0.0)
quantity_type: Mapped[str] = mapped_column(String(8), default="bags")
calculated_kg: Mapped[float] = mapped_column(Float, default=0.0)
staff_name: Mapped[str | None] = mapped_column(String(255), nullable=True)
notes: Mapped[str | None] = mapped_column(Text, nullable=True)
created_by: Mapped[str | None] = mapped_column(String(255), nullable=True)
created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow)
updated_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)
product: Mapped[ThroughputProduct | None] = relationship(back_populates="entries")
+128
View File
@@ -0,0 +1,128 @@
from __future__ import annotations
from datetime import date, datetime
from typing import Literal
from pydantic import BaseModel, ConfigDict, Field, field_validator
QuantityType = Literal["bags", "kg"]
class ThroughputProductBase(BaseModel):
model_config = ConfigDict(extra="forbid")
item_id: str | None = Field(default=None, max_length=64)
name: str = Field(min_length=1, max_length=255)
default_bag_size: float | None = Field(default=None, ge=0)
is_bulka_default: bool = False
active: bool = True
is_stock_item: bool = True
notes: str | None = Field(default=None, max_length=2000)
class ThroughputProductCreate(ThroughputProductBase):
pass
class ThroughputProductUpdate(BaseModel):
model_config = ConfigDict(extra="forbid")
item_id: str | None = Field(default=None, max_length=64)
name: str | None = Field(default=None, min_length=1, max_length=255)
default_bag_size: float | None = Field(default=None, ge=0)
is_bulka_default: bool | None = None
active: bool | None = None
is_stock_item: bool | None = None
notes: str | None = Field(default=None, max_length=2000)
class ThroughputProductRead(ThroughputProductBase):
id: int
tenant_id: str
created_at: datetime
updated_at: datetime
model_config = ConfigDict(from_attributes=True)
class ThroughputEntryBase(BaseModel):
model_config = ConfigDict(extra="forbid")
production_date: date
product_id: int | None = None
product_name_snapshot: str | None = Field(default=None, max_length=255)
bag_size: float | None = Field(default=None, ge=0)
scales_checked: bool = True
label_correct: bool = True
bag_sealed: bool = True
pallet_good_condition: bool = True
sample_box_no: str | None = Field(default=None, max_length=64)
test_weight_1: float | None = Field(default=None, ge=0)
test_weight_2: float | None = Field(default=None, ge=0)
test_weight_3: float | None = Field(default=None, ge=0)
test_weight_4: float | None = Field(default=None, ge=0)
test_weight_5: float | None = Field(default=None, ge=0)
quantity: float = Field(ge=0)
quantity_type: QuantityType = "bags"
staff_name: str | None = Field(default=None, max_length=255)
notes: str | None = Field(default=None, max_length=2000)
@field_validator("staff_name")
@classmethod
def _normalize_staff(cls, value: str | None) -> str | None:
if value is None:
return None
stripped = value.strip()
return stripped or None
class ThroughputEntryCreate(ThroughputEntryBase):
pass
class ThroughputEntryUpdate(BaseModel):
model_config = ConfigDict(extra="forbid")
production_date: date | None = None
product_id: int | None = None
product_name_snapshot: str | None = Field(default=None, max_length=255)
bag_size: float | None = Field(default=None, ge=0)
scales_checked: bool | None = None
label_correct: bool | None = None
bag_sealed: bool | None = None
pallet_good_condition: bool | None = None
sample_box_no: str | None = Field(default=None, max_length=64)
test_weight_1: float | None = Field(default=None, ge=0)
test_weight_2: float | None = Field(default=None, ge=0)
test_weight_3: float | None = Field(default=None, ge=0)
test_weight_4: float | None = Field(default=None, ge=0)
test_weight_5: float | None = Field(default=None, ge=0)
quantity: float | None = Field(default=None, ge=0)
quantity_type: QuantityType | None = None
staff_name: str | None = Field(default=None, max_length=255)
notes: str | None = Field(default=None, max_length=2000)
class ThroughputEntryRead(BaseModel):
id: int
tenant_id: str
production_date: date
product_id: int | None
product_name_snapshot: str
bag_size: float | None
scales_checked: bool
label_correct: bool
bag_sealed: bool
pallet_good_condition: bool
sample_box_no: str | None
test_weight_1: float | None
test_weight_2: float | None
test_weight_3: float | None
test_weight_4: float | None
test_weight_5: float | None
quantity: float
quantity_type: QuantityType
calculated_kg: float
staff_name: str | None
notes: str | None
qa_passed: bool
created_by: str | None
created_at: datetime
updated_at: datetime
model_config = ConfigDict(from_attributes=True)
+318 -13
View File
@@ -9,21 +9,26 @@ import re
from openpyxl import load_workbook from openpyxl import load_workbook
from sqlalchemy import select from sqlalchemy import select
from sqlalchemy.orm import selectinload
from app.db.session import Base, SessionLocal, engine from app.db.session import Base, SessionLocal, engine
from app.models.assumption import FreightCostRule, PackagingCostRule, ProcessCostRule from app.models.assumption import FreightCostRule, PackagingCostRule, ProcessCostRule
from app.models.client_access import ClientAccessAuditEvent, ClientAccount, ClientFeatureAccess, ClientUser, ClientUserModulePermission from app.models.client_access import ClientAccessAuditEvent, ClientAccount, ClientFeatureAccess, ClientUser, ClientUserModulePermission
from app.models.mix import Mix, MixIngredient from app.models.mix import Mix, MixIngredient
from app.models.product import Product from app.models.product import Product, ProductIngredient
from app.models.raw_material import RawMaterial, RawMaterialPriceVersion from app.models.raw_material import RawMaterial, RawMaterialPriceVersion
from app.models.throughput import ProductionThroughput, ThroughputProduct
from app.seed_access import seed_access from app.seed_access import seed_access
from app.services.client_access_service import MODULE_CATALOG, default_access_level_for_role from app.services.client_access_service import MODULE_CATALOG, default_access_level_for_role
from app.services.throughput_service import import_workbook as import_throughput_workbook
from app.services.throughput_service import resolve_workbook_path as resolve_throughput_workbook_path
TENANT_ID = "hunter-premium-produce" TENANT_ID = "hunter-premium-produce"
WORKBOOK_EFFECTIVE_DATE = date(2025, 9, 1) WORKBOOK_EFFECTIVE_DATE = date(2025, 9, 1)
WORKBOOK_SENTINEL_ITEM_ID = "404266" WORKBOOK_SENTINEL_ITEM_ID = "404266"
WORKBOOK_FILENAME = "Input Cost Spreadsheet(1).xlsx" WORKBOOK_FILENAME = "1.xlsx"
LEGACY_WORKBOOK_FILENAME = "Input Cost Spreadsheet(1).xlsx"
logger = logging.getLogger("data_entry_app.seed") logger = logging.getLogger("data_entry_app.seed")
HIDDEN_PRODUCT_CLIENTS = frozenset( HIDDEN_PRODUCT_CLIENTS = frozenset(
{ {
@@ -46,11 +51,18 @@ def _workbook_candidates() -> list[Path]:
candidates = [ candidates = [
Path(env_path) if env_path else None, Path(env_path) if env_path else None,
repo_root / "input_data" / WORKBOOK_FILENAME,
cwd / "input_data" / WORKBOOK_FILENAME,
Path("/srv/lean101-clients") / WORKBOOK_FILENAME, Path("/srv/lean101-clients") / WORKBOOK_FILENAME,
repo_root / WORKBOOK_FILENAME, repo_root / WORKBOOK_FILENAME,
cwd / WORKBOOK_FILENAME, cwd / WORKBOOK_FILENAME,
Path("/app") / WORKBOOK_FILENAME, Path("/app") / WORKBOOK_FILENAME,
Path("/") / WORKBOOK_FILENAME, Path("/") / WORKBOOK_FILENAME,
repo_root / LEGACY_WORKBOOK_FILENAME,
cwd / LEGACY_WORKBOOK_FILENAME,
Path("/srv/lean101-clients") / LEGACY_WORKBOOK_FILENAME,
Path("/app") / LEGACY_WORKBOOK_FILENAME,
Path("/") / LEGACY_WORKBOOK_FILENAME,
] ]
ordered: list[Path] = [] ordered: list[Path] = []
@@ -73,9 +85,6 @@ def _resolve_workbook_path() -> Path:
return _workbook_candidates()[0] return _workbook_candidates()[0]
WORKBOOK_PATH = _resolve_workbook_path()
def _text(value) -> str | None: def _text(value) -> str | None:
if value is None: if value is None:
return None return None
@@ -178,7 +187,21 @@ def _build_process_key(label, grading_cost: float, bagging_cost: float, cracking
return f"{base}_g{int(round(grading_cost * 1000))}_b{int(round(bagging_cost * 1000))}_c{int(round(cracking_cost * 1000))}" return f"{base}_g{int(round(grading_cost * 1000))}_b{int(round(bagging_cost * 1000))}_c{int(round(cracking_cost * 1000))}"
def _load_workbook(): def _load_workbook(*required_sheets: str):
for candidate in _workbook_candidates():
if not candidate.exists():
continue
workbook = load_workbook(candidate, data_only=True)
if all(sheet_name in workbook.sheetnames for sheet_name in required_sheets):
return workbook
if required_sheets:
raise FileNotFoundError(
"No workbook with required sheets found. "
f"Required sheets: {', '.join(required_sheets)}. "
f"Checked: {', '.join(str(path) for path in _workbook_candidates())}"
)
workbook_path = _resolve_workbook_path() workbook_path = _resolve_workbook_path()
if not workbook_path.exists(): if not workbook_path.exists():
raise FileNotFoundError( raise FileNotFoundError(
@@ -258,6 +281,44 @@ def _read_mix_rows(workbook) -> dict[tuple[str, str], dict]:
return best_rows return best_rows
def _read_product_ingredient_rows(workbook) -> dict[tuple[str, str], dict]:
worksheet = workbook["mix_quantites_per_client_per_pr"]
header_row = next(worksheet.iter_rows(min_row=1, max_row=1, values_only=True))
ingredient_names = [_text(value) for value in header_row[3:] if _text(value)]
rows: dict[tuple[str, str], dict] = {}
for row in worksheet.iter_rows(min_row=2, values_only=True):
client_name = _text(row[0])
product_name = _text(row[1])
if not client_name or not product_name:
continue
ingredients = []
for sort_order, (ingredient_name, quantity) in enumerate(zip(ingredient_names, row[3 : 3 + len(ingredient_names)]), start=1):
numeric_quantity = _number(quantity)
if ingredient_name and numeric_quantity and numeric_quantity > 0:
ingredients.append(
{
"raw_material_name": ingredient_name,
"quantity_kg": numeric_quantity,
"sort_order": sort_order,
}
)
if not ingredients:
continue
total_kg = _number(row[2]) or round(sum(item["quantity_kg"] for item in ingredients), 4)
rows[(client_name, product_name)] = {
"client_name": client_name,
"product_name": product_name,
"total_kg": total_kg,
"ingredients": ingredients,
}
return rows
def _read_product_rows(workbook) -> list[dict]: def _read_product_rows(workbook) -> list[dict]:
worksheet = workbook["Product Cost - Price"] worksheet = workbook["Product Cost - Price"]
raw_rows: list[dict] = [] raw_rows: list[dict] = []
@@ -606,6 +667,184 @@ def _upsert_products(db, products: list[dict], mix_lookup: dict[tuple[str, str],
product.notes = "Seeded from Input Cost Spreadsheet(1).xlsx" product.notes = "Seeded from Input Cost Spreadsheet(1).xlsx"
def _upsert_product_ingredients(
db,
*,
product_rows: list[dict],
product_ingredient_rows: dict[tuple[str, str], dict],
raw_material_map: dict[str, RawMaterial],
) -> None:
products = db.scalars(
select(Product).where(Product.tenant_id == TENANT_ID).options(selectinload(Product.mix))
).all()
products_by_formula_key: dict[tuple[str, str], list[Product]] = {}
for product in products:
candidate_keys = {
(product.client_name, product.name),
}
if product.mix is not None:
candidate_keys.add((product.client_name, product.mix.name))
for key in candidate_keys:
products_by_formula_key.setdefault(key, []).append(product)
for key, formula in product_ingredient_rows.items():
matched_products = products_by_formula_key.get(key, [])
if not matched_products:
continue
for product in matched_products:
existing_ingredients = {
ingredient.raw_material_id: ingredient
for ingredient in db.scalars(select(ProductIngredient).where(ProductIngredient.product_id == product.id)).all()
}
desired_ids: set[int] = set()
for row in formula["ingredients"]:
raw_material = raw_material_map.get(row["raw_material_name"])
if raw_material is None:
continue
desired_ids.add(raw_material.id)
ingredient = existing_ingredients.get(raw_material.id)
if ingredient is None:
db.add(
ProductIngredient(
tenant_id=TENANT_ID,
product_id=product.id,
raw_material_id=raw_material.id,
quantity_kg=row["quantity_kg"],
sort_order=row["sort_order"],
)
)
else:
ingredient.quantity_kg = row["quantity_kg"]
ingredient.sort_order = row["sort_order"]
for raw_material_id, ingredient in existing_ingredients.items():
if raw_material_id not in desired_ids:
db.delete(ingredient)
def _infer_throughput_bag_size(product: Product) -> float | None:
if product.sale_type == "bulka":
return None
unit = (product.unit_of_measure or "").strip().lower()
match = re.search(r"(\d+(?:\.\d+)?)\s*kg", unit)
if match:
return float(match.group(1))
if unit == "kg":
return 1.0
if unit == "tonne":
return 1000.0
return None
def _infer_throughput_bulka_default(product: Product) -> bool:
unit = (product.unit_of_measure or "").lower()
return product.sale_type == "bulka" or "bulka" in product.name.lower() or "bulka" in unit
def seed_throughput_products_from_costing(db) -> dict[str, int]:
"""Mirror costing products into the throughput product dropdown."""
costing_products = db.scalars(
select(Product)
.where(Product.tenant_id == TENANT_ID)
.order_by(Product.name, Product.id)
).all()
if not costing_products:
return {"created": 0, "updated": 0, "skipped": 0}
throughput_products = db.scalars(
select(ThroughputProduct).where(ThroughputProduct.tenant_id == TENANT_ID)
).all()
by_item = {
throughput_product.item_id: throughput_product
for throughput_product in throughput_products
if throughput_product.item_id
}
by_name = {
throughput_product.name.strip().lower(): throughput_product
for throughput_product in throughput_products
if throughput_product.name
}
created = 0
updated = 0
skipped = 0
seen_item_ids: set[str] = set()
seen_names: set[str] = set()
for costing_product in costing_products:
name = (costing_product.name or "").strip()
if not name:
skipped += 1
continue
item_id = (costing_product.item_id or "").strip() or None
name_key = name.lower()
if item_id and item_id in seen_item_ids:
skipped += 1
continue
if not item_id and name_key in seen_names:
skipped += 1
continue
if item_id:
seen_item_ids.add(item_id)
seen_names.add(name_key)
default_bag_size = _infer_throughput_bag_size(costing_product)
is_bulka_default = _infer_throughput_bulka_default(costing_product)
product = (by_item.get(item_id) if item_id else None) or by_name.get(name_key)
if product is None:
product = ThroughputProduct(
tenant_id=TENANT_ID,
item_id=item_id,
name=name,
default_bag_size=default_bag_size,
is_bulka_default=is_bulka_default,
active=costing_product.visible,
is_stock_item=True,
notes="Seeded from costing products",
)
db.add(product)
created += 1
if item_id:
by_item[item_id] = product
by_name[name_key] = product
continue
changed = False
if item_id and product.item_id != item_id:
product.item_id = item_id
changed = True
if product.name != name:
old_name_key = product.name.strip().lower() if product.name else None
product.name = name
if old_name_key:
by_name.pop(old_name_key, None)
by_name[name_key] = product
changed = True
if product.default_bag_size != default_bag_size:
product.default_bag_size = default_bag_size
changed = True
if product.is_bulka_default != is_bulka_default:
product.is_bulka_default = is_bulka_default
changed = True
if product.active != costing_product.visible:
product.active = costing_product.visible
changed = True
if product.is_stock_item is not True:
product.is_stock_item = True
changed = True
if product.notes in {None, "", "Seeded from costing products"}:
product.notes = "Seeded from costing products"
if changed:
updated += 1
db.flush()
return {"created": created, "updated": updated, "skipped": skipped}
def seed_client_access(db): def seed_client_access(db):
existing = db.scalar(select(ClientAccount.id)) existing = db.scalar(select(ClientAccount.id))
if existing is not None: if existing is not None:
@@ -667,7 +906,7 @@ def seed_client_access(db):
) )
enabled_feature_map = { enabled_feature_map = {
TENANT_ID: {"dashboard", "raw_materials", "mix_master", "mix_calculator", "products", "scenarios", "powerbi_export", "client_access"}, TENANT_ID: {"dashboard", "raw_materials", "mix_master", "mix_calculator", "products", "scenarios", "powerbi_export", "client_access", "operations_throughput"},
"loft-grains": {"dashboard", "mix_calculator", "products", "powerbi_export"}, "loft-grains": {"dashboard", "mix_calculator", "products", "powerbi_export"},
} }
@@ -713,10 +952,13 @@ def seed_client_access(db):
def seed_costing_workspace(db): def seed_costing_workspace(db):
workbook = _load_workbook() costing_workbook = _load_workbook("C- Raw Products Costs", "M - All", "Product Cost - Price")
raw_material_rows = _read_raw_material_rows(workbook) formula_workbook = _load_workbook("mix_quantites_per_client_per_pr")
mix_rows = _read_mix_rows(workbook)
product_rows = _read_product_rows(workbook) raw_material_rows = _read_raw_material_rows(costing_workbook)
mix_rows = _read_mix_rows(costing_workbook)
product_rows = _read_product_rows(costing_workbook)
product_ingredient_rows = _read_product_ingredient_rows(formula_workbook)
raw_material_map = _upsert_raw_materials(db, raw_material_rows) raw_material_map = _upsert_raw_materials(db, raw_material_rows)
_upsert_process_rules(db, product_rows) _upsert_process_rules(db, product_rows)
@@ -735,9 +977,53 @@ def seed_costing_workspace(db):
mix_cache[(mix_row["client_name"], mix_row["name"])] = mix mix_cache[(mix_row["client_name"], mix_row["name"])] = mix
_upsert_products(db, product_rows, mix_cache, raw_material_map) _upsert_products(db, product_rows, mix_cache, raw_material_map)
_upsert_product_ingredients(
db,
product_rows=product_rows,
product_ingredient_rows=product_ingredient_rows,
raw_material_map=raw_material_map,
)
def seed_if_empty(): def seed_throughput_workbook(db):
"""Import the Operations Throughput workbook on first run if tables are empty."""
has_products = db.scalar(select(ThroughputProduct.id)) is not None
has_entries = db.scalar(select(ProductionThroughput.id)) is not None
if not has_products and not has_entries:
workbook_path = resolve_throughput_workbook_path()
if workbook_path is None:
logger.info("Operations Throughput workbook not found; seeding throughput products from costing products")
else:
try:
report = import_throughput_workbook(db, workbook_path, TENANT_ID)
except Exception:
logger.exception("Failed to seed Operations Throughput workbook from %s", workbook_path)
else:
logger.info("Operations Throughput seeded from %s: %s", workbook_path, report)
report = seed_throughput_products_from_costing(db)
if any(report.values()):
logger.info("Throughput products synced from costing products: %s", report)
def seed_throughput_products(db):
"""Sync throughput products from costing products without importing historical entries."""
report = seed_throughput_products_from_costing(db)
if any(report.values()):
logger.info("Throughput products synced from costing products: %s", report)
return
def seed_startup_basics():
Base.metadata.create_all(bind=engine)
with SessionLocal() as db:
seed_client_access(db)
seed_access(db)
seed_throughput_workbook(db)
db.commit()
def seed_all():
Base.metadata.create_all(bind=engine) Base.metadata.create_all(bind=engine)
with SessionLocal() as db: with SessionLocal() as db:
workbook_path = _resolve_workbook_path() workbook_path = _resolve_workbook_path()
@@ -748,10 +1034,29 @@ def seed_if_empty():
"Skipping costing workspace seed because workbook is missing. Checked: %s", "Skipping costing workspace seed because workbook is missing. Checked: %s",
", ".join(str(path) for path in _workbook_candidates()), ", ".join(str(path) for path in _workbook_candidates()),
) )
seed_throughput_products(db)
seed_client_access(db)
seed_access(db)
db.commit()
def seed_if_empty():
Base.metadata.create_all(bind=engine)
with SessionLocal() as db:
if db.scalar(select(RawMaterial.id)) is None:
workbook_path = _resolve_workbook_path()
if workbook_path.exists():
seed_costing_workspace(db)
else:
logger.warning(
"Skipping costing workspace seed because workbook is missing. Checked: %s",
", ".join(str(path) for path in _workbook_candidates()),
)
seed_throughput_products(db)
seed_client_access(db) seed_client_access(db)
seed_access(db) seed_access(db)
db.commit() db.commit()
if __name__ == "__main__": if __name__ == "__main__":
seed_if_empty() seed_all()
+27 -2
View File
@@ -3,13 +3,18 @@
Re-running this is safe: it upserts permissions, syncs each role's permission Re-running this is safe: it upserts permissions, syncs each role's permission
set to the declared list, and creates or updates the seed users without set to the declared list, and creates or updates the seed users without
duplicating rows. Permission grants are the source of truth change them duplicating rows. Permission grants are the source of truth change them
here (or in the DB) rather than in route code. here (or in the DB) rather than in route code. Existing password hashes are
left intact; only users with no password hash get the current default
``ADMIN_PASSWORD`` hashed into the row.
""" """
from __future__ import annotations from __future__ import annotations
from sqlalchemy import select from sqlalchemy import select
from sqlalchemy.orm import Session, selectinload from sqlalchemy.orm import Session, selectinload
from app.core.config import settings
from app.core.security import hash_password
from app.db.session import SessionLocal
from app.models.access import Permission, Role, User from app.models.access import Permission, Role, User
@@ -24,6 +29,8 @@ PERMISSION_DEFINITIONS: tuple[tuple[str, str], ...] = (
("edit_products", "Create and edit finished products"), ("edit_products", "Create and edit finished products"),
("view_mixes", "View mix master recipes"), ("view_mixes", "View mix master recipes"),
("edit_mixes", "Create and edit mix master recipes"), ("edit_mixes", "Create and edit mix master recipes"),
("view_throughput", "View operations throughput"),
("edit_throughput", "Create and edit operations throughput entries"),
("view_users", "View internal users and roles"), ("view_users", "View internal users and roles"),
("manage_users", "Create, deactivate, and assign user roles"), ("manage_users", "Create, deactivate, and assign user roles"),
("manage_permissions", "Modify roles and role-permission assignments"), ("manage_permissions", "Modify roles and role-permission assignments"),
@@ -44,6 +51,8 @@ ROLE_DEFINITIONS: dict[str, dict] = {
"edit_raw_materials", "edit_raw_materials",
"view_products", "view_products",
"view_mixes", "view_mixes",
"view_throughput",
"edit_throughput",
"view_users", "view_users",
"manage_users", "manage_users",
"manage_permissions", "manage_permissions",
@@ -52,11 +61,13 @@ ROLE_DEFINITIONS: dict[str, dict] = {
], ],
}, },
"Operations": { "Operations": {
"description": "Mix calculator only — cannot edit raw materials, products, mixes, users, or settings.", "description": "Mix calculator and operations throughput — cannot edit raw materials, products, mixes, users, or settings.",
"permissions": [ "permissions": [
"view_mix_calculator", "view_mix_calculator",
"use_mix_calculator", "use_mix_calculator",
"save_mix_calculator_session", "save_mix_calculator_session",
"view_throughput",
"edit_throughput",
], ],
}, },
"Full Access": { "Full Access": {
@@ -72,6 +83,8 @@ ROLE_DEFINITIONS: dict[str, dict] = {
"edit_products", "edit_products",
"view_mixes", "view_mixes",
"edit_mixes", "edit_mixes",
"view_throughput",
"edit_throughput",
], ],
}, },
} }
@@ -154,6 +167,8 @@ def _upsert_users(db: Session, roles_by_name: dict[str, Role]) -> None:
user.role_id = role.id user.role_id = role.id
if not user.is_active: if not user.is_active:
user.is_active = True user.is_active = True
if user.password_hash is None:
user.password_hash = hash_password(settings.admin_password)
db.flush() db.flush()
@@ -162,3 +177,13 @@ def seed_access(db: Session) -> None:
permissions_by_key = _upsert_permissions(db) permissions_by_key = _upsert_permissions(db)
roles_by_name = _upsert_roles(db, permissions_by_key) roles_by_name = _upsert_roles(db, permissions_by_key)
_upsert_users(db, roles_by_name) _upsert_users(db, roles_by_name)
def seed_access_from_session() -> None:
with SessionLocal() as db:
seed_access(db)
db.commit()
if __name__ == "__main__":
seed_access_from_session()
@@ -20,6 +20,7 @@ MODULE_CATALOG = (
("mix_calculator", "Mix Calculator", "production", "Create and review client-specific mix calculation sessions"), ("mix_calculator", "Mix Calculator", "production", "Create and review client-specific mix calculation sessions"),
("products", "Products", "pricing", "Review finished product pricing"), ("products", "Products", "pricing", "Review finished product pricing"),
("scenarios", "Scenarios", "planning", "Run scenario overrides and comparisons"), ("scenarios", "Scenarios", "planning", "Run scenario overrides and comparisons"),
("operations_throughput", "Operations Throughput", "production", "Log production throughput and QA checks for grain/feed packing"),
("powerbi_export", "Power BI Export", "reporting", "Expose client access data to BI consumers"), ("powerbi_export", "Power BI Export", "reporting", "Expose client access data to BI consumers"),
("client_access", "Client Access", "administration", "Manage user access, module permissions, and audit history"), ("client_access", "Client Access", "administration", "Manage user access, module permissions, and audit history"),
) )
@@ -78,15 +79,15 @@ def has_access_level(access_level: str | None, minimum_level: str) -> bool:
def default_access_level_for_role(role: str, module_key: str) -> str: def default_access_level_for_role(role: str, module_key: str) -> str:
normalized = role.strip().lower() normalized = role.strip().lower()
if normalized == "superadmin": if normalized == "superadmin":
return "manage" if module_key in {"client_access", "mix_calculator"} else "edit" return "manage" if module_key in {"client_access", "mix_calculator", "operations_throughput"} else "edit"
if normalized == "admin": if normalized == "admin":
if module_key == "mix_calculator": if module_key in {"mix_calculator", "operations_throughput"}:
return "manage" return "manage"
return "edit" if module_key != "client_access" else "none" return "edit" if module_key != "client_access" else "none"
if normalized == "operator": if normalized == "operator":
return "edit" if module_key in {"dashboard", "raw_materials", "mix_master", "mix_calculator", "products", "scenarios"} else "none" return "edit" if module_key in {"dashboard", "raw_materials", "mix_master", "mix_calculator", "products", "scenarios", "operations_throughput"} else "none"
if normalized == "viewer": if normalized == "viewer":
return "view" if module_key in {"dashboard", "mix_calculator", "products", "powerbi_export"} else "none" return "view" if module_key in {"dashboard", "mix_calculator", "products", "powerbi_export", "operations_throughput"} else "none"
return "none" return "none"
+84 -2
View File
@@ -8,7 +8,7 @@ from sqlalchemy.orm import Session, selectinload
from app.models.assumption import FreightCostRule, PackagingCostRule, ProcessCostRule from app.models.assumption import FreightCostRule, PackagingCostRule, ProcessCostRule
from app.models.mix import Mix, MixIngredient from app.models.mix import Mix, MixIngredient
from app.models.product import Product from app.models.product import Product, ProductIngredient
from app.models.raw_material import RawMaterial, RawMaterialPriceVersion from app.models.raw_material import RawMaterial, RawMaterialPriceVersion
@@ -119,6 +119,78 @@ def calculate_mix_cost(db: Session, mix_id: int, overrides: dict | None = None)
} }
def _calculate_formula_cost_from_product_ingredients(
product_ingredients: list[ProductIngredient],
overrides: dict | None = None,
) -> dict:
overrides = overrides or {}
total_mix_kg = 0.0
total_mix_cost = 0.0
warnings: list[str] = []
lines: list[dict] = []
for ingredient in product_ingredients:
raw_material = ingredient.raw_material
active_price = get_active_price(raw_material)
if active_price is None:
warnings.append(f"{raw_material.name} has no active price")
lines.append(
{
"id": ingredient.id,
"raw_material_id": raw_material.id,
"raw_material_name": raw_material.name,
"quantity_kg": ingredient.quantity_kg,
"cost_per_kg": None,
"line_cost": None,
"notes": ingredient.notes,
}
)
total_mix_kg += ingredient.quantity_kg
continue
market_value = overrides.get("raw_material_market_values", {}).get(str(raw_material.id), active_price.market_value)
waste_percentage = overrides.get("raw_material_waste_percentages", {}).get(str(raw_material.id), active_price.waste_percentage)
price_stub = RawMaterialPriceVersion(
raw_material_id=raw_material.id,
market_value=market_value,
waste_percentage=waste_percentage,
effective_date=active_price.effective_date,
status=active_price.status,
)
price_comp = calculate_raw_material_cost(raw_material, price_stub)
line_cost = round(ingredient.quantity_kg * price_comp.cost_per_kg, 4)
total_mix_kg += ingredient.quantity_kg
total_mix_cost += line_cost
lines.append(
{
"id": ingredient.id,
"raw_material_id": raw_material.id,
"raw_material_name": raw_material.name,
"quantity_kg": ingredient.quantity_kg,
"cost_per_kg": price_comp.cost_per_kg,
"line_cost": line_cost,
"notes": ingredient.notes,
}
)
if total_mix_kg == 0:
warnings.append("Mix total kg is zero")
mix_cost_per_kg = None
else:
mix_cost_per_kg = round(total_mix_cost / total_mix_kg, 4)
if not product_ingredients:
warnings.append("Mix has no ingredients")
return {
"ingredients": lines,
"total_mix_kg": round(total_mix_kg, 4),
"total_mix_cost": round(total_mix_cost, 4),
"mix_cost_per_kg": mix_cost_per_kg,
"warnings": warnings,
}
def _get_process_costs(db: Session, process_name: str | None, overrides: dict) -> tuple[float, float, float, list[str]]: def _get_process_costs(db: Session, process_name: str | None, overrides: dict) -> tuple[float, float, float, list[str]]:
if not process_name: if not process_name:
return 0.0, 0.0, 0.0, ["Missing bagging process"] return 0.0, 0.0, 0.0, ["Missing bagging process"]
@@ -192,11 +264,21 @@ def extract_unit_quantity_kg(unit_of_measure: str) -> float:
def calculate_product_cost(db: Session, product_id: int, overrides: dict | None = None) -> dict: def calculate_product_cost(db: Session, product_id: int, overrides: dict | None = None) -> dict:
overrides = overrides or {} overrides = overrides or {}
overrides = {**overrides, "tenant_id": overrides.get("tenant_id")} overrides = {**overrides, "tenant_id": overrides.get("tenant_id")}
product = db.scalar(select(Product).where(Product.id == product_id).options(selectinload(Product.mix))) product = db.scalar(
select(Product)
.where(Product.id == product_id)
.options(
selectinload(Product.mix),
selectinload(Product.ingredients).selectinload(ProductIngredient.raw_material).selectinload(RawMaterial.price_versions),
)
)
if product is None: if product is None:
raise ValueError(f"Product {product_id} not found") raise ValueError(f"Product {product_id} not found")
overrides["tenant_id"] = product.tenant_id overrides["tenant_id"] = product.tenant_id
if product.ingredients:
mix_result = _calculate_formula_cost_from_product_ingredients(product.ingredients, overrides=overrides)
else:
mix_result = calculate_mix_cost(db, product.mix_id, overrides=overrides) mix_result = calculate_mix_cost(db, product.mix_id, overrides=overrides)
warnings = list(mix_result["warnings"]) warnings = list(mix_result["warnings"])
sale_unit_kg = extract_unit_quantity_kg(product.unit_of_measure) sale_unit_kg = extract_unit_quantity_kg(product.unit_of_measure)
@@ -3,8 +3,14 @@ from __future__ import annotations
import re import re
from app.models.mix_calculator import MixCalculatorSession from app.models.mix_calculator import MixCalculatorSession
from app.schemas.mix_calculator import MixCalculatorPreviewRead
def mix_calculator_pdf_filename(session_record: MixCalculatorSession) -> str: def mix_calculator_pdf_filename(session_record: MixCalculatorSession) -> str:
raw = f"{session_record.session_number}_{session_record.client_name}_{session_record.product_name}.pdf" raw = f"{session_record.session_number}_{session_record.client_name}_{session_record.product_name}.pdf"
return re.sub(r"[^\w.\-]+", "_", raw) return re.sub(r"[^\w.\-]+", "_", raw)
def mix_calculator_preview_pdf_filename(preview: MixCalculatorPreviewRead) -> str:
raw = f"MixCalculator_{preview.client_name}_{preview.product_name}_{preview.mix_date}.pdf"
return re.sub(r"[^\w.\-]+", "_", raw)
+297 -239
View File
@@ -1,7 +1,8 @@
from __future__ import annotations from __future__ import annotations
from io import BytesIO from io import BytesIO
from math import ceil from pathlib import Path
from types import SimpleNamespace
from app.models.mix_calculator import MixCalculatorSession from app.models.mix_calculator import MixCalculatorSession
@@ -24,272 +25,329 @@ def _fractional_bag_warning(session_record: MixCalculatorSession) -> str | None:
) )
def build_mix_calculator_pdf(session_record: MixCalculatorSession) -> bytes: def _coerce_pdf_source(source):
if isinstance(source, dict):
lines = [SimpleNamespace(**line) if isinstance(line, dict) else line for line in source.get("lines", [])]
return SimpleNamespace(**{**source, "lines": lines})
return source
def build_mix_calculator_pdf(session_record: MixCalculatorSession | dict) -> bytes:
session_record = _coerce_pdf_source(session_record)
try: try:
from reportlab.lib import colors from reportlab.lib import colors
from reportlab.lib.pagesizes import A4 from reportlab.lib.pagesizes import A4
from reportlab.lib.styles import ParagraphStyle, getSampleStyleSheet from reportlab.lib.utils import ImageReader
from reportlab.lib.units import mm from reportlab.pdfbase.pdfmetrics import stringWidth
from reportlab.platypus import Paragraph, SimpleDocTemplate, Spacer, Table, TableStyle from reportlab.pdfgen import canvas
except ModuleNotFoundError as exc: except ModuleNotFoundError as exc:
raise MixCalculatorPdfUnavailableError( raise MixCalculatorPdfUnavailableError(
"PDF generation is unavailable because 'reportlab' is not installed. " "PDF generation is unavailable because 'reportlab' is not installed. "
"Install backend dependencies again to enable PDF export." "Install backend dependencies again to enable PDF export."
) from exc ) from exc
page_width, page_height = A4
margin = 26
gutter = 10
content_width = page_width - (margin * 2)
page_top = page_height - 40
palette = {
"page": colors.HexColor("#FFFFFF"),
"line": colors.HexColor("#000000"),
"muted": colors.HexColor("#000000"),
"text": colors.HexColor("#000000"),
"warning_bg": colors.HexColor("#FFFFFF"),
"warning_text": colors.HexColor("#000000"),
}
logo_path = Path(__file__).resolve().parents[3] / "frontend" / "static" / "logo-hsf.png"
def clamp(value: float, minimum: float, maximum: float) -> float:
return max(minimum, min(maximum, value))
def fit_text(value: str, font_name: str, font_size: float, max_width: float) -> str:
if stringWidth(value, font_name, font_size) <= max_width:
return value
ellipsis = "..."
available = max_width - stringWidth(ellipsis, font_name, font_size)
trimmed = value
while trimmed and stringWidth(trimmed, font_name, font_size) > available:
trimmed = trimmed[:-1]
return f"{trimmed.rstrip()}{ellipsis}" if trimmed else ellipsis
def wrap_text(value: str, font_name: str, font_size: float, max_width: float, max_lines: int) -> list[str]:
words = value.split()
if not words:
return []
lines: list[str] = []
current = words[0]
for word in words[1:]:
candidate = f"{current} {word}"
if stringWidth(candidate, font_name, font_size) <= max_width:
current = candidate
else:
lines.append(current)
current = word
if len(lines) == max_lines - 1:
break
if len(lines) < max_lines:
lines.append(current)
remaining_words = words[len(" ".join(lines).split()) :]
if remaining_words and lines:
lines[-1] = fit_text(f"{lines[-1]} {' '.join(remaining_words)}", font_name, font_size, max_width)
return lines[:max_lines]
def draw_box(pdf: canvas.Canvas, x: float, y_top: float, width: float, height: float):
pdf.setFillColor(palette["page"])
pdf.setStrokeColor(palette["line"])
pdf.setLineWidth(1)
pdf.rect(x, y_top - height, width, height, fill=1, stroke=1)
def draw_label_value_card(
pdf: canvas.Canvas,
x: float,
y_top: float,
width: float,
height: float,
label: str,
value: str,
subtitle: str | None = None,
value_font_size: float = 14,
):
draw_box(pdf, x, y_top, width, height)
inset_x = x + 14
label_y = y_top - 16
pdf.setFillColor(palette["muted"])
pdf.setFont("Helvetica-Bold", 7.5)
pdf.drawString(inset_x, label_y, label.upper())
value_y = y_top - 38
pdf.setFillColor(palette["text"])
pdf.setFont("Helvetica-Bold", value_font_size)
pdf.drawString(inset_x, value_y, fit_text(value, "Helvetica-Bold", value_font_size, width - 28))
if subtitle:
pdf.setFillColor(palette["muted"])
pdf.setFont("Helvetica", 8)
pdf.drawString(inset_x, y_top - height + 14, fit_text(subtitle, "Helvetica", 8, width - 28))
buffer = BytesIO() buffer = BytesIO()
document = SimpleDocTemplate( pdf = canvas.Canvas(buffer, pagesize=A4)
buffer, session_number = getattr(session_record, "session_number", None)
pagesize=A4, document_title = (
leftMargin=14 * mm, f"{session_number} - {session_record.product_name}"
rightMargin=14 * mm, if session_number
topMargin=14 * mm, else f"Mix Calculator - {session_record.product_name}"
bottomMargin=14 * mm,
title=f"{session_record.session_number} - {session_record.product_name}",
author="Lean 101 Clients",
) )
pdf.setTitle(document_title)
pdf.setAuthor("Lean 101 Clients")
styles = getSampleStyleSheet() pdf.setFillColor(palette["page"])
eyebrow = ParagraphStyle( pdf.rect(0, 0, page_width, page_height, stroke=0, fill=1)
"Eyebrow",
parent=styles["BodyText"],
fontName="Helvetica-Bold",
fontSize=8,
leading=10,
textColor=colors.HexColor("#62736B"),
spaceAfter=5,
)
title = ParagraphStyle(
"Title",
parent=styles["Heading1"],
fontName="Helvetica-Bold",
fontSize=24,
leading=26,
textColor=colors.HexColor("#21312A"),
spaceAfter=6,
)
subtitle = ParagraphStyle(
"Subtitle",
parent=styles["BodyText"],
fontName="Helvetica",
fontSize=10,
leading=13,
textColor=colors.HexColor("#6B7A73"),
)
label = ParagraphStyle(
"Label",
parent=styles["BodyText"],
fontName="Helvetica-Bold",
fontSize=7,
leading=9,
textColor=colors.HexColor("#6B7A73"),
)
value = ParagraphStyle(
"Value",
parent=styles["BodyText"],
fontName="Helvetica-Bold",
fontSize=11,
leading=13,
textColor=colors.HexColor("#21312A"),
)
card_value = ParagraphStyle(
"CardValue",
parent=value,
fontSize=16,
leading=18,
)
body = ParagraphStyle(
"Body",
parent=styles["BodyText"],
fontName="Helvetica",
fontSize=9,
leading=12,
textColor=colors.HexColor("#304038"),
)
section_title = ParagraphStyle(
"SectionTitle",
parent=styles["Heading2"],
fontName="Helvetica-Bold",
fontSize=13,
leading=15,
textColor=colors.HexColor("#21312A"),
)
warnings = [] current_y = page_top
bag_warning = _fractional_bag_warning(session_record) mix_date_label = f"{session_record.mix_date.day} {session_record.mix_date.strftime('%B %Y')}"
if bag_warning:
warnings.append(bag_warning)
story = [ if logo_path.exists():
Paragraph(f"Mix Calculator | {session_record.session_number}", eyebrow), logo_source = str(logo_path)
Paragraph(session_record.product_name, title), try:
Paragraph(f"{session_record.client_name} &nbsp;&middot;&nbsp; {session_record.mix_name}", subtitle), from PIL import Image
Spacer(1, 8),
]
header_table = Table( logo_source = Image.open(logo_path).convert("L")
[ except ModuleNotFoundError:
[ pass
[
Paragraph("Mix date", label),
Paragraph(session_record.mix_date.strftime("%d %b %Y"), value),
],
[
Paragraph("Prepared by", label),
Paragraph(session_record.prepared_by_name, value),
],
[
Paragraph("Status", label),
Paragraph(session_record.status.title(), value),
],
]
],
colWidths=[60 * mm, 60 * mm, 52 * mm],
)
header_table.setStyle(
TableStyle(
[
("VALIGN", (0, 0), (-1, -1), "TOP"),
("BOX", (0, 0), (-1, -1), 0.8, colors.HexColor("#DBE4DE")),
("INNERGRID", (0, 0), (-1, -1), 0.8, colors.HexColor("#DBE4DE")),
("BACKGROUND", (0, 0), (-1, -1), colors.white),
("LEFTPADDING", (0, 0), (-1, -1), 10),
("RIGHTPADDING", (0, 0), (-1, -1), 10),
("TOPPADDING", (0, 0), (-1, -1), 9),
("BOTTOMPADDING", (0, 0), (-1, -1), 9),
]
)
)
story.extend([header_table, Spacer(1, 10)])
summary_table = Table( logo_reader = ImageReader(logo_source)
[ logo_width, logo_height = logo_reader.getSize()
[ aspect_ratio = logo_height / max(logo_width, 1)
[Paragraph("Batch size", label), Paragraph(f"{_fmt_number(session_record.batch_size_kg)}kg", card_value)], draw_width = 108
[Paragraph("Total output", label), Paragraph(f"{_fmt_number(session_record.total_kg)}kg", card_value)], draw_height = draw_width * aspect_ratio
[Paragraph("Bags", label), Paragraph(_fmt_number(session_record.total_bags), card_value)], pdf.drawImage(
[Paragraph("Unit pack", label), Paragraph(f"{_fmt_number(session_record.product_unit_size_kg)}kg", card_value)], logo_reader,
] margin,
], current_y - draw_height,
colWidths=[43 * mm, 43 * mm, 43 * mm, 43 * mm], width=draw_width,
height=draw_height,
preserveAspectRatio=True,
mask="auto",
) )
summary_table.setStyle( current_y -= draw_height + 20
TableStyle(
[
("BOX", (0, 0), (-1, -1), 0.8, colors.HexColor("#DBE4DE")),
("INNERGRID", (0, 0), (-1, -1), 0.8, colors.HexColor("#DBE4DE")),
("BACKGROUND", (0, 0), (-1, -1), colors.HexColor("#F9FBFA")),
("VALIGN", (0, 0), (-1, -1), "TOP"),
("LEFTPADDING", (0, 0), (-1, -1), 10),
("RIGHTPADDING", (0, 0), (-1, -1), 10),
("TOPPADDING", (0, 0), (-1, -1), 10),
("BOTTOMPADDING", (0, 0), (-1, -1), 10),
]
)
)
story.extend([summary_table, Spacer(1, 10)])
detail_table = Table( pdf.setFillColor(palette["text"])
[ pdf.setFont("Helvetica-Bold", 15)
[ pdf.drawString(margin, current_y, "Calculated Output")
[Paragraph("Mix source", label), Paragraph(session_record.mix_name, value), Paragraph(f"Saved against {session_record.product_unit_of_measure} units.", body)], current_y -= 16
[Paragraph("Composition", label), Paragraph(f"{_fmt_number(sum(line.mix_percentage for line in session_record.lines))}%", value), Paragraph(f"{len(session_record.lines)} raw material{'s' if len(session_record.lines) != 1 else ''} in the blend.", body)],
[Paragraph("Estimated pages", label), Paragraph(str(max(1, ceil(len(session_record.lines) / 18))), value), Paragraph("Formatted for A4 PDF export.", body)], pdf.setFillColor(palette["muted"])
] pdf.setFont("Helvetica", 10)
], pdf.drawString(margin, current_y, "Snapshot of the scaled raw material requirements.")
colWidths=[60 * mm, 60 * mm, 52 * mm], current_y -= 20
stat_height = 66
stat_width = (content_width - (gutter * 2)) / 3
draw_label_value_card(
pdf,
margin,
current_y,
stat_width,
stat_height,
"Total kg",
_fmt_number(session_record.total_kg),
"Scaled batch size",
value_font_size=18,
) )
detail_table.setStyle( draw_label_value_card(
TableStyle( pdf,
[ margin + stat_width + gutter,
("BACKGROUND", (0, 0), (-1, -1), colors.HexColor("#F4F8F5")), current_y,
("VALIGN", (0, 0), (-1, -1), "TOP"), stat_width,
("LEFTPADDING", (0, 0), (-1, -1), 10), stat_height,
("RIGHTPADDING", (0, 0), (-1, -1), 10), "Total bags",
("TOPPADDING", (0, 0), (-1, -1), 10), _fmt_number(session_record.total_bags),
("BOTTOMPADDING", (0, 0), (-1, -1), 10), session_record.product_unit_of_measure,
] value_font_size=18,
) )
draw_label_value_card(
pdf,
margin + ((stat_width + gutter) * 2),
current_y,
stat_width,
stat_height,
"Prepared by",
session_record.prepared_by_name,
mix_date_label,
value_font_size=10.5,
) )
story.extend([detail_table, Spacer(1, 10)]) current_y -= stat_height + 12
detail_height = 52
detail_width = (content_width - gutter) / 2
draw_label_value_card(
pdf,
margin,
current_y,
detail_width,
detail_height,
"Client",
session_record.client_name,
value_font_size=12,
)
draw_label_value_card(
pdf,
margin + detail_width + gutter,
current_y,
detail_width,
detail_height,
"Product",
session_record.product_name,
value_font_size=12,
)
current_y -= detail_height + 8
draw_label_value_card(
pdf,
margin,
current_y,
detail_width,
detail_height,
"Mix source",
session_record.mix_name,
value_font_size=11,
)
draw_label_value_card(
pdf,
margin + detail_width + gutter,
current_y,
detail_width,
detail_height,
"Unit size",
f"{_fmt_number(session_record.product_unit_size_kg)}kg",
value_font_size=12,
)
current_y -= detail_height + 10
warning = _fractional_bag_warning(session_record)
note_lines: list[str] = []
warning_lines: list[str] = []
strip_height = 0
if session_record.notes: if session_record.notes:
notes_table = Table( note_lines = wrap_text(session_record.notes.replace("\n", " "), "Helvetica", 7.5, content_width - 28, 2)
[[Paragraph("Notes", label)], [Paragraph(session_record.notes.replace("\n", "<br/>"), body)]], strip_height += 30
colWidths=[172 * mm], if warning:
) warning_lines = wrap_text(warning, "Helvetica", 7.5, content_width - 28, 2)
notes_table.setStyle( strip_height += 30
TableStyle( if strip_height:
[ strip_height += 6
("BACKGROUND", (0, 0), (-1, 0), colors.HexColor("#F4F8F5")),
("BOX", (0, 0), (-1, -1), 0.8, colors.HexColor("#DBE4DE")),
("LEFTPADDING", (0, 0), (-1, -1), 10),
("RIGHTPADDING", (0, 0), (-1, -1), 10),
("TOPPADDING", (0, 0), (-1, -1), 8),
("BOTTOMPADDING", (0, 0), (-1, -1), 8),
]
)
)
story.extend([notes_table, Spacer(1, 10)])
if warnings: table_header_height = 24
warning_rows = [[Paragraph("Warnings", label)]] table_bottom_padding = 12
warning_rows.extend([[Paragraph(warning, body)] for warning in warnings]) table_top = current_y
warnings_table = Table(warning_rows, colWidths=[172 * mm]) available_table_height = table_top - margin - strip_height - table_header_height - table_bottom_padding
warnings_table.setStyle( row_count = max(len(session_record.lines), 1)
TableStyle( row_height = clamp(available_table_height / row_count, 16, 32)
[ table_font_size = clamp(row_height * 0.36, 7, 11)
("BACKGROUND", (0, 0), (-1, 0), colors.HexColor("#FFF5E6")), table_height = table_header_height + (row_height * row_count)
("BACKGROUND", (0, 1), (-1, -1), colors.HexColor("#FFF9EF")), table_bottom = table_top - table_height
("BOX", (0, 0), (-1, -1), 0.8, colors.HexColor("#E8C483")),
("LEFTPADDING", (0, 0), (-1, -1), 10),
("RIGHTPADDING", (0, 0), (-1, -1), 10),
("TOPPADDING", (0, 0), (-1, -1), 8),
("BOTTOMPADDING", (0, 0), (-1, -1), 8),
]
)
)
story.extend([warnings_table, Spacer(1, 10)])
story.extend( pdf.setFillColor(palette["muted"])
[ pdf.setFont("Helvetica-Bold", 7.5)
Paragraph("Required Raw Materials", label), pdf.drawString(margin + 4, table_top - 7, "RAW MATERIAL")
Paragraph("Blend composition", section_title), pdf.drawString(margin + content_width - 190, table_top - 7, "REQUIRED KG")
Paragraph(f"{session_record.product_unit_of_measure} · {_fmt_number(session_record.product_unit_size_kg)}kg per unit", subtitle),
Spacer(1, 6), pdf.setStrokeColor(palette["line"])
] pdf.setLineWidth(0.8)
) pdf.line(margin, table_top - table_header_height, margin + content_width, table_top - table_header_height)
left_col_x = margin + 6
right_col_x = margin + content_width - 190
y_cursor = table_top - table_header_height
table_rows = [["Raw material", "Mix %", "Required kg", "Unit"]]
for line in session_record.lines: for line in session_record.lines:
table_rows.append( y_cursor -= row_height
[ pdf.setStrokeColor(palette["line"])
Paragraph(f"<b>{line.raw_material_name}</b>", body), pdf.setLineWidth(0.6)
Paragraph(f"{_fmt_number(line.mix_percentage)}%", body), pdf.line(margin, y_cursor, margin + content_width, y_cursor)
Paragraph(f"{_fmt_number(line.required_kg)}kg", body),
Paragraph(line.unit, body),
]
)
composition_table = Table(table_rows, colWidths=[88 * mm, 24 * mm, 34 * mm, 26 * mm], repeatRows=1) text_y = y_cursor + (row_height / 2) - (table_font_size * 0.35)
composition_table.setStyle( pdf.setFillColor(palette["text"])
TableStyle( pdf.setFont("Helvetica-Bold", table_font_size)
[ pdf.drawString(
("BACKGROUND", (0, 0), (-1, 0), colors.HexColor("#EEF4F0")), left_col_x,
("TEXTCOLOR", (0, 0), (-1, 0), colors.HexColor("#4F6158")), text_y,
("FONTNAME", (0, 0), (-1, 0), "Helvetica-Bold"), fit_text(line.raw_material_name, "Helvetica-Bold", table_font_size, content_width - 210),
("FONTSIZE", (0, 0), (-1, 0), 8),
("BOTTOMPADDING", (0, 0), (-1, 0), 8),
("TOPPADDING", (0, 0), (-1, 0), 8),
("LEFTPADDING", (0, 0), (-1, -1), 9),
("RIGHTPADDING", (0, 0), (-1, -1), 9),
("GRID", (0, 0), (-1, -1), 0.6, colors.HexColor("#DBE4DE")),
("VALIGN", (0, 0), (-1, -1), "TOP"),
("ROWBACKGROUNDS", (0, 1), (-1, -1), [colors.white, colors.HexColor("#FBFCFB")]),
]
) )
) pdf.setFont("Helvetica", table_font_size)
story.append(composition_table) pdf.drawString(right_col_x, text_y, f"{_fmt_number(line.required_kg)}kg")
document.build(story) strip_y = table_bottom - 6
if note_lines:
note_height = 24 if len(note_lines) == 1 else 30
pdf.setFillColor(palette["page"])
pdf.setStrokeColor(palette["line"])
pdf.rect(margin, strip_y - note_height, content_width, note_height, fill=1, stroke=1)
pdf.setFillColor(palette["muted"])
pdf.setFont("Helvetica-Bold", 7)
pdf.drawString(margin + 10, strip_y - 10, "NOTES")
pdf.setFillColor(palette["text"])
pdf.setFont("Helvetica", 7.5)
for idx, text in enumerate(note_lines):
pdf.drawString(margin + 10, strip_y - 20 - (idx * 8), text)
strip_y -= note_height + 6
if warning_lines:
warning_height = 24 if len(warning_lines) == 1 else 30
pdf.setFillColor(palette["warning_bg"])
pdf.setStrokeColor(palette["line"])
pdf.rect(margin, strip_y - warning_height, content_width, warning_height, fill=1, stroke=1)
pdf.setFillColor(palette["warning_text"])
pdf.setFont("Helvetica-Bold", 7)
pdf.drawString(margin + 10, strip_y - 10, "WARNING")
pdf.setFont("Helvetica", 7.5)
for idx, text in enumerate(warning_lines):
pdf.drawString(margin + 10, strip_y - 20 - (idx * 8), text)
pdf.showPage()
pdf.save()
return buffer.getvalue() return buffer.getvalue()
+57 -21
View File
@@ -8,7 +8,7 @@ from sqlalchemy.orm import Session, joinedload, selectinload
from app.api.deps import AuthSession from app.api.deps import AuthSession
from app.models.mix import Mix, MixIngredient from app.models.mix import Mix, MixIngredient
from app.models.mix_calculator import MixCalculatorSession, MixCalculatorSessionLine from app.models.mix_calculator import MixCalculatorSession, MixCalculatorSessionLine
from app.models.product import Product from app.models.product import Product, ProductIngredient
from app.schemas.mix_calculator import MixCalculatorSessionCreate, MixCalculatorSessionUpdate from app.schemas.mix_calculator import MixCalculatorSessionCreate, MixCalculatorSessionUpdate
from app.services.costing_engine import extract_unit_quantity_kg from app.services.costing_engine import extract_unit_quantity_kg
@@ -28,8 +28,42 @@ def _load_product_for_calculation(db: Session, tenant_id: str, product_id: int)
return db.scalar( return db.scalar(
select(Product) select(Product)
.where(Product.id == product_id, Product.tenant_id == tenant_id, Product.visible.is_(True)) .where(Product.id == product_id, Product.tenant_id == tenant_id, Product.visible.is_(True))
.options(selectinload(Product.mix).selectinload(Mix.ingredients).selectinload(MixIngredient.raw_material)) .options(
selectinload(Product.ingredients).selectinload(ProductIngredient.raw_material),
selectinload(Product.mix).selectinload(Mix.ingredients).selectinload(MixIngredient.raw_material),
) )
)
def _resolved_formula_rows(product: Product) -> tuple[list[dict], float]:
if product.ingredients:
rows = [
{
"raw_material_id": ingredient.raw_material_id,
"raw_material_name": ingredient.raw_material.name,
"quantity_kg": ingredient.quantity_kg,
"unit": ingredient.raw_material.unit_of_measure,
"sort_order": ingredient.sort_order,
}
for ingredient in product.ingredients
if ingredient.raw_material is not None
]
elif product.mix is not None:
rows = [
{
"raw_material_id": ingredient.raw_material_id,
"raw_material_name": ingredient.raw_material.name if ingredient.raw_material is not None else f"Raw material {ingredient.raw_material_id}",
"quantity_kg": ingredient.quantity_kg,
"unit": ingredient.raw_material.unit_of_measure if ingredient.raw_material is not None else "kg",
"sort_order": index,
}
for index, ingredient in enumerate(product.mix.ingredients, start=1)
]
else:
rows = []
rows.sort(key=lambda row: (row["sort_order"], row["raw_material_name"]))
return rows, round(sum(row["quantity_kg"] for row in rows), 4)
def _fractional_bag_warning(batch_size_kg: float, total_bags: float, unit_of_measure: str) -> str | None: def _fractional_bag_warning(batch_size_kg: float, total_bags: float, unit_of_measure: str) -> str | None:
@@ -54,12 +88,9 @@ def calculate_mix_calculator_preview(
raise ValueError("Product not found") raise ValueError("Product not found")
if product.client_name != values["client_name"]: if product.client_name != values["client_name"]:
raise ValueError("Selected product does not belong to the chosen client") raise ValueError("Selected product does not belong to the chosen client")
if product.mix is None: formula_rows, source_total_kg = _resolved_formula_rows(product)
raise ValueError("Product mix is not configured")
source_total_kg = round(sum(ingredient.quantity_kg for ingredient in product.mix.ingredients), 4)
if source_total_kg <= 0: if source_total_kg <= 0:
raise ValueError("Product mix has no source kilograms to scale") raise ValueError("Product has no source kilograms to scale")
batch_size_kg = float(values["batch_size_kg"]) batch_size_kg = float(values["batch_size_kg"])
scale_factor = batch_size_kg / source_total_kg scale_factor = batch_size_kg / source_total_kg
@@ -72,18 +103,17 @@ def calculate_mix_calculator_preview(
warnings.append(bag_warning) warnings.append(bag_warning)
lines = [] lines = []
for index, ingredient in enumerate(product.mix.ingredients, start=1): for index, ingredient in enumerate(formula_rows, start=1):
mix_percentage = round((ingredient.quantity_kg / source_total_kg) * 100, 4) mix_percentage = round((ingredient["quantity_kg"] / source_total_kg) * 100, 4)
required_kg = round(ingredient.quantity_kg * scale_factor, 4) required_kg = round(ingredient["quantity_kg"] * scale_factor, 4)
raw_material = ingredient.raw_material
lines.append( lines.append(
{ {
"raw_material_id": raw_material.id if raw_material is not None else ingredient.raw_material_id, "raw_material_id": ingredient["raw_material_id"],
"raw_material_name": raw_material.name if raw_material is not None else f"Raw material {ingredient.raw_material_id}", "raw_material_name": ingredient["raw_material_name"],
"required_kg": required_kg, "required_kg": required_kg,
"mix_percentage": mix_percentage, "mix_percentage": mix_percentage,
"unit": raw_material.unit_of_measure if raw_material is not None else "kg", "unit": ingredient["unit"],
"sort_order": index, "sort_order": ingredient["sort_order"] or index,
} }
) )
@@ -92,7 +122,7 @@ def calculate_mix_calculator_preview(
"product_id": product.id, "product_id": product.id,
"product_name": product.name, "product_name": product.name,
"mix_id": product.mix_id, "mix_id": product.mix_id,
"mix_name": product.mix.name, "mix_name": product.mix.name if product.mix else product.name,
"mix_date": values["mix_date"], "mix_date": values["mix_date"],
"batch_size_kg": round(batch_size_kg, 4), "batch_size_kg": round(batch_size_kg, 4),
"total_bags": total_bags, "total_bags": total_bags,
@@ -108,10 +138,16 @@ def calculate_mix_calculator_preview(
def build_mix_calculator_options(db: Session, *, tenant_id: str) -> dict: def build_mix_calculator_options(db: Session, *, tenant_id: str) -> dict:
# Aggregate mix totals in a single query instead of loading every # Prefer product-specific formulas where present; fall back to the shared
# ingredient row for every product. The previous implementation was the # mix master for legacy rows that have not been migrated yet.
# main slow path on first Mix Calculator open — it streamed the entire product_totals_rows = db.execute(
# tenant's recipe table just to compute one sum per product. select(ProductIngredient.product_id, func.coalesce(func.sum(ProductIngredient.quantity_kg), 0.0))
.join(Product, Product.id == ProductIngredient.product_id)
.where(Product.tenant_id == tenant_id)
.group_by(ProductIngredient.product_id)
).all()
product_totals: dict[int, float] = {product_id: round(total or 0.0, 4) for product_id, total in product_totals_rows}
mix_totals_rows = db.execute( mix_totals_rows = db.execute(
select(MixIngredient.mix_id, func.coalesce(func.sum(MixIngredient.quantity_kg), 0.0)) select(MixIngredient.mix_id, func.coalesce(func.sum(MixIngredient.quantity_kg), 0.0))
.join(Mix, Mix.id == MixIngredient.mix_id) .join(Mix, Mix.id == MixIngredient.mix_id)
@@ -137,7 +173,7 @@ def build_mix_calculator_options(db: Session, *, tenant_id: str) -> dict:
"mix_name": product.mix.name if product.mix else "", "mix_name": product.mix.name if product.mix else "",
"unit_of_measure": product.unit_of_measure, "unit_of_measure": product.unit_of_measure,
"unit_size_kg": round(extract_unit_quantity_kg(product.unit_of_measure), 4), "unit_size_kg": round(extract_unit_quantity_kg(product.unit_of_measure), 4),
"mix_total_kg": mix_totals.get(product.mix_id, 0.0), "mix_total_kg": product_totals.get(product.id, mix_totals.get(product.mix_id, 0.0)),
} }
for product in products for product in products
] ]
+348
View File
@@ -0,0 +1,348 @@
from __future__ import annotations
import logging
from datetime import date, datetime
from pathlib import Path
from typing import Iterable
from openpyxl import load_workbook
from sqlalchemy import select
from sqlalchemy.orm import Session
from app.models.throughput import ProductionThroughput, ThroughputProduct
logger = logging.getLogger("data_entry_app.throughput")
PRODUCTION_SHEET = "Production"
NAMES_SHEET = "Names"
# Anything at or above this kg/bag is treated as a bulka batch, not a per-bag count.
_BULKA_BAG_SIZE_THRESHOLD = 100.0
def normalise_staff_name(value: object) -> str | None:
if value is None:
return None
text = str(value).strip()
if not text:
return None
# Collapse internal whitespace, title-case for consistency.
cleaned = " ".join(text.split())
return cleaned
def calculate_kg(quantity: float | None, quantity_type: str, bag_size: float | None) -> float:
if quantity is None:
return 0.0
if quantity_type == "kg":
return float(quantity)
if bag_size is None:
return 0.0
return float(quantity) * float(bag_size)
def qa_passed(entry: ProductionThroughput) -> bool:
return bool(entry.scales_checked and entry.label_correct and entry.bag_sealed and entry.pallet_good_condition)
def serialize_entry(entry: ProductionThroughput) -> dict:
return {
"id": entry.id,
"tenant_id": entry.tenant_id,
"production_date": entry.production_date,
"product_id": entry.product_id,
"product_name_snapshot": entry.product_name_snapshot,
"bag_size": entry.bag_size,
"scales_checked": entry.scales_checked,
"label_correct": entry.label_correct,
"bag_sealed": entry.bag_sealed,
"pallet_good_condition": entry.pallet_good_condition,
"sample_box_no": entry.sample_box_no,
"test_weight_1": entry.test_weight_1,
"test_weight_2": entry.test_weight_2,
"test_weight_3": entry.test_weight_3,
"test_weight_4": entry.test_weight_4,
"test_weight_5": entry.test_weight_5,
"quantity": entry.quantity,
"quantity_type": entry.quantity_type,
"calculated_kg": entry.calculated_kg,
"staff_name": entry.staff_name,
"notes": entry.notes,
"qa_passed": qa_passed(entry),
"created_by": entry.created_by,
"created_at": entry.created_at,
"updated_at": entry.updated_at,
}
def _coerce_bool(value: object) -> bool:
if isinstance(value, bool):
return value
if value is None:
return True
if isinstance(value, (int, float)):
return bool(value)
text = str(value).strip().lower()
if text in {"yes", "y", "true", "1", "pass", "ok", "x", "checked"}:
return True
if text in {"no", "n", "false", "0", "fail"}:
return False
return True
def _coerce_float(value: object) -> float | None:
if value is None or value == "":
return None
if isinstance(value, bool):
return float(value)
if isinstance(value, (int, float)):
return float(value)
text = str(value).strip().replace(",", "")
if not text:
return None
try:
return float(text)
except ValueError:
return None
def _coerce_text(value: object) -> str | None:
if value is None:
return None
text = str(value).strip()
if not text or text.lower() in {"#value!", "#n/a", "n/a"}:
return None
return text
def _coerce_date(value: object) -> date | None:
if value is None:
return None
if isinstance(value, datetime):
return value.date()
if isinstance(value, date):
return value
text = str(value).strip()
if not text:
return None
for fmt in ("%Y-%m-%d", "%d/%m/%Y", "%m/%d/%Y"):
try:
return datetime.strptime(text, fmt).date()
except ValueError:
continue
return None
def _infer_bulka_default(name: str, bag_size: float | None) -> bool:
lowered = name.lower()
if "bulka" in lowered:
return True
if bag_size is None:
return False
return bag_size >= _BULKA_BAG_SIZE_THRESHOLD
def import_names_sheet(db: Session, workbook, tenant_id: str) -> tuple[int, int]:
"""Upsert product master from the Names sheet. Returns (created, updated)."""
if NAMES_SHEET not in workbook.sheetnames:
return (0, 0)
ws = workbook[NAMES_SHEET]
existing: dict[tuple[str, str | None], ThroughputProduct] = {}
by_item: dict[str, ThroughputProduct] = {}
by_name: dict[str, ThroughputProduct] = {}
for product in db.scalars(
select(ThroughputProduct).where(ThroughputProduct.tenant_id == tenant_id)
).all():
if product.item_id:
by_item[str(product.item_id)] = product
by_name[product.name.lower()] = product
created = 0
updated = 0
for row in ws.iter_rows(min_row=2, values_only=True):
if not row:
continue
name = _coerce_text(row[0] if len(row) > 0 else None)
if not name:
continue
item_id_raw = row[1] if len(row) > 1 else None
item_id = None
if item_id_raw is not None:
if isinstance(item_id_raw, float) and item_id_raw.is_integer():
item_id = str(int(item_id_raw))
else:
item_id = _coerce_text(item_id_raw)
product = (by_item.get(item_id) if item_id else None) or by_name.get(name.lower())
if product is None:
product = ThroughputProduct(
tenant_id=tenant_id,
item_id=item_id,
name=name,
default_bag_size=None,
is_bulka_default="bulka" in name.lower(),
active=True,
notes="Imported from Operations Throughput.xlsx",
)
db.add(product)
created += 1
if item_id:
by_item[item_id] = product
by_name[name.lower()] = product
else:
if item_id and not product.item_id:
product.item_id = item_id
if name and product.name != name:
product.name = name
updated += 1
db.flush()
return (created, updated)
def import_production_sheet(db: Session, workbook, tenant_id: str) -> tuple[int, int]:
"""Import the Production sheet. Returns (imported, skipped)."""
if PRODUCTION_SHEET not in workbook.sheetnames:
return (0, 0)
ws = workbook[PRODUCTION_SHEET]
# Header row is row 3 in the sheet (rows 1 and 2 are display banners).
products_by_name: dict[str, ThroughputProduct] = {
product.name.lower(): product
for product in db.scalars(
select(ThroughputProduct).where(ThroughputProduct.tenant_id == tenant_id)
).all()
}
bag_size_seen: dict[int, list[float]] = {}
imported = 0
skipped = 0
for row in ws.iter_rows(min_row=4, values_only=True):
if not row or len(row) < 15:
skipped += 1
continue
production_date = _coerce_date(row[0])
product_name = _coerce_text(row[1])
if production_date is None or not product_name:
skipped += 1
continue
bag_size = _coerce_float(row[2])
scales = _coerce_bool(row[3])
label = _coerce_bool(row[4])
sealed = _coerce_bool(row[5])
pallet = _coerce_bool(row[6])
sample_box = _coerce_text(row[7])
tw1 = _coerce_float(row[8])
tw2 = _coerce_float(row[9])
tw3 = _coerce_float(row[10])
tw4 = _coerce_float(row[11])
tw5 = _coerce_float(row[12])
quantity = _coerce_float(row[13]) or 0.0
staff = normalise_staff_name(row[14])
notes = _coerce_text(row[15]) if len(row) > 15 else None
# Infer quantity_type: bulka-style rows have a blank or very large bag size.
if bag_size is None or bag_size >= _BULKA_BAG_SIZE_THRESHOLD or "bulka" in product_name.lower():
quantity_type = "kg"
else:
quantity_type = "bags"
product = products_by_name.get(product_name.lower())
if product is None:
product = ThroughputProduct(
tenant_id=tenant_id,
item_id=None,
name=product_name,
default_bag_size=bag_size,
is_bulka_default=_infer_bulka_default(product_name, bag_size),
active=True,
notes="Auto-created during Operations Throughput import",
)
db.add(product)
db.flush()
products_by_name[product_name.lower()] = product
if product.id is not None and bag_size is not None and bag_size > 0:
bag_size_seen.setdefault(product.id, []).append(bag_size)
calculated = calculate_kg(quantity, quantity_type, bag_size)
entry = ProductionThroughput(
tenant_id=tenant_id,
production_date=production_date,
product_id=product.id,
product_name_snapshot=product_name,
bag_size=bag_size,
scales_checked=scales,
label_correct=label,
bag_sealed=sealed,
pallet_good_condition=pallet,
sample_box_no=sample_box,
test_weight_1=tw1,
test_weight_2=tw2,
test_weight_3=tw3,
test_weight_4=tw4,
test_weight_5=tw5,
quantity=quantity,
quantity_type=quantity_type,
calculated_kg=calculated,
staff_name=staff,
notes=notes,
created_by="workbook-import",
)
db.add(entry)
imported += 1
# Backfill default_bag_size on products that don't have one but appear in entries.
for product_id, sizes in bag_size_seen.items():
product = db.get(ThroughputProduct, product_id)
if product and product.default_bag_size is None:
# Use the most common bag size seen.
common = max(set(sizes), key=sizes.count)
product.default_bag_size = common
if not product.is_bulka_default:
product.is_bulka_default = _infer_bulka_default(product.name, common)
db.flush()
return (imported, skipped)
def import_workbook(db: Session, workbook_path: Path, tenant_id: str) -> dict:
workbook = load_workbook(workbook_path, data_only=True)
products_created, products_updated = import_names_sheet(db, workbook, tenant_id)
entries_imported, entries_skipped = import_production_sheet(db, workbook, tenant_id)
return {
"products_created": products_created,
"products_updated": products_updated,
"entries_imported": entries_imported,
"entries_skipped": entries_skipped,
}
def workbook_candidates() -> Iterable[Path]:
repo_root = Path(__file__).resolve().parents[3]
candidates = [
repo_root / "Operations Throughput.xlsx",
repo_root.parent / "Operations Throughput.xlsx",
Path.cwd() / "Operations Throughput.xlsx",
Path("/srv/lean101-clients") / "Operations Throughput.xlsx",
Path("/app") / "Operations Throughput.xlsx",
]
seen: set[str] = set()
ordered: list[Path] = []
for candidate in candidates:
key = str(candidate)
if key in seen:
continue
seen.add(key)
ordered.append(candidate)
return ordered
def resolve_workbook_path() -> Path | None:
for candidate in workbook_candidates():
if candidate.exists():
return candidate
return None
+1 -1
View File
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
[project] [project]
name = "data-entry-app-backend" name = "data-entry-app-backend"
version = "0.1.5" version = "0.1.8"
description = "Costing platform MVP backend" description = "Costing platform MVP backend"
requires-python = ">=3.11" requires-python = ">=3.11"
dependencies = [ dependencies = [
+83 -3
View File
@@ -7,6 +7,7 @@ from sqlalchemy import create_engine
from sqlalchemy.orm import Session, sessionmaker from sqlalchemy.orm import Session, sessionmaker
from sqlalchemy.pool import StaticPool from sqlalchemy.pool import StaticPool
from app.api.access import router as access_router
from app.core.access import ( from app.core.access import (
INTERNAL_USER_SUBJECT, INTERNAL_USER_SUBJECT,
get_user_permissions, get_user_permissions,
@@ -15,7 +16,8 @@ from app.core.access import (
require_permission, require_permission,
user_has_permission, user_has_permission,
) )
from app.core.security import issue_token from app.core.config import settings
from app.core.security import issue_token, verify_password
from app.db.session import Base, get_db from app.db.session import Base, get_db
from app.models.access import Permission, Role, User from app.models.access import Permission, Role, User
from app.seed_access import PERMISSION_DEFINITIONS, ROLE_DEFINITIONS, SEED_USERS, seed_access from app.seed_access import PERMISSION_DEFINITIONS, ROLE_DEFINITIONS, SEED_USERS, seed_access
@@ -42,6 +44,10 @@ def test_seed_creates_roles_permissions_and_users():
assert {role.name for role in db.query(Role).all()} == set(ROLE_DEFINITIONS.keys()) assert {role.name for role in db.query(Role).all()} == set(ROLE_DEFINITIONS.keys())
assert {p.key for p in db.query(Permission).all()} == {key for key, _ in PERMISSION_DEFINITIONS} assert {p.key for p in db.query(Permission).all()} == {key for key, _ in PERMISSION_DEFINITIONS}
assert {user.email for user in db.query(User).all()} == {entry["email"] for entry in SEED_USERS} assert {user.email for user in db.query(User).all()} == {entry["email"] for entry in SEED_USERS}
for user in db.query(User).all():
assert user.password_hash is not None
assert user.password_hash != settings.admin_password
assert verify_password(settings.admin_password, user.password_hash)
def test_seed_is_idempotent(): def test_seed_is_idempotent():
@@ -71,14 +77,20 @@ def test_admin_role_permissions_match_spec():
assert "edit_mixes" not in granted assert "edit_mixes" not in granted
def test_operations_role_is_mix_calculator_only(): def test_operations_role_is_mix_calculator_and_throughput_only():
db = _build_session() db = _build_session()
seed_access(db) seed_access(db)
ops = db.query(User).filter_by(email="ops@hunterstockfeeds.com").one() ops = db.query(User).filter_by(email="ops@hunterstockfeeds.com").one()
granted = get_user_permissions(ops) granted = get_user_permissions(ops)
assert granted == {"view_mix_calculator", "use_mix_calculator", "save_mix_calculator_session"} assert granted == {
"view_mix_calculator",
"use_mix_calculator",
"save_mix_calculator_session",
"view_throughput",
"edit_throughput",
}
assert not user_has_permission(ops, "edit_raw_materials") assert not user_has_permission(ops, "edit_raw_materials")
assert not user_has_permission(ops, "view_dashboard") assert not user_has_permission(ops, "view_dashboard")
assert not user_has_permission(ops, "manage_users") assert not user_has_permission(ops, "manage_users")
@@ -158,6 +170,22 @@ def _token_for(user: User) -> str:
return issue_token({"sub": INTERNAL_USER_SUBJECT, "user_id": user.id, "email": user.email}) return issue_token({"sub": INTERNAL_USER_SUBJECT, "user_id": user.id, "email": user.email})
@pytest.fixture()
def access_app_and_db():
db = _build_session()
seed_access(db)
db.commit()
app = FastAPI()
def override_get_db():
yield db
app.dependency_overrides[get_db] = override_get_db
app.include_router(access_router)
return TestClient(app), db
def test_route_allows_user_with_permission(app_and_db): def test_route_allows_user_with_permission(app_and_db):
client, db = app_and_db client, db = app_and_db
craig = db.query(User).filter_by(email="craig@hunterstockfeeds.com").one() craig = db.query(User).filter_by(email="craig@hunterstockfeeds.com").one()
@@ -234,3 +262,55 @@ def test_require_all_permissions(app_and_db):
denied = client.get("/needs-all", headers={"Authorization": f"Bearer {_token_for(ops)}"}) denied = client.get("/needs-all", headers={"Authorization": f"Bearer {_token_for(ops)}"})
assert denied.status_code == 403 assert denied.status_code == 403
def test_internal_login_uses_user_password_hash(access_app_and_db):
client, db = access_app_and_db
admin = db.query(User).filter_by(email="admin@hunterstockfeeds.com").one()
admin.password_hash = issue_token({"not": "a password"})
db.commit()
denied = client.post(
"/api/access/login",
json={"email": admin.email, "password": settings.admin_password},
)
assert denied.status_code == 401
def test_internal_user_can_change_own_password(access_app_and_db):
client, db = access_app_and_db
admin = db.query(User).filter_by(email="admin@hunterstockfeeds.com").one()
login_response = client.post(
"/api/access/login",
json={"email": admin.email, "password": settings.admin_password},
)
assert login_response.status_code == 200
update_response = client.patch(
"/api/access/me",
json={
"current_password": settings.admin_password,
"new_password": "new-personal-password",
},
cookies=login_response.cookies,
)
assert update_response.status_code == 200
db.refresh(admin)
assert admin.password_hash is not None
assert verify_password("new-personal-password", admin.password_hash)
assert not verify_password(settings.admin_password, admin.password_hash)
old_login = client.post(
"/api/access/login",
json={"email": admin.email, "password": settings.admin_password},
)
assert old_login.status_code == 401
new_login = client.post(
"/api/access/login",
json={"email": admin.email, "password": "new-personal-password"},
)
assert new_login.status_code == 200
+106 -7
View File
@@ -1,7 +1,7 @@
from datetime import date from datetime import date
from fastapi.testclient import TestClient from fastapi.testclient import TestClient
from sqlalchemy import create_engine, inspect, text from sqlalchemy import create_engine, inspect, select, text
from sqlalchemy.orm import Session, sessionmaker from sqlalchemy.orm import Session, sessionmaker
from sqlalchemy.pool import StaticPool from sqlalchemy.pool import StaticPool
@@ -13,7 +13,7 @@ from app.models.assumption import FreightCostRule, PackagingCostRule, ProcessCos
from app.models.client_access import ClientAccessAuditEvent, ClientAccount, ClientFeatureAccess, ClientUser from app.models.client_access import ClientAccessAuditEvent, ClientAccount, ClientFeatureAccess, ClientUser
from app.schemas.mix_calculator import MixCalculatorSessionCreate from app.schemas.mix_calculator import MixCalculatorSessionCreate
from app.models.mix import Mix, MixIngredient from app.models.mix import Mix, MixIngredient
from app.models.product import Product from app.models.product import Product, ProductIngredient
from app.models.raw_material import RawMaterial, RawMaterialPriceVersion from app.models.raw_material import RawMaterial, RawMaterialPriceVersion
from app.services.client_access_service import build_client_access_export, ensure_user_module_permissions, serialize_client_account from app.services.client_access_service import build_client_access_export, ensure_user_module_permissions, serialize_client_account
from app.services.costing_engine import calculate_mix_cost, calculate_product_cost, calculate_raw_material_cost, serialize_raw_material from app.services.costing_engine import calculate_mix_cost, calculate_product_cost, calculate_raw_material_cost, serialize_raw_material
@@ -97,12 +97,13 @@ def test_mix_and_product_cost_breakdown():
assert product_result["wholesale_price"] == 17.3268 assert product_result["wholesale_price"] == 17.3268
def test_mix_calculator_preview_scales_saved_mix_and_warns_on_fractional_bags(): def test_mix_calculator_preview_prefers_product_specific_ingredients_and_warns_on_fractional_bags():
db = build_session() db = build_session()
maize = RawMaterial(name="Maize", unit_of_measure="tonne", kg_per_unit=1000, status="active") maize = RawMaterial(name="Maize", unit_of_measure="tonne", kg_per_unit=1000, status="active")
barley = RawMaterial(name="Barley", unit_of_measure="tonne", kg_per_unit=1000, status="active") barley = RawMaterial(name="Barley", unit_of_measure="tonne", kg_per_unit=1000, status="active")
db.add_all([maize, barley]) wheat = RawMaterial(name="Wheat", unit_of_measure="tonne", kg_per_unit=1000, status="active")
db.add_all([maize, barley, wheat])
db.flush() db.flush()
mix = Mix(tenant_id="specialty-feeds", client_name="Specialty Feeds", name="Pigeon Mix", status="active", version=1) mix = Mix(tenant_id="specialty-feeds", client_name="Specialty Feeds", name="Pigeon Mix", status="active", version=1)
@@ -128,6 +129,25 @@ def test_mix_calculator_preview_scales_saved_mix_and_warns_on_fractional_bags():
bagging_process="standard_bagging", bagging_process="standard_bagging",
) )
db.add(product) db.add(product)
db.flush()
db.add_all(
[
ProductIngredient(
tenant_id="specialty-feeds",
product_id=product.id,
raw_material_id=maize.id,
quantity_kg=300,
sort_order=1,
),
ProductIngredient(
tenant_id="specialty-feeds",
product_id=product.id,
raw_material_id=wheat.id,
quantity_kg=200,
sort_order=2,
),
]
)
db.commit() db.commit()
preview = calculate_mix_calculator_preview( preview = calculate_mix_calculator_preview(
@@ -145,8 +165,9 @@ def test_mix_calculator_preview_scales_saved_mix_and_warns_on_fractional_bags():
assert preview["batch_size_kg"] == 550 assert preview["batch_size_kg"] == 550
assert preview["total_bags"] == 27.5 assert preview["total_bags"] == 27.5
assert preview["lines"][0]["required_kg"] == 353.5714 assert [line["raw_material_name"] for line in preview["lines"]] == ["Maize", "Wheat"]
assert preview["lines"][1]["required_kg"] == 196.4286 assert preview["lines"][0]["required_kg"] == 330
assert preview["lines"][1]["required_kg"] == 220
assert len(preview["warnings"]) == 1 assert len(preview["warnings"]) == 1
assert "not a whole-bag quantity" in preview["warnings"][0] assert "not a whole-bag quantity" in preview["warnings"][0]
@@ -155,7 +176,8 @@ def test_mix_calculator_options_hide_invisible_products_and_clients():
db = build_session() db = build_session()
maize = RawMaterial(name="Maize", unit_of_measure="tonne", kg_per_unit=1000, status="active") maize = RawMaterial(name="Maize", unit_of_measure="tonne", kg_per_unit=1000, status="active")
db.add(maize) barley = RawMaterial(name="Barley", unit_of_measure="tonne", kg_per_unit=1000, status="active")
db.add_all([maize, barley])
db.flush() db.flush()
visible_mix = Mix(tenant_id="hunter-premium-produce", client_name="Peckish", name="Visible Mix", status="active", version=1) visible_mix = Mix(tenant_id="hunter-premium-produce", client_name="Peckish", name="Visible Mix", status="active", version=1)
@@ -197,12 +219,89 @@ def test_mix_calculator_options_hide_invisible_products_and_clients():
), ),
] ]
) )
db.flush()
visible_product = db.scalar(select(Product).where(Product.name == "Visible Product"))
assert visible_product is not None
db.add_all(
[
ProductIngredient(
tenant_id="hunter-premium-produce",
product_id=visible_product.id,
raw_material_id=maize.id,
quantity_kg=12,
sort_order=1,
),
ProductIngredient(
tenant_id="hunter-premium-produce",
product_id=visible_product.id,
raw_material_id=barley.id,
quantity_kg=8,
sort_order=2,
),
]
)
db.commit() db.commit()
options = build_mix_calculator_options(db, tenant_id="hunter-premium-produce") options = build_mix_calculator_options(db, tenant_id="hunter-premium-produce")
assert options["clients"] == ["Peckish"] assert options["clients"] == ["Peckish"]
assert [product["product_name"] for product in options["products"]] == ["Visible Product"] assert [product["product_name"] for product in options["products"]] == ["Visible Product"]
assert options["products"][0]["mix_total_kg"] == 20
def test_calculate_product_cost_prefers_product_specific_ingredients():
db = build_session()
maize = RawMaterial(name="Maize", unit_of_measure="tonne", kg_per_unit=1000, status="active")
maize.price_versions.append(RawMaterialPriceVersion(market_value=520, waste_percentage=0.02, effective_date=date(2026, 4, 1)))
barley = RawMaterial(name="Barley", unit_of_measure="tonne", kg_per_unit=1000, status="active")
barley.price_versions.append(RawMaterialPriceVersion(market_value=470, waste_percentage=0.015, effective_date=date(2026, 4, 1)))
wheat = RawMaterial(name="Wheat", unit_of_measure="tonne", kg_per_unit=1000, status="active")
wheat.price_versions.append(RawMaterialPriceVersion(market_value=600, waste_percentage=0.01, effective_date=date(2026, 4, 1)))
db.add_all([maize, barley, wheat])
db.flush()
mix = Mix(client_name="Specialty Feeds", name="Pigeon Mix", status="active", version=1)
db.add(mix)
db.flush()
db.add_all(
[
MixIngredient(mix_id=mix.id, raw_material_id=maize.id, quantity_kg=180),
MixIngredient(mix_id=mix.id, raw_material_id=barley.id, quantity_kg=100),
]
)
db.add(ProcessCostRule(process_name="standard_bagging", grading_cost=0.055, bagging_cost=0.04, cracking_cost=0.0))
db.add(PackagingCostRule(sale_type="standard", unit_of_measure="20kg bag", own_bag=False, bag_cost=0.63))
db.add(FreightCostRule(sale_type="standard", unit_of_measure="20kg bag", cost_per_unit=1.45))
db.flush()
product = Product(
client_name="Specialty Feeds",
name="Specialty Pigeon Breeder 20kg",
mix_id=mix.id,
sale_type="standard",
own_bag=False,
unit_of_measure="20kg bag",
items_per_pallet=50,
bagging_process="standard_bagging",
distributor_margin=0.225,
wholesale_margin=0.18,
)
db.add(product)
db.flush()
db.add_all(
[
ProductIngredient(product_id=product.id, raw_material_id=maize.id, quantity_kg=300, sort_order=1),
ProductIngredient(product_id=product.id, raw_material_id=wheat.id, quantity_kg=200, sort_order=2),
]
)
db.commit()
product_result = calculate_product_cost(db, product.id)
assert product_result["finished_product_delivered"] == 15.192
assert product_result["distributor_price"] == 19.6026
assert product_result["wholesale_price"] == 18.5268
def test_sync_product_visibility_hides_configured_clients(): def test_sync_product_visibility_hides_configured_clients():
+239
View File
@@ -0,0 +1,239 @@
from datetime import date
from io import BytesIO
from openpyxl import Workbook
from sqlalchemy import create_engine, select
from sqlalchemy.orm import Session, sessionmaker
from app.db.session import Base
from app.models.mix import Mix
from app.models.product import Product
from app.models.raw_material import RawMaterial
from app.models.throughput import ProductionThroughput, ThroughputProduct
from app.seed import seed_throughput_products_from_costing
from app.services.throughput_service import (
calculate_kg,
import_names_sheet,
import_production_sheet,
normalise_staff_name,
qa_passed,
)
def _session() -> Session:
engine = create_engine("sqlite:///:memory:")
Base.metadata.create_all(bind=engine)
return sessionmaker(bind=engine, expire_on_commit=False)()
def _costing_mix(db: Session, tenant_id: str = "hunter-premium-produce") -> Mix:
raw_material = RawMaterial(
tenant_id=tenant_id,
name="Maize",
unit_of_measure="tonne",
kg_per_unit=1000,
status="active",
)
db.add(raw_material)
db.flush()
mix = Mix(tenant_id=tenant_id, client_name="Hunter", name="Maize Mix")
db.add(mix)
db.flush()
return mix
def test_calculate_kg_bags():
assert calculate_kg(10, "bags", 20) == 200.0
def test_calculate_kg_kg_ignores_bag_size():
assert calculate_kg(550, "kg", None) == 550.0
assert calculate_kg(550, "kg", 20) == 550.0
def test_calculate_kg_zero_quantity():
assert calculate_kg(0, "bags", 20) == 0.0
assert calculate_kg(None, "bags", 20) == 0.0
def test_staff_name_normalisation():
assert normalise_staff_name(" Jake ") == "Jake"
assert normalise_staff_name("jake smith") == "jake smith"
assert normalise_staff_name("") is None
assert normalise_staff_name(None) is None
def test_qa_passed_flag():
entry = ProductionThroughput(
production_date=date(2026, 1, 1),
product_name_snapshot="X",
quantity=1,
quantity_type="bags",
scales_checked=True,
label_correct=True,
bag_sealed=True,
pallet_good_condition=True,
)
assert qa_passed(entry) is True
entry.bag_sealed = False
assert qa_passed(entry) is False
def _make_workbook() -> BytesIO:
wb = Workbook()
names = wb.active
names.title = "Names"
names.append(["Name", "Item ID"])
names.append(["Whole Wheat 20kg", 1001])
names.append(["Bulka Maize", 1002])
production = wb.create_sheet("Production")
production.append(["#VALUE!", "Operations Throughput"])
production.append([None] * 8 + ["TEST WEIGHT"])
production.append([
"DATE", "GRAIN", "BAG SIZE", "SCALES", "LABEL", "SEALED", "PALLET",
"BOX", 1, 2, 3, 4, 5, "QTY", "STAFF", "NOTES",
])
production.append([date(2026, 4, 1), "Whole Wheat 20kg", 20, True, True, True, True, None, None, None, None, None, None, 100, " Jake ", None])
production.append([date(2026, 4, 1), "Bulka Maize", None, True, True, False, True, "B7", None, None, None, None, None, 1500, "Alex", "ok"])
production.append([date(2026, 4, 2), "Whole Wheat 20kg", 20, False, True, True, True, None, None, None, None, None, None, 50, "Jake", None])
buf = BytesIO()
wb.save(buf)
buf.seek(0)
return buf
def test_import_names_and_production():
from openpyxl import load_workbook
db = _session()
wb = load_workbook(_make_workbook(), data_only=True)
created, _ = import_names_sheet(db, wb, "test-tenant")
assert created == 2
imported, skipped = import_production_sheet(db, wb, "test-tenant")
assert imported == 3
assert skipped == 0
entries = db.scalars(select(ProductionThroughput).order_by(ProductionThroughput.id)).all()
bags_entry = entries[0]
assert bags_entry.quantity_type == "bags"
assert bags_entry.calculated_kg == 2000.0
assert bags_entry.staff_name == "Jake" # whitespace trimmed
bulka_entry = entries[1]
assert bulka_entry.quantity_type == "kg"
assert bulka_entry.calculated_kg == 1500.0
assert qa_passed(bulka_entry) is False # bag_sealed was False
# Product master should have absorbed default_bag_size for the wheat product
wheat = db.scalar(
select(ThroughputProduct).where(ThroughputProduct.name == "Whole Wheat 20kg")
)
assert wheat is not None
assert wheat.default_bag_size == 20
def test_product_name_snapshot_preserved_when_product_renamed():
db = _session()
product = ThroughputProduct(tenant_id="t", name="Original Name", default_bag_size=20)
db.add(product)
db.flush()
entry = ProductionThroughput(
tenant_id="t",
production_date=date(2026, 4, 1),
product_id=product.id,
product_name_snapshot=product.name,
bag_size=20,
quantity=10,
quantity_type="bags",
calculated_kg=200,
)
db.add(entry)
db.flush()
product.name = "Renamed Product"
db.flush()
reloaded = db.scalar(select(ProductionThroughput).where(ProductionThroughput.id == entry.id))
assert reloaded.product_name_snapshot == "Original Name"
def test_seed_throughput_products_from_costing_products():
db = _session()
mix = _costing_mix(db)
db.add_all(
[
Product(
tenant_id="hunter-premium-produce",
client_name="Hunter",
item_id="1001",
name="Whole Wheat 20kg",
mix_id=mix.id,
sale_type="standard",
unit_of_measure="20kg bag",
visible=True,
),
Product(
tenant_id="hunter-premium-produce",
client_name="Hunter",
item_id="1002",
name="Bulka Maize",
mix_id=mix.id,
sale_type="bulka",
unit_of_measure="tonne",
visible=False,
),
]
)
db.flush()
report = seed_throughput_products_from_costing(db)
assert report == {"created": 2, "updated": 0, "skipped": 0}
products = db.scalars(select(ThroughputProduct).order_by(ThroughputProduct.item_id)).all()
assert [product.name for product in products] == ["Whole Wheat 20kg", "Bulka Maize"]
assert products[0].default_bag_size == 20
assert products[0].is_bulka_default is False
assert products[0].active is True
assert products[1].default_bag_size is None
assert products[1].is_bulka_default is True
assert products[1].active is False
def test_seed_throughput_products_from_costing_updates_existing_by_item_id():
db = _session()
mix = _costing_mix(db)
db.add(
Product(
tenant_id="hunter-premium-produce",
client_name="Hunter",
item_id="1001",
name="Updated Wheat 25kg",
mix_id=mix.id,
sale_type="standard",
unit_of_measure="25kg bag",
visible=True,
)
)
db.add(
ThroughputProduct(
tenant_id="hunter-premium-produce",
item_id="1001",
name="Old Wheat",
default_bag_size=20,
active=False,
notes="Seeded from costing products",
)
)
db.flush()
report = seed_throughput_products_from_costing(db)
assert report == {"created": 0, "updated": 1, "skipped": 0}
products = db.scalars(select(ThroughputProduct)).all()
assert len(products) == 1
assert products[0].name == "Updated Wheat 25kg"
assert products[0].default_bag_size == 25
assert products[0].active is True
+440 -37
View File
@@ -1,6 +1,6 @@
<# <#
.SYNOPSIS .SYNOPSIS
Deploy the Lean 101 Clients app to a Digital Ocean droplet over SSH. Lean 101 deployment script ships an app to a Digital Ocean droplet over SSH.
.DESCRIPTION .DESCRIPTION
Tars the local source tree, uploads it to the droplet, and runs Tars the local source tree, uploads it to the droplet, and runs
@@ -8,14 +8,33 @@
The same script handles first-time setup and subsequent updates. The same script handles first-time setup and subsequent updates.
Designed to be swappable across Lean 101 apps. Override -AppName and
-AppSlug (and any defaults derived from them) to deploy a different app.
.PARAMETER RemoteHost .PARAMETER RemoteHost
Hostname or IP of the Digital Ocean droplet. Required. Hostname or IP of the Digital Ocean droplet. Required.
.PARAMETER RemoteUser .PARAMETER RemoteUser
SSH user. Defaults to 'root'. SSH user. Defaults to 'root'.
.PARAMETER AppName
Human-readable app name shown in the banner and log output.
Defaults to 'Clients'.
.PARAMETER AppSlug
Lowercase slug used to derive default remote path, container names, env files,
and archive names. Defaults to 'clients'. Override to retarget the script at
a different Lean 101 app (e.g. -AppSlug 'ops' for the Ops portal).
.PARAMETER RemotePath .PARAMETER RemotePath
Absolute path on the droplet. Defaults to '/srv/lean101-clients'. Absolute path on the droplet. Defaults to '/srv/lean101-<AppSlug>'.
.PARAMETER BackendContainer
Backend container name to inspect for health. Defaults to 'lean101-<AppSlug>-backend'.
.PARAMETER PortEnvKey
Env var name in the env file that holds the published port.
Defaults to '<APPSLUG_UPPER>_APP_PORT' (e.g. CLIENTS_APP_PORT).
.PARAMETER EnvFile .PARAMETER EnvFile
Local path to the production env file. Defaults to '.env.production'. Local path to the production env file. Defaults to '.env.production'.
@@ -23,6 +42,11 @@
.PARAMETER SshKey .PARAMETER SshKey
Optional path to an SSH private key. Optional path to an SSH private key.
.PARAMETER Password
Optional SSH password for password-based auth (no key). Requires sshpass on
PATH. The password is handed to ssh/scp via the SSHPASS environment variable
rather than an interactive prompt, because the script redirects ssh's I/O.
.PARAMETER ComposeFile .PARAMETER ComposeFile
Compose file name on the remote host. Defaults to 'docker-compose.production.yml'. Compose file name on the remote host. Defaults to 'docker-compose.production.yml'.
@@ -35,54 +59,276 @@
.PARAMETER SkipBuild .PARAMETER SkipBuild
Pass --no-build to docker compose (use when only env changed). Pass --no-build to docker compose (use when only env changed).
.PARAMETER Force
Skip the remote port-availability preflight. Use when you know the port is
held by this same stack and want to redeploy over it regardless.
.PARAMETER NoBanner
Suppress the ASCII banner (useful in CI).
.EXAMPLE .EXAMPLE
./deploy/Deploy.ps1 -RemoteHost 209.38.24.231 ./deploy/Deploy.ps1 -RemoteHost 209.38.24.231
.EXAMPLE .EXAMPLE
./deploy/Deploy.ps1 -RemoteHost 209.38.24.231 -Seed -Logs ./deploy/Deploy.ps1 -RemoteHost 209.38.24.231 -Seed -Logs
.EXAMPLE
# Password auth instead of an SSH key (requires sshpass on PATH):
./deploy/Deploy.ps1 -RemoteHost 209.38.24.231 -Password 'your-password' -Seed -Logs
.EXAMPLE
# Retarget the script at a different Lean 101 app:
./deploy/Deploy.ps1 -RemoteHost 1.2.3.4 -AppName 'Ops' -AppSlug 'ops'
#> #>
[CmdletBinding()] [CmdletBinding()]
param( param(
[Parameter(Mandatory = $true)] [string] $RemoteHost, [Parameter(Mandatory = $true)] [string] $RemoteHost,
[string] $RemoteUser = "root", [string] $RemoteUser = "root",
[string] $RemotePath = "/srv/lean101-clients", [string] $AppName = "Clients",
[string] $AppSlug = "clients",
[string] $RemotePath,
[string] $BackendContainer,
[string] $PortEnvKey,
[string] $EnvFile = ".env.production", [string] $EnvFile = ".env.production",
[string] $SshKey, [string] $SshKey,
[string] $Password,
[string] $ComposeFile = "docker-compose.production.yml", [string] $ComposeFile = "docker-compose.production.yml",
[switch] $Seed, [switch] $Seed,
[switch] $Logs, [switch] $Logs,
[switch] $SkipBuild [switch] $SkipBuild,
[switch] $Force,
[switch] $NoBanner
) )
$ErrorActionPreference = "Stop" $ErrorActionPreference = "Stop"
Set-StrictMode -Version Latest Set-StrictMode -Version Latest
# ── Helpers ─────────────────────────────────────────────────────────────────── # ── App identity (swappable) ──────────────────────────────────────────────────
function Write-Step($msg) { Write-Host "==> $msg" -ForegroundColor Cyan } $AppSlug = $AppSlug.ToLowerInvariant()
function Write-Warn($msg) { Write-Host "!! $msg" -ForegroundColor Yellow } $AppStack = "lean101-$AppSlug"
if (-not $RemotePath) { $RemotePath = "/srv/$AppStack" }
if (-not $BackendContainer) { $BackendContainer = "$AppStack-backend" }
if (-not $PortEnvKey) { $PortEnvKey = "$($AppSlug.ToUpperInvariant())_APP_PORT" }
# ── Palette ───────────────────────────────────────────────────────────────────
# ANSI escapes give us truecolor + dim/bold that Write-Host -ForegroundColor cannot.
$Esc = [char]27
$C = @{
Reset = "$Esc[0m"
Dim = "$Esc[2m"
Bold = "$Esc[1m"
Italic = "$Esc[3m"
Magenta = "$Esc[38;5;177m" # soft violet
Pink = "$Esc[38;5;213m"
Cyan = "$Esc[38;5;87m"
Teal = "$Esc[38;5;79m"
Green = "$Esc[38;5;120m"
Yellow = "$Esc[38;5;221m"
Red = "$Esc[38;5;203m"
Grey = "$Esc[38;5;244m"
Blue = "$Esc[38;5;117m"
}
$Glyph = @{
Step = '▸'
OK = '✓'
Warn = '!'
Info = '·'
Fail = '✗'
Arrow = '→'
Spark = '✦'
}
function Write-Banner {
if ($NoBanner) { return }
$line = '─' * 62
$title = "Lean 101 Deployment Script"
$sub = "App: $AppName Slug: $AppSlug Target: $RemoteUser@$RemoteHost"
Write-Host ""
Write-Host ("$($C.Magenta)$line$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset)$($C.Bold)$($C.Pink) ██╗ ███████╗ █████╗ ███╗ ██╗ ███╗ ██████╗ ███╗$($C.Reset)$($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset)$($C.Bold)$($C.Pink) ██║ ██╔════╝██╔══██╗████╗ ██║ ████║██╔═████╗ ██║$($C.Reset)$($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset)$($C.Bold)$($C.Pink) ██║ █████╗ ███████║██╔██╗ ██║ ██╔██║██║██╔██║ ██║$($C.Reset)$($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset)$($C.Bold)$($C.Pink) ██║ ██╔══╝ ██╔══██║██║╚██╗██║ ╚═╝██║████╔╝██║ ██║$($C.Reset)$($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset)$($C.Bold)$($C.Pink) ███████╗███████╗██║ ██║██║ ╚████║ ███████╗╚██████╔╝ ██║$($C.Reset)$($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset)$($C.Bold)$($C.Pink) ╚══════╝╚══════╝╚═╝ ╚═╝╚═╝ ╚═══╝ ╚══════╝ ╚═════╝ ╚═╝$($C.Reset)$($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset) $($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset) $($C.Bold)$($C.Cyan)$($Glyph.Spark) $title$($C.Reset)" + (' ' * (60 - $title.Length - 3)) + "$($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$($C.Reset) $($C.Dim)$($C.Grey)$sub$($C.Reset)" + (' ' * [Math]::Max(0, 60 - $sub.Length - 2)) + "$($C.Magenta)$($C.Reset)")
Write-Host ("$($C.Magenta)$line$($C.Reset)")
Write-Host ""
}
# ── Output helpers ────────────────────────────────────────────────────────────
$script:StepIndex = 0
function Write-Step([string]$msg) {
$script:StepIndex++
$num = "{0:D2}" -f $script:StepIndex
Write-Host ("$($C.Dim)$($C.Grey)[$num]$($C.Reset) $($C.Bold)$($C.Cyan)$($Glyph.Step)$($C.Reset) $($C.Bold)$msg$($C.Reset)")
}
function Write-Ok([string]$msg) { Write-Host (" $($C.Green)$($Glyph.OK)$($C.Reset) $($C.Green)$msg$($C.Reset)") }
function Write-Warn([string]$msg) { Write-Host (" $($C.Yellow)$($Glyph.Warn)$($C.Reset) $($C.Yellow)$msg$($C.Reset)") }
function Write-Info([string]$msg) { Write-Host (" $($C.Dim)$($C.Grey)$($Glyph.Info) $msg$($C.Reset)") }
function Write-Fail([string]$msg) { Write-Host (" $($C.Red)$($Glyph.Fail)$($C.Reset) $($C.Red)$msg$($C.Reset)") }
# ── Spinner ───────────────────────────────────────────────────────────────────
$Spinner = @('⠋','⠙','⠹','⠸','⠼','⠴','⠦','⠧','⠇','⠏')
function Invoke-Spinner {
<#
Runs an external process while animating a braille spinner with elapsed time.
Captures stdout/stderr and surfaces them on failure (or via -Quiet:$false).
#>
param(
[Parameter(Mandatory = $true)] [string] $Label,
[Parameter(Mandatory = $true)] [string] $FilePath,
[string[]] $ArgList = @(),
[string] $StdinFile,
[switch] $ShowOutput
)
$outFile = [System.IO.Path]::GetTempFileName()
$errFile = [System.IO.Path]::GetTempFileName()
$startParams = @{
FilePath = $FilePath
NoNewWindow = $true
PassThru = $true
RedirectStandardOutput = $outFile
RedirectStandardError = $errFile
}
if ($ArgList.Count -gt 0) { $startParams.ArgumentList = $ArgList }
if ($StdinFile) { $startParams.RedirectStandardInput = $StdinFile }
$proc = Start-Process @startParams
$start = Get-Date
$i = 0
try {
while (-not $proc.HasExited) {
$elapsed = ((Get-Date) - $start).TotalSeconds
$frame = $Spinner[$i % $Spinner.Count]
Write-Host ("`r $($C.Cyan)$frame$($C.Reset) $($C.Dim)$($C.Grey)$Label $($C.Reset)$($C.Teal)$('{0,5:0.0}s' -f $elapsed)$($C.Reset) ") -NoNewline
Start-Sleep -Milliseconds 90
$i++
}
$proc.WaitForExit()
$elapsed = ((Get-Date) - $start).TotalSeconds
$stdout = if (Test-Path $outFile) { Get-Content $outFile -Raw } else { '' }
$stderr = if (Test-Path $errFile) { Get-Content $errFile -Raw } else { '' }
if ($proc.ExitCode -eq 0) {
Write-Host ("`r $($C.Green)$($Glyph.OK)$($C.Reset) $Label $($C.Dim)$($C.Grey)$('{0,5:0.0}s' -f $elapsed)$($C.Reset)" + (' ' * 12))
if ($ShowOutput -and $stdout) {
foreach ($ln in ($stdout -split "`r?`n")) { if ($ln) { Write-Info $ln } }
}
return $stdout
}
else {
Write-Host ("`r $($C.Red)$($Glyph.Fail)$($C.Reset) $Label $($C.Dim)$($C.Grey)$('{0,5:0.0}s' -f $elapsed) (exit $($proc.ExitCode))$($C.Reset)" + (' ' * 8))
if ($stdout) { foreach ($ln in ($stdout -split "`r?`n")) { if ($ln) { Write-Host " $($C.Dim)$($C.Grey)$ln$($C.Reset)" } } }
if ($stderr) { foreach ($ln in ($stderr -split "`r?`n")) { if ($ln) { Write-Host " $($C.Red)$ln$($C.Reset)" } } }
throw "$Label failed (exit $($proc.ExitCode))"
}
}
finally {
Remove-Item $outFile -Force -ErrorAction SilentlyContinue
Remove-Item $errFile -Force -ErrorAction SilentlyContinue
}
}
# ── Helpers ───────────────────────────────────────────────────────────────────
function Get-RepoRoot { function Get-RepoRoot {
$dir = Split-Path -Parent $PSScriptRoot $dir = Split-Path -Parent $PSScriptRoot
if (-not $dir) { $dir = (Get-Location).Path } if (-not $dir) { $dir = (Get-Location).Path }
return $dir return $dir
} }
function Get-EnvValue([string] $path, [string] $key) {
foreach ($line in Get-Content $path) {
$trimmed = $line.Trim()
if (-not $trimmed -or $trimmed.StartsWith("#")) { continue }
if ($trimmed -notmatch "=") { continue }
$parts = $trimmed -split "=", 2
if ($parts[0].Trim() -eq $key) {
return $parts[1].Trim()
}
}
return $null
}
$RepoRoot = Get-RepoRoot $RepoRoot = Get-RepoRoot
$SshTarget = "$RemoteUser@$RemoteHost" $SshTarget = "$RemoteUser@$RemoteHost"
$SshOpts = @("-o", "StrictHostKeyChecking=accept-new", "-o", "BatchMode=no") $SshOpts = @("-o", "StrictHostKeyChecking=accept-new", "-o", "BatchMode=no")
if ($SshKey) { $SshOpts += @("-i", $SshKey) } if ($SshKey) { $SshOpts += @("-i", $SshKey) }
function Invoke-Ssh([string] $cmd) { # Password auth (no SSH key): the spinner runs ssh/scp with redirected I/O, so
& ssh @SshOpts $SshTarget $cmd # there is no terminal for an interactive password prompt. Feed the password
if ($LASTEXITCODE -ne 0) { throw "Remote command failed (exit $LASTEXITCODE): $cmd" } # non-interactively via sshpass instead, reading it from the SSHPASS env var.
$SshExe = 'ssh'
$ScpExe = 'scp'
$SshPrefix = @()
$ScpPrefix = @()
if ($Password) {
if (-not (Get-Command sshpass -ErrorAction SilentlyContinue)) {
throw "sshpass is required for -Password but was not found on PATH. Install it (e.g. 'scoop install sshpass') or use an SSH key."
}
$env:SSHPASS = $Password
$SshExe = 'sshpass'; $SshPrefix = @('-e', 'ssh')
$ScpExe = 'sshpass'; $ScpPrefix = @('-e', 'scp')
$SshOpts += @("-o", "PubkeyAuthentication=no", "-o", "PreferredAuthentications=password")
} }
function Invoke-Scp([string] $local, [string] $remote) { function Write-RemoteScript([string] $path, [string] $content) {
& scp @SshOpts $local "${SshTarget}:${remote}" # Remote bash chokes on Windows CRLF line endings (each line gets a trailing
if ($LASTEXITCODE -ne 0) { throw "scp failed: $local -> $remote" } # \r) and on a UTF-8 BOM. Write LF-only, no BOM.
$lf = $content -replace "`r`n", "`n" -replace "`r", "`n"
[System.IO.File]::WriteAllText($path, $lf, (New-Object System.Text.UTF8Encoding($false)))
} }
function Invoke-Ssh([string] $cmd, [string] $Label, [switch] $ShowOutput) {
if (-not $Label) { $Label = "ssh $($cmd.Substring(0, [Math]::Min(48, $cmd.Length)))..." }
# Send the command over stdin to `bash -s` rather than as a command-line
# argument. Complex commands (quotes, pipes, $(), newlines) get corrupted
# when passed as an argv token through sshpass's cygwin argument parsing;
# stdin keeps the only argv tokens simple ("bash -s").
$tmp = [System.IO.Path]::GetTempFileName()
try {
Write-RemoteScript $tmp $cmd
Invoke-Spinner -Label $Label -FilePath $SshExe -ArgList (@($SshPrefix) + @($SshOpts) + @($SshTarget, "bash -s")) -StdinFile $tmp -ShowOutput:$ShowOutput
} finally {
Remove-Item $tmp -Force -ErrorAction SilentlyContinue
}
}
function Invoke-Scp([string] $local, [string] $remote, [string] $Label) {
if (-not $Label) { $Label = "scp $(Split-Path -Leaf $local) $($Glyph.Arrow) $remote" }
Invoke-Spinner -Label $Label -FilePath $ScpExe -ArgList (@($ScpPrefix) + @($SshOpts) + @($local, "${SshTarget}:${remote}"))
}
function Try-Ssh([string] $cmd) {
$tmp = [System.IO.Path]::GetTempFileName()
try {
Write-RemoteScript $tmp $cmd
Get-Content $tmp -Raw | & $SshExe @SshPrefix @SshOpts $SshTarget "bash -s"
return $LASTEXITCODE
} finally {
Remove-Item $tmp -Force -ErrorAction SilentlyContinue
}
}
function Invoke-SshScript([string] $script, [string] $Label) {
if (-not $Label) { $Label = "ssh (remote script)" }
$tmp = [System.IO.Path]::GetTempFileName()
try {
Write-RemoteScript $tmp $script
Invoke-Spinner -Label $Label -FilePath $SshExe -ArgList (@($SshPrefix) + @($SshOpts) + @($SshTarget, "bash -s")) -StdinFile $tmp -ShowOutput
} finally {
Remove-Item $tmp -Force -ErrorAction SilentlyContinue
}
}
# ── Render banner ─────────────────────────────────────────────────────────────
Write-Banner
# ── Resolve paths ───────────────────────────────────────────────────────────── # ── Resolve paths ─────────────────────────────────────────────────────────────
Push-Location $RepoRoot Push-Location $RepoRoot
try { try {
@@ -90,15 +336,69 @@ try {
if (-not (Test-Path $EnvPath)) { if (-not (Test-Path $EnvPath)) {
throw "Env file not found at '$EnvPath'. Copy .env.production.example and fill in secrets." throw "Env file not found at '$EnvPath'. Copy .env.production.example and fill in secrets."
} }
$WorkbookCandidates = @(
(Join-Path $RepoRoot "input_data\\1.xlsx"),
(Join-Path $RepoRoot "Input Cost Spreadsheet(1).xlsx")
)
$WorkbookPath = $WorkbookCandidates | Where-Object { Test-Path $_ } | Select-Object -First 1
if (-not $WorkbookPath) {
throw "Workbook not found. Checked: $($WorkbookCandidates -join ', '). The production seed expects at least one workbook file to exist."
}
$AppPort = Get-EnvValue $EnvPath $PortEnvKey
if (-not $AppPort) { $AppPort = "8081" }
$Origin = Get-EnvValue $EnvPath "ORIGIN"
if (-not $Origin) { $Origin = "https://clients.lean-101.com.au" }
Write-Step "Preflight"
Write-Info "App : $AppName ($AppSlug)"
Write-Info "Remote host : $SshTarget"
Write-Info "Remote path : $RemotePath"
Write-Info "Container : $BackendContainer"
Write-Info "Env file : $EnvPath"
Write-Info "Compose file : $ComposeFile"
Write-Info "Origin : $Origin"
Write-Info "Port ($PortEnvKey) : $AppPort"
# ── Connectivity check ────────────────────────────────────────────────────── # ── Connectivity check ──────────────────────────────────────────────────────
Write-Step "Checking SSH connectivity to $SshTarget" Write-Step "Verifying SSH connectivity"
Invoke-Ssh "echo connected as `$(whoami) on `$(hostname)" Invoke-Ssh "echo connected as `$(whoami) on `$(hostname)" -Label "ssh handshake" -ShowOutput
# ── Remote port availability ───────────────────────────────────────────────
# A re-deploy of this same stack is expected to replace its containers in
# place, so any container belonging to this stack (name == STACK or STACK-*)
# is treated as "ours". Only a genuinely different service triggers an abort.
if ($Force) {
Write-Step "Skipping remote port check for $AppStack (-Force)"
}
else {
Write-Step "Checking that remote port $AppPort is free for $AppStack"
$portCheckCmd = @'
set -e
PORT='__APP_PORT__'
STACK='__APP_STACK__'
OWNER=$(docker ps --format '{{.Names}} {{.Ports}}' | grep -m1 ":${PORT}->" | cut -d' ' -f1 || true)
if [ -n "$OWNER" ]; then
case "$OWNER" in
"$STACK"|"$STACK"-*) : ;;
*)
echo "Port $PORT is already owned by container: $OWNER" >&2
exit 2 ;;
esac
fi
'@.Replace('__APP_PORT__', $AppPort).Replace('__APP_STACK__', $AppStack)
try {
Invoke-Ssh $portCheckCmd -Label "port $AppPort availability"
}
catch {
throw "Remote port $AppPort is already in use by a different service. Change $PortEnvKey in '$EnvPath', retire the conflicting service, or re-run with -Force to deploy anyway."
}
}
# ── Package source files ──────────────────────────────────────────────────── # ── Package source files ────────────────────────────────────────────────────
Write-Step "Packaging source files (excluding node_modules, caches, etc.)" Write-Step "Packaging source tree (excluding node_modules, caches, secrets)"
$TarFile = Join-Path $env:TEMP "lean101-deploy-$(Get-Date -Format 'yyyyMMdd-HHmmss').tar.gz" $TarFile = Join-Path $env:TEMP "$AppStack-deploy-$(Get-Date -Format 'yyyyMMdd-HHmmss').tar.gz"
$excludes = @( $excludes = @(
"--exclude=./node_modules", "--exclude=./node_modules",
@@ -106,9 +406,14 @@ try {
"--exclude=./frontend/.svelte-kit", "--exclude=./frontend/.svelte-kit",
"--exclude=./frontend/build", "--exclude=./frontend/build",
"--exclude=./.git", "--exclude=./.git",
"--exclude=./.pytest_cache",
"--exclude=./__pycache__", "--exclude=./__pycache__",
"--exclude=./backend/__pycache__", "--exclude=./backend/__pycache__",
"--exclude=./backend/app/__pycache__", "--exclude=./backend/app/__pycache__",
"--exclude=./backend/.pytest_cache",
"--exclude=./backend/.tmp",
"--exclude=./backend/.venv",
"--exclude=./backend/data_entry_app_backend.egg-info",
"--exclude=./**/__pycache__", "--exclude=./**/__pycache__",
"--exclude=./*.pyc", "--exclude=./*.pyc",
"--exclude=./.env", "--exclude=./.env",
@@ -118,39 +423,48 @@ try {
"--exclude=./*.db" "--exclude=./*.db"
) )
& tar -czf $TarFile @excludes -C $RepoRoot . Invoke-Spinner -Label "tar -czf $(Split-Path -Leaf $TarFile)" -FilePath 'tar' -ArgList (@('-czf', $TarFile) + $excludes + @('-C', $RepoRoot, '.'))
if ($LASTEXITCODE -ne 0) { throw "tar failed" }
$TarSize = [math]::Round((Get-Item $TarFile).Length / 1MB, 1) $TarSize = [math]::Round((Get-Item $TarFile).Length / 1MB, 1)
Write-Host " Archive: $TarFile ($TarSize MB)" Write-Info "Archive: $TarFile ($($C.Bold)$TarSize MB$($C.Reset))"
# ── Upload env file ───────────────────────────────────────────────────────── # ── Upload env file ─────────────────────────────────────────────────────────
Write-Step "Uploading env file" Write-Step "Uploading env file to droplet"
Invoke-Scp $EnvPath "$RemotePath/.env.production" Invoke-Scp $EnvPath "$RemotePath/.env.production" -Label "scp .env.production $($Glyph.Arrow) $RemotePath/"
Invoke-Ssh "chmod 600 '$RemotePath/.env.production'" Invoke-Ssh "chmod 600 '$RemotePath/.env.production'" -Label "chmod 600 .env.production"
# ── Upload and extract source ──────────────────────────────────────────────── # ── Upload and extract source ────────────────────────────────────────────────
Write-Step "Uploading source archive" Write-Step "Uploading source archive ($TarSize MB)"
Invoke-Scp $TarFile "/tmp/lean101-deploy.tar.gz" Invoke-Scp $TarFile "/tmp/$AppStack-deploy.tar.gz" -Label "scp archive $($Glyph.Arrow) /tmp/"
Remove-Item $TarFile -Force Remove-Item $TarFile -Force
Write-Step "Extracting on server" Write-Step "Extracting archive on server"
Invoke-Ssh "mkdir -p '$RemotePath' && tar -xzf /tmp/lean101-deploy.tar.gz -C '$RemotePath' && rm /tmp/lean101-deploy.tar.gz" Invoke-Ssh "mkdir -p '$RemotePath' && tar -xzf /tmp/$AppStack-deploy.tar.gz -C '$RemotePath' && rm /tmp/$AppStack-deploy.tar.gz" -Label "untar into $RemotePath"
# ── Docker compose up ─────────────────────────────────────────────────────── # ── Docker compose up ───────────────────────────────────────────────────────
$ComposeArgs = "--env-file .env.production -f $ComposeFile" $ComposeArgs = "--env-file .env.production -f $ComposeFile"
$BuildFlag = if ($SkipBuild) { "--no-build" } else { "--build" } $BuildFlag = if ($SkipBuild) { "--no-build" } else { "--build" }
Write-Step "Bringing stack up (build=$(-not $SkipBuild))" $buildMsg = if ($SkipBuild) { "without rebuild" } else { "with --build" }
Invoke-Ssh "cd '$RemotePath' && docker compose $ComposeArgs up -d $BuildFlag --remove-orphans" Write-Step "Bringing the $AppName stack up $buildMsg"
$composeUpCmd = "cd '$RemotePath' && docker compose $ComposeArgs up -d $BuildFlag --remove-orphans"
try {
Invoke-Ssh $composeUpCmd -Label "docker compose up $BuildFlag"
}
catch {
Write-Warn "docker compose up failed; collecting remote status and backend logs"
Try-Ssh "cd '$RemotePath' && docker compose $ComposeArgs ps"
Try-Ssh "cd '$RemotePath' && docker compose $ComposeArgs logs --tail=120 backend"
throw
}
# ── Health check ──────────────────────────────────────────────────────────── # ── Health check ────────────────────────────────────────────────────────────
Write-Step "Waiting for backend health check" Write-Step "Waiting for backend health check ($BackendContainer)"
$healthScript = @" $healthScript = @"
set -e set -e
cd '$RemotePath' cd '$RemotePath'
for i in `$(seq 1 30); do for i in `$(seq 1 30); do
status=`$(docker inspect --format='{{if .State.Health}}{{.State.Health.Status}}{{else}}{{.State.Status}}{{end}}' lean101-clients-backend 2>/dev/null || echo missing) status=`$(docker inspect --format='{{if .State.Health}}{{.State.Health.Status}}{{else}}{{.State.Status}}{{end}}' $BackendContainer 2>/dev/null || echo missing)
case "`$status" in case "`$status" in
healthy|running) echo "backend is `$status"; exit 0 ;; healthy|running) echo "backend is `$status"; exit 0 ;;
*) printf '.'; sleep 4 ;; *) printf '.'; sleep 4 ;;
@@ -158,25 +472,114 @@ for i in `$(seq 1 30); do
done done
echo; echo 'backend did not become healthy in time' >&2; exit 1 echo; echo 'backend did not become healthy in time' >&2; exit 1
"@ "@
Invoke-Ssh $healthScript Invoke-SshScript $healthScript -Label "backend health (up to ~2 min)"
# ── Seed access ─────────────────────────────────────────────────────────────
Write-Step "Seeding default internal users and permissions"
Invoke-Ssh "cd '$RemotePath' && docker compose $ComposeArgs exec -T backend python -m app.seed_access" -Label "python -m app.seed_access"
# ── Optional seed ───────────────────────────────────────────────────────────
if ($Seed) { if ($Seed) {
Write-Step "Seeding reference data" Write-Step "Seeding reference data"
Invoke-Ssh "cd '$RemotePath' && docker compose $ComposeArgs exec -T backend python -m app.seed" Invoke-Ssh "cd '$RemotePath' && docker compose $ComposeArgs exec -T backend python -m app.seed" -Label "python -m app.seed"
} }
# ── Final status ──────────────────────────────────────────────────────────── # ── Final status ────────────────────────────────────────────────────────────
Write-Step "Stack status" Write-Step "Stack status"
Invoke-Ssh "cd '$RemotePath' && docker compose $ComposeArgs ps" Invoke-Ssh "cd '$RemotePath' && docker compose $ComposeArgs ps" -Label "docker compose ps" -ShowOutput
# ── Frontend asset verification ─────────────────────────────────────────────
Write-Step "Verifying published Inter font asset"
$fontCheckBody = @'
set -e
ORIGIN="__ORIGIN__"
fetch() {
URL="$1"
if command -v curl >/dev/null 2>&1; then
curl -fsSL "$URL"
return
fi
if command -v wget >/dev/null 2>&1; then
wget -qO- "$URL"
return
fi
echo "Neither curl nor wget is available on the remote host." >&2
exit 1
}
fetch_status() {
URL="$1"
if command -v curl >/dev/null 2>&1; then
curl -fsS -o /dev/null -w "%{http_code}" "$URL"
return
fi
if command -v wget >/dev/null 2>&1; then
wget -S --spider "$URL" 2>&1 | awk '/^ HTTP\// { code=$2 } END { if (code) print code; else print "000" }'
return
fi
echo "000"
}
INDEX_HTML=$(fetch "$ORIGIN")
CSS_PATHS=$(printf '%s' "$INDEX_HTML" | grep -oE '/_app/immutable/assets/[^"]+\.css' | sort -u || true)
if [ -z "$CSS_PATHS" ]; then
echo "Could not find a built CSS asset on $ORIGIN" >&2
exit 1
fi
FONT_URL=""
for CSS_PATH in $CSS_PATHS; do
CSS_URL="${ORIGIN%/}${CSS_PATH}"
CSS_CONTENT=$(fetch "$CSS_URL")
FONT_PATH=$(printf '%s' "$CSS_CONTENT" | grep -oE 'inter-latin-400-normal\.[^")]+\.woff2' | head -n 1 || true)
if [ -n "$FONT_PATH" ]; then
FONT_URL="${CSS_URL%/*}/${FONT_PATH}"
break
fi
done
if [ -z "$FONT_PATH" ]; then
echo "Could not find the Inter Latin 400 woff2 asset in any built CSS asset from $ORIGIN" >&2
exit 1
fi
FONT_STATUS=$(fetch_status "$FONT_URL")
if [ "$FONT_STATUS" != "200" ]; then
echo "Inter font check failed: $FONT_URL returned HTTP $FONT_STATUS" >&2
exit 1
fi
echo "Verified Inter font asset: $FONT_URL"
'@.Replace('__ORIGIN__', $Origin)
Invoke-SshScript $fontCheckBody -Label "Inter font asset check"
if ($Logs) { if ($Logs) {
Write-Step "Recent logs (last 60 lines)" Write-Step "Recent logs (last 60 lines)"
Invoke-Ssh "cd '$RemotePath' && docker compose $ComposeArgs logs --tail=60" Invoke-Ssh "cd '$RemotePath' && docker compose $ComposeArgs logs --tail=60" -Label "docker compose logs --tail=60" -ShowOutput
} }
Write-Step "Published access"
Write-Info "Container port : $($C.Bold)$AppPort$($C.Reset)"
Write-Info "Origin : $($C.Bold)$($C.Blue)$Origin$($C.Reset)"
if ($AppPort -ne "80" -and $AppPort -ne "443") {
Write-Warn "This stack is published on port $AppPort. The public domain may still point at another service until you swap the reverse proxy or port mapping."
}
# ── Done ────────────────────────────────────────────────────────────────────
$line = '─' * 62
Write-Host "" Write-Host ""
Write-Host "Deployment complete -> https://clients.lean-101.com.au" -ForegroundColor Green Write-Host ("$($C.Green)$line$($C.Reset)")
$done = "$($Glyph.Spark) $AppName deployed $($Glyph.Arrow) $Origin"
$pad = [Math]::Max(0, 60 - ($done.Length - 2)) # subtract 2 for two-byte glyphs
Write-Host ("$($C.Green)$($C.Reset) $($C.Bold)$($C.Green)$done$($C.Reset)" + (' ' * $pad) + "$($C.Green)$($C.Reset)")
Write-Host ("$($C.Green)$line$($C.Reset)")
Write-Host ""
}
catch {
Write-Host ""
Write-Fail "Deployment aborted: $($_.Exception.Message)"
Write-Host ""
throw
} }
finally { finally {
Pop-Location Pop-Location
+4 -1
View File
@@ -27,7 +27,7 @@ server {
add_header X-Content-Type-Options "nosniff" always; add_header X-Content-Type-Options "nosniff" always;
add_header Referrer-Policy "strict-origin-when-cross-origin" always; add_header Referrer-Policy "strict-origin-when-cross-origin" always;
add_header Permissions-Policy "camera=(), microphone=(), geolocation=()" always; add_header Permissions-Policy "camera=(), microphone=(), geolocation=()" always;
add_header Content-Security-Policy "default-src 'self'; img-src 'self' data:; style-src 'self' 'unsafe-inline'; script-src 'self'; font-src 'self' data:; connect-src 'self'; frame-ancestors 'self'; base-uri 'self'; form-action 'self'" always; add_header Content-Security-Policy "default-src 'self'; img-src 'self' data:; style-src 'self' 'unsafe-inline'; script-src 'self' 'unsafe-inline'; font-src 'self' data:; connect-src 'self'; frame-ancestors 'self'; base-uri 'self'; form-action 'self'" always;
location /_app/immutable/ { location /_app/immutable/ {
expires 1y; expires 1y;
@@ -88,6 +88,9 @@ server {
} }
location / { location / {
add_header Cache-Control "no-store, no-cache, must-revalidate" always;
add_header Pragma "no-cache" always;
expires -1;
proxy_pass http://lean101_clients_frontend; proxy_pass http://lean101_clients_frontend;
proxy_http_version 1.1; proxy_http_version 1.1;
proxy_set_header Host $host; proxy_set_header Host $host;
+12 -2
View File
@@ -1,13 +1,14 @@
{ {
"name": "hunter-app", "name": "hunter-app",
"version": "1.5.6", "version": "0.1.8",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "hunter-app", "name": "hunter-app",
"version": "1.5.6", "version": "0.1.8",
"dependencies": { "dependencies": {
"@fontsource/inter": "^5.2.8",
"lucide-svelte": "^1.0.1" "lucide-svelte": "^1.0.1"
}, },
"devDependencies": { "devDependencies": {
@@ -54,6 +55,15 @@
"tslib": "^2.4.0" "tslib": "^2.4.0"
} }
}, },
"node_modules/@fontsource/inter": {
"version": "5.2.8",
"resolved": "https://registry.npmjs.org/@fontsource/inter/-/inter-5.2.8.tgz",
"integrity": "sha512-P6r5WnJoKiNVV+zvW2xM13gNdFhAEpQ9dQJHt3naLvfg+LkF2ldgSLiF4T41lf1SQCM9QmkqPTn4TH568IRagg==",
"license": "OFL-1.1",
"funding": {
"url": "https://github.com/sponsors/ayuhito"
}
},
"node_modules/@jridgewell/gen-mapping": { "node_modules/@jridgewell/gen-mapping": {
"version": "0.3.13", "version": "0.3.13",
"resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
+2 -1
View File
@@ -1,6 +1,6 @@
{ {
"name": "hunter-app", "name": "hunter-app",
"version": "1.5.6", "version": "0.1.8",
"private": true, "private": true,
"type": "module", "type": "module",
"scripts": { "scripts": {
@@ -19,6 +19,7 @@
"vitest": "^4.0.0" "vitest": "^4.0.0"
}, },
"dependencies": { "dependencies": {
"@fontsource/inter": "^5.2.8",
"lucide-svelte": "^1.0.1" "lucide-svelte": "^1.0.1"
} }
} }
+49 -3
View File
@@ -33,7 +33,13 @@ import type {
RawMaterial, RawMaterial,
RawMaterialCreateInput, RawMaterialCreateInput,
RawMaterialPriceCreateInput, RawMaterialPriceCreateInput,
Scenario Scenario,
ThroughputEntry,
ThroughputEntryCreateInput,
ThroughputEntryListParams,
ThroughputProduct,
ThroughputProductCreateInput,
ThroughputProductUpdateInput
} from '$lib/types'; } from '$lib/types';
import { getStoredAdminSession, getStoredClientSession } from '$lib/session'; import { getStoredAdminSession, getStoredClientSession } from '$lib/session';
@@ -248,12 +254,18 @@ async function request<T>(
async function requestBlob( async function requestBlob(
path: string, path: string,
options: RequestInit = {},
auth: AuthMode = 'none', auth: AuthMode = 'none',
fetcher: ApiFetch = fetch fetcher: ApiFetch = fetch
): Promise<Blob> { ): Promise<Blob> {
try { try {
const response = await fetcher(resolveRequestUrl(path, fetcher), { const response = await fetcher(resolveRequestUrl(path, fetcher), {
credentials: 'include' headers: {
'Content-Type': 'application/json',
...(options.headers ?? {})
},
credentials: 'include',
...options
}); });
if (!response.ok) { if (!response.ok) {
@@ -284,12 +296,17 @@ export const api = {
mixCalculatorSession: (sessionId: number, fetcher?: ApiFetch) => mixCalculatorSession: (sessionId: number, fetcher?: ApiFetch) =>
request<MixCalculatorSession>(`/api/mix-calculator/${sessionId}`, { method: 'GET' }, 'client', fetcher), request<MixCalculatorSession>(`/api/mix-calculator/${sessionId}`, { method: 'GET' }, 'client', fetcher),
mixCalculatorSessionPdf: (sessionId: number, fetcher?: ApiFetch) => mixCalculatorSessionPdf: (sessionId: number, fetcher?: ApiFetch) =>
requestBlob(`/api/mix-calculator/${sessionId}/pdf`, 'client', fetcher), requestBlob(`/api/mix-calculator/${sessionId}/pdf`, {}, 'client', fetcher),
previewMixCalculatorSession: (payload: MixCalculatorCreateInput) => previewMixCalculatorSession: (payload: MixCalculatorCreateInput) =>
request<MixCalculatorPreview>('/api/mix-calculator/preview', { request<MixCalculatorPreview>('/api/mix-calculator/preview', {
method: 'POST', method: 'POST',
body: JSON.stringify(payload) body: JSON.stringify(payload)
}, 'client'), }, 'client'),
previewMixCalculatorPdf: (payload: MixCalculatorCreateInput) =>
requestBlob('/api/mix-calculator/preview/pdf', {
method: 'POST',
body: JSON.stringify(payload)
}, 'client'),
createMixCalculatorSession: (payload: MixCalculatorCreateInput) => createMixCalculatorSession: (payload: MixCalculatorCreateInput) =>
request<MixCalculatorSession>('/api/mix-calculator', { request<MixCalculatorSession>('/api/mix-calculator', {
method: 'POST', method: 'POST',
@@ -304,6 +321,35 @@ export const api = {
productCosts: (fetcher?: ApiFetch) => productCosts: (fetcher?: ApiFetch) =>
cachedFetchJson<ProductCostBreakdown[]>('/api/powerbi/product-costs', mockCosts, 'client', fetcher), cachedFetchJson<ProductCostBreakdown[]>('/api/powerbi/product-costs', mockCosts, 'client', fetcher),
scenarios: (fetcher?: ApiFetch) => cachedFetchJson<Scenario[]>('/api/scenarios', mockScenarios, 'client', fetcher), scenarios: (fetcher?: ApiFetch) => cachedFetchJson<Scenario[]>('/api/scenarios', mockScenarios, 'client', fetcher),
throughputProducts: (fetcher?: ApiFetch) =>
cachedFetchJson<ThroughputProduct[]>('/api/throughput/products', [], 'client', fetcher),
throughputEntries: (params?: ThroughputEntryListParams, fetcher?: ApiFetch) => {
const search = new URLSearchParams();
if (params?.date_from) search.set('date_from', params.date_from);
if (params?.date_to) search.set('date_to', params.date_to);
if (params?.product_id != null) search.set('product_id', String(params.product_id));
if (params?.staff_name) search.set('staff_name', params.staff_name);
if (params?.quantity_type) search.set('quantity_type', params.quantity_type);
if (params?.limit) search.set('limit', String(params.limit));
const qs = search.toString();
const path = qs ? `/api/throughput/entries?${qs}` : '/api/throughput/entries';
return cachedFetchJson<ThroughputEntry[]>(path, [], 'client', fetcher);
},
createThroughputEntry: (payload: ThroughputEntryCreateInput) =>
request<ThroughputEntry>('/api/throughput/entries', {
method: 'POST',
body: JSON.stringify(payload)
}, 'client'),
createThroughputProduct: (payload: ThroughputProductCreateInput) =>
request<ThroughputProduct>('/api/throughput/products', {
method: 'POST',
body: JSON.stringify(payload)
}, 'client'),
updateThroughputProduct: (productId: number, payload: ThroughputProductUpdateInput) =>
request<ThroughputProduct>(`/api/throughput/products/${productId}`, {
method: 'PATCH',
body: JSON.stringify(payload)
}, 'client'),
clientAccess: (fetcher?: ApiFetch) => cachedFetchJson<ClientAccessAccount[]>('/api/client-access', mockClientAccess, 'manager', fetcher), clientAccess: (fetcher?: ApiFetch) => cachedFetchJson<ClientAccessAccount[]>('/api/client-access', mockClientAccess, 'manager', fetcher),
clientAccessExport: (fetcher?: ApiFetch) => clientAccessExport: (fetcher?: ApiFetch) =>
cachedFetchJson<ClientAccessPowerBiExport>('/api/powerbi/client-access', mockClientAccessExport, 'manager', fetcher), cachedFetchJson<ClientAccessPowerBiExport>('/api/powerbi/client-access', mockClientAccessExport, 'manager', fetcher),
+37 -11
View File
@@ -21,6 +21,7 @@
canOpenReporting as sessionCanOpenReporting, canOpenReporting as sessionCanOpenReporting,
canOpenScenarios as sessionCanOpenScenarios, canOpenScenarios as sessionCanOpenScenarios,
canOpenSettings as sessionCanOpenSettings, canOpenSettings as sessionCanOpenSettings,
canOpenThroughput as sessionCanOpenThroughput,
canUseWorkspaceSearch as sessionCanUseWorkspaceSearch, canUseWorkspaceSearch as sessionCanUseWorkspaceSearch,
getWorkspaceRole, getWorkspaceRole,
getWorkspaceHomeHref as sessionWorkspaceHomeHref, getWorkspaceHomeHref as sessionWorkspaceHomeHref,
@@ -36,6 +37,7 @@
mixCalculatorItem, mixCalculatorItem,
pageTitle, pageTitle,
reportingItem, reportingItem,
throughputItem,
type FooterLink, type FooterLink,
type SearchItem, type SearchItem,
type NavItem, type NavItem,
@@ -67,7 +69,6 @@
let seededSearchKey = $state<string | null>(null); let seededSearchKey = $state<string | null>(null);
let paletteInput: HTMLInputElement | null = $state(null); let paletteInput: HTMLInputElement | null = $state(null);
const appVersion = `v${packageInfo.version}`; const appVersion = `v${packageInfo.version}`;
const releaseStage = 'Beta';
const currentYear = new Date().getFullYear(); const currentYear = new Date().getFullYear();
const canOpenDashboard = $derived(sessionCanOpenDashboard($clientSession)); const canOpenDashboard = $derived(sessionCanOpenDashboard($clientSession));
const canOpenRawMaterials = $derived(sessionCanOpenRawMaterials($clientSession)); const canOpenRawMaterials = $derived(sessionCanOpenRawMaterials($clientSession));
@@ -85,7 +86,9 @@
const routeGuardPending = $derived(!!$clientSession && (isRestoringSession || !currentRouteAllowed)); const routeGuardPending = $derived(!!$clientSession && (isRestoringSession || !currentRouteAllowed));
const shellPathname = $derived(routeGuardPending ? workspaceHomeHref : page.url.pathname); const shellPathname = $derived(routeGuardPending ? workspaceHomeHref : page.url.pathname);
const shellTitle = $derived(routeGuardPending ? 'Loading Workspace' : pageTitle(page.url.pathname)); const shellTitle = $derived(routeGuardPending ? 'Loading Workspace' : pageTitle(page.url.pathname));
const shellBreadcrumbs = $derived(routeGuardPending ? clientBreadcrumbs(workspaceHomeHref) : clientBreadcrumbs(page.url.pathname)); const shellBreadcrumbs = $derived(
routeGuardPending ? clientBreadcrumbs(workspaceHomeHref, $clientSession) : clientBreadcrumbs(page.url.pathname, $clientSession)
);
const visibleDashboardItem = $derived(canOpenDashboard ? dashboardItem : null); const visibleDashboardItem = $derived(canOpenDashboard ? dashboardItem : null);
const visibleWorkingDocumentItems = $derived( const visibleWorkingDocumentItems = $derived(
!$clientSession !$clientSession
@@ -99,6 +102,8 @@
}) })
); );
const visibleMixCalculatorItem = $derived(canOpenMixCalculator ? mixCalculatorItem : null); const visibleMixCalculatorItem = $derived(canOpenMixCalculator ? mixCalculatorItem : null);
const canOpenThroughput = $derived(sessionCanOpenThroughput($clientSession));
const visibleThroughputItem = $derived(canOpenThroughput ? throughputItem : null);
const visibleReportingItem = $derived(sessionCanOpenReporting($clientSession) ? reportingItem : null); const visibleReportingItem = $derived(sessionCanOpenReporting($clientSession) ? reportingItem : null);
const isOperationsUser = $derived($clientSession?.role_name === 'Operations'); const isOperationsUser = $derived($clientSession?.role_name === 'Operations');
const workspaceRole = $derived(getWorkspaceRole($clientSession)); const workspaceRole = $derived(getWorkspaceRole($clientSession));
@@ -125,7 +130,6 @@
if (item.href === '/mixes') return canOpenMixMaster; if (item.href === '/mixes') return canOpenMixMaster;
if (item.href === '/mixes/new') return canCreateMixWorksheet; if (item.href === '/mixes/new') return canCreateMixWorksheet;
if (item.href === '/mix-calculator') return canOpenMixCalculator; if (item.href === '/mix-calculator') return canOpenMixCalculator;
if (item.href === '/mix-calculator/new') return canCreateMixSession;
if (item.href === '/products') return canOpenProducts; if (item.href === '/products') return canOpenProducts;
if (item.href === '/reporting') return sessionCanOpenReporting($clientSession); if (item.href === '/reporting') return sessionCanOpenReporting($clientSession);
if (item.href === '/settings') return canOpenSettings; if (item.href === '/settings') return canOpenSettings;
@@ -390,13 +394,13 @@
primaryItems={[ primaryItems={[
...(visibleDashboardItem ? [visibleDashboardItem] : []), ...(visibleDashboardItem ? [visibleDashboardItem] : []),
...(visibleMixCalculatorItem ? [visibleMixCalculatorItem] : []), ...(visibleMixCalculatorItem ? [visibleMixCalculatorItem] : []),
...(visibleThroughputItem ? [visibleThroughputItem] : []),
...(visibleReportingItem ? [visibleReportingItem] : []) ...(visibleReportingItem ? [visibleReportingItem] : [])
]} ]}
brandHref={workspaceHomeHref} brandHref={workspaceHomeHref}
workingDocumentItems={visibleWorkingDocumentItems} workingDocumentItems={visibleWorkingDocumentItems}
footerItems={visibleFooterLinks} footerItems={visibleFooterLinks}
{appVersion} {appVersion}
{releaseStage}
{currentYear} {currentYear}
{canOpenSettings} {canOpenSettings}
onOpenSettings={openSettings} onOpenSettings={openSettings}
@@ -449,10 +453,10 @@
<a href="/mixes/new">Create mix worksheet</a> <a href="/mixes/new">Create mix worksheet</a>
{/if} {/if}
{#if canOpenMixCalculator} {#if canOpenMixCalculator}
<a href={featureFlags.mixCalculatorSessionHistory ? '/mix-calculator' : '/mix-calculator/new'}>Open mix calculator</a> <a href="/mix-calculator">Open mix calculator</a>
{/if} {/if}
{#if canCreateMixSession} {#if canCreateMixSession}
<a href="/mix-calculator/new">Create mix session</a> <a href="/mix-calculator">Create mix session</a>
{/if} {/if}
{#if canOpenProducts} {#if canOpenProducts}
<a href="/products">Review delivered pricing</a> <a href="/products">Review delivered pricing</a>
@@ -540,6 +544,15 @@
</a> </a>
{/if} {/if}
{#if visibleThroughputItem}
{@const Icon = visibleThroughputItem.icon}
<a class:active={matchesRoute(visibleThroughputItem.href, page.url.pathname)} href={visibleThroughputItem.href} onclick={() => (navOpen = false)}>
<span class="nav-icon"><Icon size={18} strokeWidth={1.75} /></span>
<span>{visibleThroughputItem.label}</span>
{#if visibleThroughputItem.badge}<span class="drawer-badge">{visibleThroughputItem.badge}</span>{/if}
</a>
{/if}
{#if visibleReportingItem} {#if visibleReportingItem}
{@const Icon = visibleReportingItem.icon} {@const Icon = visibleReportingItem.icon}
<a class:active={matchesRoute(visibleReportingItem.href, page.url.pathname)} href={visibleReportingItem.href} onclick={() => (navOpen = false)}> <a class:active={matchesRoute(visibleReportingItem.href, page.url.pathname)} href={visibleReportingItem.href} onclick={() => (navOpen = false)}>
@@ -569,7 +582,7 @@
</a> </a>
{/if} {/if}
{#if canCreateMixSession} {#if canCreateMixSession}
<a href="/mix-calculator/new" onclick={() => (navOpen = false)}> <a href="/mix-calculator" onclick={() => (navOpen = false)}>
<span class="nav-icon"><Calculator size={18} strokeWidth={1.75} /></span> <span class="nav-icon"><Calculator size={18} strokeWidth={1.75} /></span>
<span>Create mix session</span> <span>Create mix session</span>
</a> </a>
@@ -659,8 +672,6 @@
{/if} {/if}
<style> <style>
@import url('https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&display=swap');
:global(:root) { :global(:root) {
/* ── Brand ──────────────────────────────────────────────── */ /* ── Brand ──────────────────────────────────────────────── */
--color-brand: #15803d; --color-brand: #15803d;
@@ -705,7 +716,7 @@
min-height: 100%; min-height: 100%;
background: var(--color-bg-app); background: var(--color-bg-app);
color: var(--color-text-primary); color: var(--color-text-primary);
font-family: Inter, "Segoe UI", sans-serif; font-family: "Inter", "Segoe UI", sans-serif;
font-size: 14px; font-size: 14px;
} }
@@ -714,7 +725,7 @@
} }
:global(h1, h2, h3, h4, h5, h6) { :global(h1, h2, h3, h4, h5, h6) {
font-family: Inter, "Segoe UI", sans-serif; font-family: "Inter", "Segoe UI", sans-serif;
letter-spacing: -0.03em; letter-spacing: -0.03em;
} }
@@ -734,6 +745,7 @@
display: grid; display: grid;
grid-template-columns: 252px minmax(0, 1fr); grid-template-columns: 252px minmax(0, 1fr);
min-height: 100vh; min-height: 100vh;
background: var(--color-bg-app);
} }
.signed-out-shell { .signed-out-shell {
@@ -954,6 +966,7 @@
min-height: 100vh; min-height: 100vh;
height: 100vh; height: 100vh;
overflow: hidden; overflow: hidden;
background: var(--color-bg-app);
} }
.workspace-label { .workspace-label {
@@ -1069,6 +1082,7 @@
min-width: 0; min-width: 0;
padding: var(--content-padding); padding: var(--content-padding);
overflow: auto; overflow: auto;
background: var(--color-bg-app);
} }
.locked-card { .locked-card {
@@ -1346,6 +1360,18 @@
background: var(--green-soft); background: var(--green-soft);
} }
.drawer-badge {
margin-left: auto;
padding: 0.08rem 0.4rem;
border-radius: 999px;
background: #fdf0d2;
color: #8a5a00;
font-size: 0.62rem;
font-weight: 700;
letter-spacing: 0.06em;
text-transform: uppercase;
}
.drawer-group { .drawer-group {
display: grid; display: grid;
gap: 0.4rem; gap: 0.4rem;
@@ -4,7 +4,7 @@
let { let {
session, session,
generatedAt = null, generatedAt = null,
showGeneratedStamp = true showGeneratedStamp = false
}: { }: {
session: MixCalculatorPreview | MixCalculatorSession; session: MixCalculatorPreview | MixCalculatorSession;
generatedAt?: string | null; generatedAt?: string | null;
@@ -35,111 +35,66 @@
const sessionNumber = $derived(hasSessionNumber(session) ? session.session_number : null); const sessionNumber = $derived(hasSessionNumber(session) ? session.session_number : null);
const issuedAt = $derived(generatedAt ?? new Date().toISOString()); const issuedAt = $derived(generatedAt ?? new Date().toISOString());
const blendTotal = $derived(session.lines.reduce((sum, line) => sum + line.mix_percentage, 0));
</script> </script>
<article class="print-document"> <article class="print-document">
<header class="hero"> <header class="document-header">
<div class="hero-copy"> <img class="brand-mark" src="/logo-hsf.png" alt="Hunter Stock Feeds" />
<div class="hero-kicker">
<span>Mix Calculator</span> <div class="title-block">
{#if sessionNumber} <p class="eyebrow">Calculated Output</p>
<strong>{sessionNumber}</strong>
{/if}
</div>
<h1>{session.product_name}</h1> <h1>{session.product_name}</h1>
<p>{session.client_name} · {session.mix_name}</p> <p class="subtitle">Snapshot of the scaled raw material requirements.</p>
</div> </div>
<div class="hero-side"> <div class="session-block">
<div> {#if sessionNumber}
<span>Mix date</span> <span>Session {sessionNumber}</span>
<strong>{formatDate(session.mix_date)}</strong> {/if}
</div> {#if showGeneratedStamp}
<div> <span>Generated {formatTimestamp(issuedAt)}</span>
<span>Prepared by</span> {/if}
<strong>{session.prepared_by_name}</strong>
</div>
<div>
<span>Status</span>
<strong>{session.status}</strong>
</div>
</div> </div>
</header> </header>
<section class="summary-band" aria-label="Session summary"> <section class="meta-grid" aria-label="Print summary">
<article> <div><span>Total kg</span><strong>{formatNumber(session.total_kg, 2)}</strong><small>Scaled batch size</small></div>
<span>Batch size</span> <div><span>Total bags</span><strong>{formatNumber(session.total_bags, 2)}</strong><small>{session.product_unit_of_measure}</small></div>
<strong>{formatNumber(session.batch_size_kg, 2)}kg</strong> <div><span>Prepared by</span><strong>{session.prepared_by_name}</strong><small>{formatDate(session.mix_date)}</small></div>
</article> <div><span>Client</span><strong>{session.client_name}</strong></div>
<article> <div><span>Product</span><strong>{session.product_name}</strong></div>
<span>Total output</span> <div><span>Mix source</span><strong>{session.mix_name}</strong></div>
<strong>{formatNumber(session.total_kg, 2)}kg</strong> <div><span>Unit size</span><strong>{formatNumber(session.product_unit_size_kg, 2)}kg</strong></div>
</article> <div><span>Batch size</span><strong>{formatNumber(session.batch_size_kg, 2)}kg</strong></div>
<article>
<span>Bags</span>
<strong>{formatNumber(session.total_bags, 2)}</strong>
</article>
<article>
<span>Unit pack</span>
<strong>{formatNumber(session.product_unit_size_kg, 2)}kg</strong>
</article>
</section>
<section class="detail-grid">
<article class="detail-card">
<span>Mix source</span>
<strong>{session.mix_name}</strong>
<p>Saved against {session.product_unit_of_measure} units.</p>
</article>
<article class="detail-card">
<span>Composition</span>
<strong>{formatNumber(blendTotal, 2)}%</strong>
<p>{session.lines.length} raw material{session.lines.length === 1 ? '' : 's'} in the blend.</p>
</article>
{#if showGeneratedStamp}
<article class="detail-card">
<span>Generated</span>
<strong>{formatTimestamp(issuedAt)}</strong>
<p>Prepared for print or PDF export.</p>
</article>
{/if}
</section> </section>
{#if session.notes} {#if session.notes}
<section class="callout notes"> <section class="inline-note">
<span>Notes</span> <span>Notes</span>
<p>{session.notes}</p> <p>{session.notes}</p>
</section> </section>
{/if} {/if}
{#if session.warnings.length} {#if session.warnings.length}
<section class="callout warning"> <section class="inline-note warning">
<span>Warnings</span> <span>Warnings</span>
<ul> <p>{session.warnings.join(' | ')}</p>
{#each session.warnings as warning}
<li>{warning}</li>
{/each}
</ul>
</section> </section>
{/if} {/if}
<section class="composition-card"> <section class="composition-card">
<div class="section-heading"> <div class="section-heading">
<div> <div>
<span>Required Raw Materials</span> <span>Raw Material</span>
<h2>Blend composition</h2>
</div> </div>
<p>{session.product_unit_of_measure} · {formatNumber(session.product_unit_size_kg, 2)}kg per unit</p> <p>Required kg</p>
</div> </div>
<table> <table>
<thead> <thead>
<tr> <tr>
<th>Raw material</th> <th>Raw material</th>
<th>Mix %</th>
<th>Required kg</th> <th>Required kg</th>
<th>Unit</th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@@ -148,9 +103,7 @@
<td> <td>
<strong>{line.raw_material_name}</strong> <strong>{line.raw_material_name}</strong>
</td> </td>
<td>{formatNumber(line.mix_percentage, 2)}%</td>
<td>{formatNumber(line.required_kg, 2)}kg</td> <td>{formatNumber(line.required_kg, 2)}kg</td>
<td>{line.unit}</td>
</tr> </tr>
{/each} {/each}
</tbody> </tbody>
@@ -162,232 +115,204 @@
:global(:root) { :global(:root) {
--print-page-width: 210mm; --print-page-width: 210mm;
--print-page-height: 297mm; --print-page-height: 297mm;
--print-page-padding-x: 14mm; --print-page-padding-x: 8mm;
--print-page-padding-y: 15mm; --print-page-padding-y: 14mm;
} }
h1, h1,
h2, h2,
p, p {
ul {
margin: 0; margin: 0;
} }
.print-document { .print-document {
display: grid; display: grid;
gap: 1.4rem; gap: 0.6rem;
width: min(100%, var(--print-page-width)); width: min(100%, var(--print-page-width));
min-height: var(--print-page-height); height: var(--print-page-height);
margin: 0 auto; margin: 0 auto;
padding: var(--print-page-padding-y) var(--print-page-padding-x); padding: var(--print-page-padding-y) var(--print-page-padding-x);
border: 1px solid #dbe4de; border: 1px solid #000;
border-radius: 0.8rem; background: #fff;
background: color: #000;
radial-gradient(circle at top right, rgba(21, 128, 61, 0.08), transparent 22rem), box-shadow: none;
linear-gradient(180deg, #fff 0%, #fbfcfb 100%); overflow: hidden;
color: #21312a; box-sizing: border-box;
box-shadow:
0 28px 48px rgba(21, 33, 26, 0.08),
0 0 0 1px rgba(219, 228, 222, 0.55);
} }
.hero { .document-header {
display: grid; display: grid;
grid-template-columns: minmax(0, 1fr) 17rem; grid-template-columns: auto 1fr auto;
gap: 1.25rem;
align-items: start; align-items: start;
padding-bottom: 1.3rem; gap: 0.75rem;
border-bottom: 1px solid color-mix(in srgb, var(--line) 88%, #dce7df); padding-bottom: 0.75rem;
border-bottom: 1px solid #000;
} }
.hero-kicker { .brand-mark {
display: inline-flex; width: 32mm;
align-items: center; height: auto;
gap: 0.65rem; object-fit: contain;
margin-bottom: 0.75rem; filter: grayscale(1) contrast(1.2);
color: #62736b;
font-size: 0.76rem;
font-weight: 700;
letter-spacing: 0.12em;
text-transform: uppercase;
} }
.hero-kicker strong { .eyebrow,
padding: 0.36rem 0.55rem; .meta-grid span,
border: 1px solid #d5dfd9; .inline-note span,
border-radius: 999px; .section-heading span,
color: #214233;
font-size: 0.68rem;
letter-spacing: 0.08em;
}
.hero h1 {
max-width: 11ch;
font-size: clamp(2rem, 4vw, 3.3rem);
line-height: 0.96;
letter-spacing: -0.05em;
}
.hero-copy p,
.section-heading p,
.detail-card p {
color: #6b7a73;
}
.hero-copy p {
margin-top: 0.7rem;
font-size: 1rem;
}
.hero-side,
.summary-band,
.detail-grid {
display: grid;
gap: 0.8rem;
}
.hero-side div,
.summary-band article,
.detail-card {
display: grid;
gap: 0.25rem;
}
.hero-side span,
.summary-band span,
.detail-card span,
.callout span,
th, th,
.section-heading span { .session-block span {
color: #6b7a73; color: #000;
font-size: 0.72rem; font-size: 0.6rem;
font-weight: 700; font-weight: 700;
letter-spacing: 0.12em; letter-spacing: 0.09em;
text-transform: uppercase; text-transform: uppercase;
} }
.hero-side strong, .title-block {
.detail-card strong { min-width: 0;
font-size: 1rem;
} }
.summary-band { .title-block h1 {
grid-template-columns: repeat(4, minmax(0, 1fr)); font-size: 1.2rem;
line-height: 1.05;
letter-spacing: -0.03em;
} }
.summary-band article { .subtitle {
min-height: 6.2rem; margin-top: 0.12rem;
padding: 1rem 1.05rem; color: #000;
border: 1px solid #dfe7e2; font-size: 0.72rem;
border-radius: 1.2rem;
background: rgba(255, 255, 255, 0.92);
} }
.summary-band strong { .session-block {
margin-top: auto;
font-size: clamp(1.4rem, 2.4vw, 2rem);
letter-spacing: -0.04em;
}
.detail-grid {
grid-template-columns: repeat(3, minmax(0, 1fr));
}
.detail-card {
min-height: 7rem;
padding: 1rem 1.05rem;
border-radius: 1.15rem;
background: #f6f9f7;
}
.callout {
display: grid; display: grid;
gap: 0.55rem; justify-items: end;
padding: 1rem 1.1rem; gap: 0.18rem;
border-radius: 1.15rem; text-align: right;
break-inside: avoid;
} }
.callout.notes { .meta-grid {
background: #f6f9f7; display: grid;
border: 1px solid #dfe7e2; grid-template-columns: repeat(4, minmax(0, 1fr));
gap: 0.38rem;
} }
.callout.warning { .meta-grid div {
background: #fff7ea; display: grid;
border: 1px solid #f0cf97; gap: 0.08rem;
color: #82561b; min-height: 2.1rem;
padding: 0.34rem 0.42rem;
border: 1px solid #000;
background: #fff;
} }
.callout ul { .meta-grid strong {
padding-left: 1rem; font-size: 0.84rem;
line-height: 1.1;
}
.meta-grid small {
font-size: 0.64rem;
line-height: 1.15;
}
.inline-note {
display: grid;
gap: 0.12rem;
padding: 0.35rem 0.42rem;
border: 1px solid #000;
background: #fff;
}
.inline-note p {
font-size: 0.68rem;
line-height: 1.25;
overflow: hidden;
}
.inline-note.warning {
border-color: #000;
background: #fff;
color: #000;
} }
.composition-card { .composition-card {
display: grid; display: grid;
gap: 0.9rem; gap: 0.35rem;
break-inside: avoid; min-height: 0;
} }
.section-heading { .section-heading {
display: flex; display: flex;
align-items: end;
justify-content: space-between; justify-content: space-between;
gap: 1rem; align-items: end;
gap: 0.5rem;
} }
.section-heading h2 { .section-heading h2 {
margin-top: 0.32rem; margin-top: 0.05rem;
font-size: 1.5rem; font-size: 0.95rem;
letter-spacing: -0.04em; line-height: 1.05;
}
.section-heading p {
color: #000;
font-size: 0.66rem;
white-space: nowrap;
} }
table { table {
width: 100%; width: 100%;
border-collapse: collapse; border-collapse: collapse;
background: rgba(255, 255, 255, 0.8); table-layout: fixed;
border: 1px solid #dfe7e2; border: 1px solid #000;
border-radius: 1.2rem; background: #fff;
overflow: hidden;
} }
th, th,
td { td {
padding: 0.95rem 1rem; padding: 0.28rem 0.42rem;
text-align: left; text-align: left;
border-bottom: 1px solid #e6ede9; border-bottom: 1px solid #000;
line-height: 1.15;
} }
thead { th:last-child,
display: table-header-group; td:last-child {
width: 32mm;
} }
tr,
td,
th { th {
break-inside: avoid; background: #fff;
font-size: 0.6rem;
}
td {
font-size: 0.7rem;
vertical-align: top;
}
td strong {
font-size: 0.72rem;
font-weight: 700;
color: #000;
} }
tbody tr:last-child td { tbody tr:last-child td {
border-bottom: none; border-bottom: none;
} }
td strong { tr,
font-size: 0.98rem; td,
font-weight: 700; th,
color: #203128; section {
break-inside: avoid;
} }
@media (max-width: 900px) { @media (max-width: 900px) {
.hero, .meta-grid {
.summary-band, grid-template-columns: repeat(2, minmax(0, 1fr));
.detail-grid {
grid-template-columns: 1fr;
}
.section-heading {
flex-direction: column;
align-items: start;
} }
} }
@@ -395,7 +320,7 @@
:global(html), :global(html),
:global(body) { :global(body) {
width: var(--print-page-width); width: var(--print-page-width);
min-height: var(--print-page-height); height: var(--print-page-height);
margin: 0; margin: 0;
background: #fff; background: #fff;
print-color-adjust: exact; print-color-adjust: exact;
@@ -404,28 +329,13 @@
.print-document { .print-document {
width: var(--print-page-width); width: var(--print-page-width);
min-height: var(--print-page-height); height: var(--print-page-height);
margin: 0; margin: 0;
padding: var(--print-page-padding-y) var(--print-page-padding-x); padding: var(--print-page-padding-y) var(--print-page-padding-x);
border: none; border: none;
border-radius: 0;
background: #fff;
color: #1e2622;
box-shadow: none; box-shadow: none;
} }
.summary-band article,
.detail-card,
table,
.callout {
border-color: #d5ddd8;
background: #fff;
}
.callout.warning {
background: #fff8ef;
}
@page { @page {
size: A4 portrait; size: A4 portrait;
margin: 0; margin: 0;
@@ -9,6 +9,13 @@
`MixCalculator_${session.client_name}_${session.product_name}_${session.mix_date}_${session.session_number}`.replace(/[^\w.-]+/g, '_') `MixCalculator_${session.client_name}_${session.product_name}_${session.mix_date}_${session.session_number}`.replace(/[^\w.-]+/g, '_')
); );
async function openPdf() {
const blob = await api.mixCalculatorSessionPdf(session.id);
const url = URL.createObjectURL(blob);
window.open(url, '_blank', 'noopener,noreferrer');
setTimeout(() => URL.revokeObjectURL(url), 60_000);
}
async function downloadPdf() { async function downloadPdf() {
const blob = await api.mixCalculatorSessionPdf(session.id); const blob = await api.mixCalculatorSessionPdf(session.id);
const url = URL.createObjectURL(blob); const url = URL.createObjectURL(blob);
@@ -29,8 +36,8 @@
<section class="print-page"> <section class="print-page">
<div class="print-toolbar"> <div class="print-toolbar">
<a class="secondary-button" href={`/mix-calculator/${session.id}`}>Back to session</a> <a class="secondary-button" href={`/mix-calculator/${session.id}`}>Back to session</a>
<button class="primary-button" type="button" onclick={openPdf}>Open Styled PDF</button>
<button class="secondary-button" type="button" onclick={downloadPdf}>Download PDF</button> <button class="secondary-button" type="button" onclick={downloadPdf}>Download PDF</button>
<button class="primary-button" type="button" onclick={() => window.print()}>Print / Save PDF</button>
</div> </div>
<MixCalculatorPrintDocument session={session} generatedAt={new Date().toISOString()} /> <MixCalculatorPrintDocument session={session} generatedAt={new Date().toISOString()} />
@@ -1,5 +1,4 @@
<script lang="ts"> <script lang="ts">
import { goto } from '$app/navigation';
import { api } from '$lib/api'; import { api } from '$lib/api';
import MixCalculatorPrintDocument from '$lib/components/MixCalculatorPrintDocument.svelte'; import MixCalculatorPrintDocument from '$lib/components/MixCalculatorPrintDocument.svelte';
import { featureFlags } from '$lib/features'; import { featureFlags } from '$lib/features';
@@ -11,7 +10,6 @@
MixCalculatorPreview, MixCalculatorPreview,
MixCalculatorSession MixCalculatorSession
} from '$lib/types'; } from '$lib/types';
import MixCalculatorPreviewModal from './MixCalculatorPreviewModal.svelte';
import MixCalculatorResultsPanel from './MixCalculatorResultsPanel.svelte'; import MixCalculatorResultsPanel from './MixCalculatorResultsPanel.svelte';
let { options, initialSession = null }: { options: MixCalculatorOptions; initialSession?: MixCalculatorSession | null } = $props(); let { options, initialSession = null }: { options: MixCalculatorOptions; initialSession?: MixCalculatorSession | null } = $props();
@@ -56,11 +54,8 @@
let formError = $state(''); let formError = $state('');
let formHint = $state('Select a mix date and prepared by name, then choose a client to unlock products.'); let formHint = $state('Select a mix date and prepared by name, then choose a client to unlock products.');
let previewLoading = $state(false); let previewLoading = $state(false);
let saveLoading = $state(false);
let previewModalOpen = $state(false);
const canEdit = $derived(hasModuleAccess($clientSession, 'mix_calculator', 'edit')); const canEdit = $derived(hasModuleAccess($clientSession, 'mix_calculator', 'edit'));
const isExistingSession = $derived(initialSession !== null);
const availableClients = $derived( const availableClients = $derived(
Array.from(new Set([...(options.clients ?? []), ...(initialSession ? [initialSession.client_name] : [])])) Array.from(new Set([...(options.clients ?? []), ...(initialSession ? [initialSession.client_name] : [])]))
); );
@@ -198,20 +193,21 @@
formHint = `Ready to calculate ${selectedProduct?.product_name ?? 'the selected product'}.`; formHint = `Ready to calculate ${selectedProduct?.product_name ?? 'the selected product'}.`;
}); });
function printPreview() { async function downloadPdf() {
if (typeof window !== 'undefined') {
window.print();
}
}
async function downloadSessionPdf(sessionId: number) {
const tid = toast.loading('Generating PDF…'); const tid = toast.loading('Generating PDF…');
try { try {
const blob = await api.mixCalculatorSessionPdf(sessionId); const payload = buildPayload();
if (!payload) {
toast.dismiss(tid);
toast.error(formError || 'Complete the mix details first.');
return;
}
const blob = await api.previewMixCalculatorPdf(payload);
const url = URL.createObjectURL(blob); const url = URL.createObjectURL(blob);
const anchor = document.createElement('a'); const anchor = document.createElement('a');
anchor.href = url; anchor.href = url;
anchor.download = `mix_calculator_${sessionId}.pdf`; anchor.download = `mix_calculator_${payload.client_name}_${payload.mix_date}.pdf`.replace(/[^\w.-]+/g, '_');
document.body.appendChild(anchor); document.body.appendChild(anchor);
anchor.click(); anchor.click();
anchor.remove(); anchor.remove();
@@ -223,43 +219,27 @@
} }
} }
function openPreviewModal() { async function openPdf() {
if (!preview) { const tid = toast.loading('Opening styled PDF…');
return; try {
}
previewModalOpen = true;
}
function closePreviewModal() {
previewModalOpen = false;
}
async function saveSession(mode: 'update' | 'create', destination: 'detail' | 'print' = 'detail') {
const payload = buildPayload(); const payload = buildPayload();
if (!payload) { if (!payload) {
toast.dismiss(tid);
toast.error(formError || 'Complete the mix details first.');
return; return;
} }
saveLoading = true; const blob = await api.previewMixCalculatorPdf(payload);
const tid = toast.loading(mode === 'update' ? 'Saving session…' : 'Creating session…'); const url = URL.createObjectURL(blob);
try { window.open(url, '_blank', 'noopener,noreferrer');
const saved = setTimeout(() => URL.revokeObjectURL(url), 60_000);
mode === 'update' && initialSession
? await api.updateMixCalculatorSession(initialSession.id, payload)
: await api.createMixCalculatorSession(payload);
toast.dismiss(tid); toast.dismiss(tid);
toast.success(mode === 'update' ? 'Session saved' : 'Session created');
const target = destination === 'print' ? `/mix-calculator/${saved.id}/print` : `/mix-calculator/${saved.id}`;
await goto(target);
} catch (error) { } catch (error) {
toast.dismiss(tid); toast.dismiss(tid);
toast.error(error instanceof Error ? error.message : 'Unable to save the mix calculator session.'); toast.error(error instanceof Error ? error.message : 'Unable to open the styled PDF.');
formError = error instanceof Error ? error.message : 'Unable to save the mix calculator session.';
saveLoading = false;
} }
} }
</script> </script>
{#if !canEdit && !initialSession} {#if !canEdit && !initialSession}
@@ -278,8 +258,9 @@
<a class="secondary-button" href="/mix-calculator">Session history</a> <a class="secondary-button" href="/mix-calculator">Session history</a>
{/if} {/if}
{#if initialSession} {#if initialSession}
<a class="secondary-button" href={`/mix-calculator/${initialSession.id}/print`}>Printable view</a> <a class="secondary-button" href={`/mix-calculator/${initialSession.id}/print`}>Open PDF page</a>
<button class="secondary-button" type="button" onclick={() => downloadSessionPdf(initialSession.id)}>Download PDF</button> <button class="primary-button" type="button" onclick={openPdf}>Open PDF in new tab</button>
<button class="secondary-button" type="button" onclick={downloadPdf}>Download PDF</button>
{/if} {/if}
</section> </section>
{/if} {/if}
@@ -364,38 +345,12 @@
{#if canEdit} {#if canEdit}
<div class="action-row"> <div class="action-row">
<button class="primary-button" disabled={previewLoading || saveLoading} type="button" onclick={calculatePreview}> <button class="primary-button" disabled={previewLoading} type="button" onclick={calculatePreview}>
<span class="button-icon" style="--button-icon-url: url('/icons/calculator.svg');" aria-hidden="true"></span> <span class="button-icon" style="--button-icon-url: url('/icons/calculator.svg');" aria-hidden="true"></span>
<span>{previewLoading ? 'Calculating...' : 'Calculate mix'}</span> <span>{previewLoading ? 'Calculating...' : 'Calculate mix'}</span>
</button> </button>
{#if featureFlags.mixCalculatorSessionSave} <button class="danger-button" disabled={previewLoading} type="button" onclick={clearForm}>
<button class="secondary-button" disabled={saveLoading || previewLoading} type="button" onclick={() => saveSession(isExistingSession ? 'update' : 'create')}>
{saveLoading ? 'Saving...' : isExistingSession ? 'Save changes' : 'Save session'}
</button>
<button class="secondary-button" disabled={saveLoading || previewLoading} type="button" onclick={() => saveSession(isExistingSession ? 'update' : 'create', 'print')}>
<span class="button-icon" style="--button-icon-url: url('/icons/print.svg');" aria-hidden="true"></span>
<span>{saveLoading ? 'Saving...' : 'Save & print'}</span>
</button>
{#if initialSession}
<button class="secondary-button" disabled={saveLoading || previewLoading} type="button" onclick={() => saveSession('create')}>
Save as new
</button>
{/if}
{:else}
<button class="secondary-button" disabled={previewLoading || saveLoading || !preview} type="button" onclick={openPreviewModal}>
<span>Preview</span>
</button>
<button class="secondary-button" disabled={previewLoading || saveLoading || !preview} type="button" onclick={printPreview}>
<span class="button-icon" style="--button-icon-url: url('/icons/print.svg');" aria-hidden="true"></span>
<span>Print</span>
</button>
{/if}
<button class="secondary-button" disabled={previewLoading || saveLoading} type="button" onclick={clearForm}>
<span class="button-icon" style="--button-icon-url: url('/icons/trash.svg');" aria-hidden="true"></span> <span class="button-icon" style="--button-icon-url: url('/icons/trash.svg');" aria-hidden="true"></span>
<span>Clear</span> <span>Clear</span>
</button> </button>
@@ -406,20 +361,12 @@
<MixCalculatorResultsPanel <MixCalculatorResultsPanel
preview={preview} preview={preview}
sessionNumber={initialSession?.session_number ?? null} sessionNumber={initialSession?.session_number ?? null}
onOpenPdf={preview ? openPdf : null}
onDownloadPdf={preview ? downloadPdf : null}
/> />
</section> </section>
{#if preview} {#if preview}
{#if previewModalOpen}
<MixCalculatorPreviewModal
preview={preview}
sessionId={initialSession?.id ?? null}
onClose={closePreviewModal}
onPrint={printPreview}
onDownloadPdf={downloadSessionPdf}
/>
{/if}
<section class="print-only" aria-hidden="true"> <section class="print-only" aria-hidden="true">
<MixCalculatorPrintDocument session={preview} /> <MixCalculatorPrintDocument session={preview} />
</section> </section>
@@ -474,7 +421,7 @@
.form-card, .form-card,
.locked-card { .locked-card {
border: 1px solid var(--line); border: 1px solid var(--line);
border-radius: 1.3rem; border-radius: 0.8rem;
background: var(--panel); background: var(--panel);
box-shadow: var(--shadow); box-shadow: var(--shadow);
} }
@@ -511,7 +458,7 @@
gap: 0.14rem; gap: 0.14rem;
padding: 0.72rem 0.82rem; padding: 0.72rem 0.82rem;
border: 1px solid var(--line); border: 1px solid var(--line);
border-radius: 0.92rem; border-radius: 0.65rem;
background: var(--panel-soft); background: var(--panel-soft);
} }
@@ -548,9 +495,20 @@
width: 100%; width: 100%;
padding: 0.78rem 0.82rem; padding: 0.78rem 0.82rem;
border: 1px solid var(--line-strong); border: 1px solid var(--line-strong);
border-radius: 0.88rem; border-radius: 0.6rem;
background: #fff; background: #fff;
color: var(--text); color: var(--text);
transition:
border-color 160ms ease,
box-shadow 160ms ease;
}
input:focus-visible,
select:focus-visible,
textarea:focus-visible {
outline: none;
border-color: var(--color-brand);
box-shadow: 0 0 0 3px color-mix(in srgb, var(--color-brand) 18%, transparent);
} }
textarea { textarea {
@@ -562,14 +520,14 @@
gap: 0.2rem; gap: 0.2rem;
margin-top: 1rem; margin-top: 1rem;
padding: 0.92rem; padding: 0.92rem;
border-radius: 1rem; border-radius: 0.65rem;
background: var(--panel-soft); background: var(--panel-soft);
} }
.message { .message {
margin-bottom: 0.85rem; margin-bottom: 0.85rem;
padding: 0.75rem 0.85rem; padding: 0.75rem 0.85rem;
border-radius: 0.88rem; border-radius: 0.6rem;
font-size: 0.9rem; font-size: 0.9rem;
} }
@@ -589,21 +547,20 @@
} }
.primary-button, .primary-button,
.secondary-button { .secondary-button,
.danger-button {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
gap: 0.5rem; gap: 0.5rem;
padding: 0.78rem 0.96rem; padding: 0.78rem 0.96rem;
border-radius: 0.9rem; border-radius: 0.6rem;
border: 1px solid var(--line-strong); border: 1px solid var(--line-strong);
font-weight: 600; font-weight: 600;
cursor: pointer; cursor: pointer;
transition: transition:
transform 140ms ease, background-color 160ms ease,
box-shadow 140ms ease, border-color 160ms ease;
background-color 140ms ease,
border-color 140ms ease;
} }
.primary-button { .primary-button {
@@ -617,20 +574,35 @@
color: #304038; color: #304038;
} }
.primary-button:hover:not(:disabled), .danger-button {
.secondary-button:hover:not(:disabled) { background: #c63d32;
transform: translateY(-1px); color: #fff;
border-color: #c63d32;
} }
.primary-button:hover:not(:disabled) { .primary-button:hover:not(:disabled) {
box-shadow: none; background: #126a33;
filter: brightness(1.04); }
.primary-button:active:not(:disabled) {
background: #0f5a2b;
} }
.secondary-button:hover:not(:disabled) { .secondary-button:hover:not(:disabled) {
border-color: #9fb0a6; border-color: var(--color-text-muted);
background: #f6faf7; background: var(--color-bg-app);
box-shadow: none; }
.danger-button:hover:not(:disabled) {
background: #b2352b;
border-color: #b2352b;
}
.primary-button:focus-visible,
.secondary-button:focus-visible,
.danger-button:focus-visible {
outline: 3px solid color-mix(in srgb, var(--color-brand) 45%, transparent);
outline-offset: 2px;
} }
.button-icon { .button-icon {
@@ -6,14 +6,14 @@
preview, preview,
sessionId = null, sessionId = null,
onClose, onClose,
onPrint, onOpenPdf,
onDownloadPdf onDownloadPdf
}: { }: {
preview: MixCalculatorPreview | MixCalculatorSession; preview: MixCalculatorPreview | MixCalculatorSession;
sessionId?: number | null; sessionId?: number | null;
onClose: () => void; onClose: () => void;
onPrint: () => void; onOpenPdf: () => void;
onDownloadPdf: (sessionId: number) => void; onDownloadPdf: () => void;
} = $props(); } = $props();
</script> </script>
@@ -22,7 +22,7 @@
class="preview-modal" class="preview-modal"
role="dialog" role="dialog"
aria-modal="true" aria-modal="true"
aria-label="Print preview" aria-label="Styled PDF preview"
tabindex="-1" tabindex="-1"
onclick={(event) => event.stopPropagation()} onclick={(event) => event.stopPropagation()}
onkeydown={(event) => { onkeydown={(event) => {
@@ -33,16 +33,16 @@
> >
<div class="preview-modal-toolbar"> <div class="preview-modal-toolbar">
<div> <div>
<p class="preview-modal-kicker">Print Preview</p> <p class="preview-modal-kicker">Styled PDF Preview</p>
<h3>{preview.product_name}</h3> <h3>{preview.product_name}</h3>
</div> </div>
<div class="preview-modal-actions"> <div class="preview-modal-actions">
<button class="secondary-button" type="button" onclick={onClose}>Close</button> <button class="secondary-button" type="button" onclick={onClose}>Close</button>
<button class="primary-button" type="button" onclick={onOpenPdf}>Open PDF in new tab</button>
{#if sessionId} {#if sessionId}
<a class="secondary-button" href={`/mix-calculator/${sessionId}/print`}>Open page</a> <a class="secondary-button" href={`/mix-calculator/${sessionId}/print`}>Open PDF page</a>
<button class="secondary-button" type="button" onclick={() => onDownloadPdf(sessionId)}>Download PDF</button>
{/if} {/if}
<button class="primary-button" type="button" onclick={onPrint}>Print / Save PDF</button> <button class="secondary-button" type="button" onclick={onDownloadPdf}>Download PDF</button>
</div> </div>
</div> </div>
@@ -77,10 +77,9 @@
width: min(1180px, 100%); width: min(1180px, 100%);
max-height: calc(100vh - 2rem); max-height: calc(100vh - 2rem);
padding: 1rem; padding: 1rem;
border: 1px solid rgba(255, 255, 255, 0.32); border: 1px solid var(--color-border);
border-radius: 1.6rem; border-radius: 0.9rem;
background: background: var(--color-bg-surface);
linear-gradient(180deg, rgba(248, 250, 248, 0.96), rgba(240, 246, 242, 0.96));
} }
.preview-modal-toolbar { .preview-modal-toolbar {
@@ -116,9 +115,9 @@
display: grid; display: grid;
place-items: start center; place-items: start center;
padding: 1.1rem; padding: 1.1rem;
border-radius: 1.35rem; border: 1px solid var(--color-border);
background: border-radius: 0.8rem;
linear-gradient(135deg, #dfe8e2 0%, #eef3ef 45%, #d7e2db 100%); background: var(--color-bg-app);
} }
.preview-sheet-scroll { .preview-sheet-scroll {
@@ -135,10 +134,13 @@
justify-content: center; justify-content: center;
gap: 0.5rem; gap: 0.5rem;
padding: 0.78rem 0.96rem; padding: 0.78rem 0.96rem;
border-radius: 0.9rem; border-radius: 0.6rem;
border: 1px solid var(--line-strong); border: 1px solid var(--line-strong);
font-weight: 600; font-weight: 600;
cursor: pointer; cursor: pointer;
transition:
background-color 160ms ease,
border-color 160ms ease;
} }
.primary-button { .primary-button {
@@ -152,6 +154,25 @@
color: #304038; color: #304038;
} }
.primary-button:hover {
background: #126a33;
}
.primary-button:active {
background: #0f5a2b;
}
.secondary-button:hover {
border-color: var(--color-text-muted);
background: var(--color-bg-app);
}
.primary-button:focus-visible,
.secondary-button:focus-visible {
outline: 3px solid color-mix(in srgb, var(--color-brand) 45%, transparent);
outline-offset: 2px;
}
@media (max-width: 720px) { @media (max-width: 720px) {
.preview-modal-toolbar { .preview-modal-toolbar {
flex-direction: column; flex-direction: column;
@@ -1,12 +1,17 @@
<script lang="ts"> <script lang="ts">
import { Download, Printer } from 'lucide-svelte';
import type { MixCalculatorPreview, MixCalculatorSession } from '$lib/types'; import type { MixCalculatorPreview, MixCalculatorSession } from '$lib/types';
let { let {
preview, preview,
sessionNumber = null sessionNumber = null,
onOpenPdf = null,
onDownloadPdf = null
}: { }: {
preview: MixCalculatorPreview | MixCalculatorSession | null; preview: MixCalculatorPreview | MixCalculatorSession | null;
sessionNumber?: string | null; sessionNumber?: string | null;
onOpenPdf?: (() => void) | null;
onDownloadPdf?: (() => void) | null;
} = $props(); } = $props();
function formatDate(value: string) { function formatDate(value: string) {
@@ -89,9 +94,7 @@
<thead> <thead>
<tr> <tr>
<th>Raw material</th> <th>Raw material</th>
<th>Mix %</th>
<th>Required kg</th> <th>Required kg</th>
<th>Unit</th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@@ -100,14 +103,23 @@
<td data-label="Raw material"> <td data-label="Raw material">
<strong>{line.raw_material_name}</strong> <strong>{line.raw_material_name}</strong>
</td> </td>
<td data-label="Mix %">{formatNumber(line.mix_percentage, 2)}%</td>
<td data-label="Required kg">{formatNumber(line.required_kg, 2)}kg</td> <td data-label="Required kg">{formatNumber(line.required_kg, 2)}kg</td>
<td data-label="Unit">{line.unit}</td>
</tr> </tr>
{/each} {/each}
</tbody> </tbody>
</table> </table>
</div> </div>
<div class="output-actions">
<button class="primary-button" disabled={!onOpenPdf} type="button" onclick={() => onOpenPdf?.()}>
<Printer size={18} strokeWidth={1.9} aria-hidden="true" />
Print
</button>
<button class="secondary-button" disabled={!onDownloadPdf} type="button" onclick={() => onDownloadPdf?.()}>
<Download size={18} strokeWidth={1.9} aria-hidden="true" />
Download PDF
</button>
</div>
{:else} {:else}
<div class="empty-state"> <div class="empty-state">
<div class="empty-shimmer-metrics"> <div class="empty-shimmer-metrics">
@@ -127,8 +139,6 @@
<div class="shimmer-row"> <div class="shimmer-row">
<div class="shimmer-line wide"></div> <div class="shimmer-line wide"></div>
<div class="shimmer-line medium"></div> <div class="shimmer-line medium"></div>
<div class="shimmer-line medium"></div>
<div class="shimmer-line short"></div>
</div> </div>
{/each} {/each}
</div> </div>
@@ -152,7 +162,7 @@
.result-card, .result-card,
.metric-card { .metric-card {
border: 1px solid var(--line); border: 1px solid var(--line);
border-radius: 1.3rem; border-radius: 0.8rem;
background: var(--panel); background: var(--panel);
box-shadow: var(--shadow); box-shadow: var(--shadow);
} }
@@ -179,7 +189,7 @@
gap: 0.14rem; gap: 0.14rem;
padding: 0.72rem 0.82rem; padding: 0.72rem 0.82rem;
border: 1px solid var(--line); border: 1px solid var(--line);
border-radius: 0.92rem; border-radius: 0.65rem;
background: var(--panel-soft); background: var(--panel-soft);
} }
@@ -222,9 +232,9 @@
gap: 0.45rem; gap: 0.45rem;
margin-top: 1rem; margin-top: 1rem;
padding: 0.92rem; padding: 0.92rem;
border-radius: 1rem; border-radius: 0.65rem;
background: #fff6e6; background: #fdf6e9;
color: #8b5b1e; color: #8a5a00;
} }
.summary-grid { .summary-grid {
@@ -235,7 +245,7 @@
.summary-grid div { .summary-grid div {
padding: 0.88rem 0.92rem; padding: 0.88rem 0.92rem;
border: 1px solid var(--line); border: 1px solid var(--line);
border-radius: 1rem; border-radius: 0.65rem;
background: var(--panel-soft); background: var(--panel-soft);
} }
@@ -251,6 +261,14 @@
overflow-x: auto; overflow-x: auto;
} }
.output-actions {
display: flex;
justify-content: flex-start;
gap: 0.75rem;
flex-wrap: wrap;
margin-top: 1rem;
}
table { table {
width: 100%; width: 100%;
min-width: 30rem; min-width: 30rem;
@@ -272,6 +290,57 @@
text-transform: uppercase; text-transform: uppercase;
} }
.primary-button,
.secondary-button {
display: inline-flex;
align-items: center;
justify-content: center;
gap: 0.5rem;
padding: 0.78rem 0.96rem;
border-radius: 0.6rem;
border: 1px solid var(--line-strong);
font-weight: 600;
cursor: pointer;
transition:
background-color 160ms ease,
border-color 160ms ease;
}
.primary-button {
border: none;
background: var(--color-brand);
color: #fff;
}
.secondary-button {
background: #fff;
color: #304038;
}
.primary-button:hover:not(:disabled) {
background: #126a33;
}
.primary-button:active:not(:disabled) {
background: #0f5a2b;
}
.secondary-button:hover:not(:disabled) {
border-color: var(--color-text-muted);
background: var(--color-bg-app);
}
.primary-button:focus-visible,
.secondary-button:focus-visible {
outline: 3px solid color-mix(in srgb, var(--color-brand) 45%, transparent);
outline-offset: 2px;
}
button:disabled {
cursor: not-allowed;
opacity: 0.65;
}
.empty-state { .empty-state {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
@@ -355,7 +424,7 @@
.shimmer-row { .shimmer-row {
display: grid; display: grid;
grid-template-columns: 2fr 1fr 1fr 0.75fr; grid-template-columns: 2fr 1fr;
gap: 1rem; gap: 1rem;
align-items: center; align-items: center;
padding: 0.78rem 1rem; padding: 0.78rem 1rem;
@@ -8,6 +8,7 @@
active?: boolean; active?: boolean;
onSelect?: () => void; onSelect?: () => void;
type?: 'button' | 'link'; type?: 'button' | 'link';
badge?: string;
}; };
let { let {
@@ -33,14 +34,14 @@
{#if Icon} {#if Icon}
<span class="nav-icon"><Icon size={18} strokeWidth={1.75} /></span> <span class="nav-icon"><Icon size={18} strokeWidth={1.75} /></span>
{/if} {/if}
<span>{item.label}</span> <span>{item.label}</span>{#if item.badge}<span class="nav-badge">{item.badge}</span>{/if}
</a> </a>
{:else} {:else}
<button type="button" class="nav-button" class:active={item.active} onclick={item.onSelect}> <button type="button" class="nav-button" class:active={item.active} onclick={item.onSelect}>
{#if Icon} {#if Icon}
<span class="nav-icon"><Icon size={18} strokeWidth={1.75} /></span> <span class="nav-icon"><Icon size={18} strokeWidth={1.75} /></span>
{/if} {/if}
<span>{item.label}</span> <span>{item.label}</span>{#if item.badge}<span class="nav-badge">{item.badge}</span>{/if}
</button> </button>
{/if} {/if}
{/each} {/each}
@@ -105,6 +106,26 @@
background: var(--nav-item-active-marker, var(--color-brand)); background: var(--nav-item-active-marker, var(--color-brand));
} }
.nav-badge {
margin-left: auto;
flex-shrink: 0;
padding: 0.08rem 0.4rem;
border-radius: 999px;
background: #fdf0d2;
color: #8a5a00;
font-size: 0.62rem;
font-weight: 700;
letter-spacing: 0.06em;
text-transform: uppercase;
line-height: 1.5;
}
.nav-list a.active .nav-badge,
.nav-button.active .nav-badge {
background: rgba(255, 255, 255, 0.92);
color: #8a5a00;
}
.nav-icon { .nav-icon {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
@@ -11,7 +11,6 @@
workingDocumentItems, workingDocumentItems,
footerItems, footerItems,
appVersion, appVersion,
releaseStage,
currentYear, currentYear,
canOpenSettings, canOpenSettings,
onOpenSettings, onOpenSettings,
@@ -23,7 +22,6 @@
workingDocumentItems: NavItem[]; workingDocumentItems: NavItem[];
footerItems: FooterLink[]; footerItems: FooterLink[];
appVersion: string; appVersion: string;
releaseStage: string;
currentYear: number; currentYear: number;
canOpenSettings: boolean; canOpenSettings: boolean;
onOpenSettings: () => void; onOpenSettings: () => void;
@@ -46,6 +44,7 @@
label: item.label, label: item.label,
href: item.href, href: item.href,
icon: item.icon, icon: item.icon,
badge: item.badge,
active: matchesRoute(item.href, currentPath) active: matchesRoute(item.href, currentPath)
}))} }))}
/> />
@@ -105,7 +104,6 @@
<span class="meta-label">Build</span> <span class="meta-label">Build</span>
<span>{appVersion}</span> <span>{appVersion}</span>
</span> </span>
<span class="release-pill">{releaseStage}</span>
</div> </div>
<div class="sidebar-meta-bottom"> <div class="sidebar-meta-bottom">
<small>&copy; {currentYear} Hunter Premium Produce</small> <small>&copy; {currentYear} Hunter Premium Produce</small>
@@ -242,18 +240,4 @@
text-transform: uppercase; text-transform: uppercase;
} }
.release-pill {
display: inline-flex;
align-items: center;
justify-content: center;
padding: 0.2rem 0.52rem;
border: 1px solid color-mix(in srgb, var(--color-brand) 14%, transparent);
border-radius: 999px;
background: color-mix(in srgb, var(--color-brand) 8%, white);
color: var(--color-brand);
font-size: 0.64rem;
font-weight: 700;
letter-spacing: 0.07em;
text-transform: uppercase;
}
</style> </style>
@@ -4,6 +4,7 @@ import {
ClipboardList, ClipboardList,
DollarSign, DollarSign,
FlaskConical, FlaskConical,
Gauge,
LayoutDashboard, LayoutDashboard,
ShieldCheck, ShieldCheck,
TrendingUp, TrendingUp,
@@ -13,6 +14,10 @@ import {
import type { ComponentType } from 'svelte'; import type { ComponentType } from 'svelte';
import { featureFlags } from '$lib/features'; import { featureFlags } from '$lib/features';
import {
canOpenDashboard
} from '$lib/workspace-access';
import type { AppSession } from '$lib/session';
export type SearchItem = { export type SearchItem = {
href: string; href: string;
@@ -27,6 +32,7 @@ export type NavItem = {
shortLabel: string; shortLabel: string;
icon: ComponentType; icon: ComponentType;
moduleKey?: string; moduleKey?: string;
badge?: string;
}; };
export type FooterLink = { export type FooterLink = {
@@ -50,7 +56,7 @@ export const dashboardItem: NavItem = {
}; };
export const mixCalculatorItem: NavItem = { export const mixCalculatorItem: NavItem = {
href: featureFlags.mixCalculatorSessionHistory ? '/mix-calculator' : '/mix-calculator/new', href: '/mix-calculator',
label: 'Mix Calculator', label: 'Mix Calculator',
shortLabel: 'MC', shortLabel: 'MC',
icon: Calculator, icon: Calculator,
@@ -65,6 +71,15 @@ export const reportingItem: NavItem = {
moduleKey: 'products' moduleKey: 'products'
}; };
export const throughputItem: NavItem = {
href: '/throughput',
label: 'Throughput',
shortLabel: 'OT',
icon: Gauge,
moduleKey: 'operations_throughput',
badge: 'test'
};
export const workingDocumentItems: NavItem[] = [ export const workingDocumentItems: NavItem[] = [
{ href: '/raw-materials', label: 'Raw Materials', shortLabel: 'RM', icon: Wheat, moduleKey: 'raw_materials' }, { href: '/raw-materials', label: 'Raw Materials', shortLabel: 'RM', icon: Wheat, moduleKey: 'raw_materials' },
{ href: '/mixes', label: 'Mix Master', shortLabel: 'MM', icon: FlaskConical, moduleKey: 'mix_master' }, { href: '/mixes', label: 'Mix Master', shortLabel: 'MM', icon: FlaskConical, moduleKey: 'mix_master' },
@@ -83,6 +98,7 @@ export const accessControlItem: NavItem = {
export const clientNavigationItems: NavItem[] = [ export const clientNavigationItems: NavItem[] = [
dashboardItem, dashboardItem,
mixCalculatorItem, mixCalculatorItem,
throughputItem,
...workingDocumentItems, ...workingDocumentItems,
accessControlItem accessControlItem
]; ];
@@ -128,8 +144,8 @@ export const baseSearchItems: SearchItem[] = [
] ]
: []), : []),
{ {
href: '/mix-calculator/new', href: '/mix-calculator',
label: 'Create Mix Calculation', label: 'Open Mix Calculator',
description: 'Run a new client-specific mix calculation session.', description: 'Run a new client-specific mix calculation session.',
keywords: 'new mix calculator session client batch size product bags print' keywords: 'new mix calculator session client batch size product bags print'
}, },
@@ -167,26 +183,30 @@ export function pageTitle(pathname: string) {
return clientNavigationItems.find((item) => matchesRoute(item.href, pathname))?.label ?? 'Dashboard'; return clientNavigationItems.find((item) => matchesRoute(item.href, pathname))?.label ?? 'Dashboard';
} }
export function clientBreadcrumbs(pathname: string): Crumb[] { export function clientBreadcrumbs(pathname: string, session?: AppSession | null): Crumb[] {
const root: Crumb = { label: 'Workspace', href: '/' }; const crumbs: Crumb[] = [];
if (canOpenDashboard(session)) {
crumbs.push({ label: 'Workspace', href: '/' });
}
if (pathname === '/') { if (pathname === '/') {
return [root, { label: 'Dashboard' }]; return crumbs.length ? [...crumbs, { label: 'Dashboard' }] : [{ label: 'Dashboard' }];
} }
if (pathname.startsWith('/mix-calculator')) { if (pathname.startsWith('/mix-calculator')) {
const trail: Crumb[] = [root, { label: 'Mix Calculator', href: '/mix-calculator' }]; return [...crumbs, { label: 'Mix Calculator' }];
if (pathname === '/mix-calculator/new') trail.push({ label: 'New Session' });
else if (pathname.endsWith('/print')) trail.push({ label: 'Print' });
else if (pathname !== '/mix-calculator') trail.push({ label: 'Session' });
return trail;
} }
if (pathname.startsWith('/mixes')) { if (pathname.startsWith('/mixes')) {
const trail: Crumb[] = [root, { label: 'Mix Master', href: '/mixes' }]; return [...crumbs, { label: 'Mix Master' }];
if (pathname === '/mixes/new') trail.push({ label: 'New Mix' }); }
else if (pathname !== '/mixes') trail.push({ label: 'Detail' });
return trail; if (pathname.startsWith('/throughput')) {
const base: Crumb[] = [...crumbs, { label: 'Throughput', href: '/throughput' }];
if (pathname === '/throughput/add') return [...base, { label: 'Add Entry' }];
if (pathname === '/throughput') return base.slice(0, -1).concat([{ label: 'Throughput' }]);
return base;
} }
const sectionMap: Record<string, string> = { const sectionMap: Record<string, string> = {
@@ -195,10 +215,11 @@ export function clientBreadcrumbs(pathname: string): Crumb[] {
'/scenarios': 'Scenarios', '/scenarios': 'Scenarios',
'/client-access': 'Client Access', '/client-access': 'Client Access',
'/reporting': 'Reporting', '/reporting': 'Reporting',
'/settings': 'Settings' '/settings': 'Settings',
'/throughput': 'Throughput'
}; };
const section = sectionMap[pathname]; const section = sectionMap[pathname];
if (section) return [root, { label: section }]; if (section) return [...crumbs, { label: section }];
return [root, { label: pageTitle(pathname) }]; return [...crumbs, { label: pageTitle(pathname) }];
} }
+86
View File
@@ -388,3 +388,89 @@ export type ClientUserUpdateInput = {
status?: string; status?: string;
is_new_user?: boolean; is_new_user?: boolean;
}; };
export type ThroughputProduct = {
id: number;
tenant_id: string;
item_id: string | null;
name: string;
default_bag_size: number | null;
is_bulka_default: boolean;
active: boolean;
is_stock_item: boolean;
notes: string | null;
created_at: string;
updated_at: string;
};
export type ThroughputQuantityType = 'bags' | 'kg';
export type ThroughputEntry = {
id: number;
tenant_id: string;
production_date: string;
product_id: number | null;
product_name_snapshot: string;
bag_size: number | null;
scales_checked: boolean;
label_correct: boolean;
bag_sealed: boolean;
pallet_good_condition: boolean;
sample_box_no: string | null;
test_weight_1: number | null;
test_weight_2: number | null;
test_weight_3: number | null;
test_weight_4: number | null;
test_weight_5: number | null;
quantity: number;
quantity_type: ThroughputQuantityType;
calculated_kg: number;
staff_name: string | null;
notes: string | null;
qa_passed: boolean;
created_by: string | null;
created_at: string;
updated_at: string;
};
export type ThroughputEntryCreateInput = {
production_date: string;
product_id?: number | null;
product_name_snapshot?: string;
bag_size?: number | null;
scales_checked?: boolean;
label_correct?: boolean;
bag_sealed?: boolean;
pallet_good_condition?: boolean;
sample_box_no?: string | null;
test_weight_1?: number | null;
test_weight_2?: number | null;
test_weight_3?: number | null;
test_weight_4?: number | null;
test_weight_5?: number | null;
quantity: number;
quantity_type: ThroughputQuantityType;
staff_name?: string | null;
notes?: string | null;
};
export type ThroughputEntryListParams = {
date_from?: string;
date_to?: string;
product_id?: number | null;
staff_name?: string;
quantity_type?: ThroughputQuantityType;
limit?: number;
};
export type ThroughputProductCreateInput = {
item_id?: string | null;
name: string;
default_bag_size?: number | null;
is_bulka_default?: boolean;
active?: boolean;
is_stock_item?: boolean;
notes?: string | null;
};
export type ThroughputProductUpdateInput = Partial<ThroughputProductCreateInput>;
+1 -1
View File
@@ -23,7 +23,7 @@ describe('workspace access policy', () => {
it('classifies operations users and sends them to mix calculator by default', () => { it('classifies operations users and sends them to mix calculator by default', () => {
expect(getWorkspaceRole(operationsSession)).toBe('operations'); expect(getWorkspaceRole(operationsSession)).toBe('operations');
expect(getDefaultRouteForRole(operationsSession)).toBe('/mix-calculator/new'); expect(getDefaultRouteForRole(operationsSession)).toBe('/mix-calculator');
}); });
it('prevents operations users from opening the dashboard route', () => { it('prevents operations users from opening the dashboard route', () => {
+16 -3
View File
@@ -1,4 +1,3 @@
import { featureFlags } from '$lib/features';
import { hasModuleAccess, hasPermission, type AppSession } from '$lib/session'; import { hasModuleAccess, hasPermission, type AppSession } from '$lib/session';
export type WorkspaceRole = 'admin' | 'operations' | 'full' | 'client' | 'unknown'; export type WorkspaceRole = 'admin' | 'operations' | 'full' | 'client' | 'unknown';
@@ -90,6 +89,14 @@ export function canOpenScenarios(session: AppSession | null | undefined) {
return !!session && hasModuleAccess(session, 'scenarios'); return !!session && hasModuleAccess(session, 'scenarios');
} }
export function canOpenThroughput(session: AppSession | null | undefined) {
return canAccessWorkspaceArea(session, 'operations_throughput', ['view_throughput', 'edit_throughput']);
}
export function canEditThroughput(session: AppSession | null | undefined) {
return canAccessWorkspaceArea(session, 'operations_throughput', ['edit_throughput'], 'edit');
}
export function canOpenReporting(session: AppSession | null | undefined) { export function canOpenReporting(session: AppSession | null | undefined) {
return canOpenProducts(session); return canOpenProducts(session);
} }
@@ -133,6 +140,11 @@ export const routeAccessRules: RouteAccessRule[] = [
path: '/client-access', path: '/client-access',
roles: ['admin', 'client'], roles: ['admin', 'client'],
matches: (pathname) => hasPathPrefix(pathname, '/client-access') matches: (pathname) => hasPathPrefix(pathname, '/client-access')
},
{
path: '/throughput',
roles: ['admin', 'operations', 'full', 'client'],
matches: (pathname) => hasPathPrefix(pathname, '/throughput')
} }
]; ];
@@ -140,12 +152,12 @@ export function getDefaultRouteForRole(session: AppSession | null | undefined) {
const role = getWorkspaceRole(session); const role = getWorkspaceRole(session);
if (role === 'operations') { if (role === 'operations') {
return featureFlags.mixCalculatorSessionHistory ? '/mix-calculator' : '/mix-calculator/new'; return '/mix-calculator';
} }
if (role === 'admin' || role === 'full' || role === 'client') { if (role === 'admin' || role === 'full' || role === 'client') {
if (canOpenDashboard(session)) return '/'; if (canOpenDashboard(session)) return '/';
if (canOpenMixCalculator(session)) return featureFlags.mixCalculatorSessionHistory ? '/mix-calculator' : '/mix-calculator/new'; if (canOpenMixCalculator(session)) return '/mix-calculator';
if (canOpenRawMaterials(session)) return '/raw-materials'; if (canOpenRawMaterials(session)) return '/raw-materials';
if (canOpenMixMaster(session)) return '/mixes'; if (canOpenMixMaster(session)) return '/mixes';
if (canOpenProducts(session)) return '/products'; if (canOpenProducts(session)) return '/products';
@@ -176,6 +188,7 @@ export function canAccessRoute(session: AppSession | null | undefined, pathname:
if (pathname.startsWith('/reporting')) return canOpenReporting(session); if (pathname.startsWith('/reporting')) return canOpenReporting(session);
if (pathname.startsWith('/settings')) return canOpenSettings(session); if (pathname.startsWith('/settings')) return canOpenSettings(session);
if (pathname.startsWith('/client-access')) return canOpenClientAccess(session); if (pathname.startsWith('/client-access')) return canOpenClientAccess(session);
if (pathname.startsWith('/throughput')) return canOpenThroughput(session);
return true; return true;
} }
+5 -14
View File
@@ -42,30 +42,21 @@
<div class="error-actions"> <div class="error-actions">
<a class="primary-link" href="/">Return to Workspace</a> <a class="primary-link" href="/">Return to Workspace</a>
<a class="secondary-link" href="/mix-calculator/new">Open Mix Calculator</a> <a class="secondary-link" href="/mix-calculator">Open Mix Calculator</a>
</div> </div>
</div> </div>
</section> </section>
<style> <style>
:global(body) {
margin: 0;
min-height: 100vh;
background:
radial-gradient(circle at top left, rgba(214, 234, 221, 0.86), transparent 36%),
linear-gradient(180deg, #f4f7f2 0%, #eef4ee 100%);
color: #1d3528;
font-family:
"Segoe UI",
system-ui,
sans-serif;
}
.error-stage { .error-stage {
min-height: 100vh; min-height: 100vh;
display: grid; display: grid;
place-items: center; place-items: center;
padding: 2rem; padding: 2rem;
background:
radial-gradient(circle at top left, rgba(214, 234, 221, 0.86), transparent 36%),
linear-gradient(180deg, #f4f7f2 0%, #eef4ee 100%);
color: #1d3528;
} }
.error-card { .error-card {
+21
View File
@@ -1,4 +1,8 @@
<script lang="ts"> <script lang="ts">
import '@fontsource/inter/latin-400.css';
import '@fontsource/inter/latin-500.css';
import '@fontsource/inter/latin-600.css';
import '@fontsource/inter/latin-700.css';
import { beforeNavigate, afterNavigate } from '$app/navigation'; import { beforeNavigate, afterNavigate } from '$app/navigation';
import { page } from '$app/state'; import { page } from '$app/state';
import AdminShell from '$lib/components/AdminShell.svelte'; import AdminShell from '$lib/components/AdminShell.svelte';
@@ -46,3 +50,20 @@
{/if} {/if}
<Toast /> <Toast />
<style>
:global(html, body) {
font-family: "Inter", "Segoe UI", sans-serif;
}
:global(button),
:global(input),
:global(select),
:global(textarea) {
font: inherit;
}
:global(h1, h2, h3, h4, h5, h6) {
font-family: "Inter", "Segoe UI", sans-serif;
}
</style>
+56 -59
View File
@@ -42,7 +42,6 @@
let loginFocusArmed = $state(true); let loginFocusArmed = $state(true);
const currentYear = new Date().getFullYear(); const currentYear = new Date().getFullYear();
const appVersion = `v${packageInfo.version}`; const appVersion = `v${packageInfo.version}`;
const releaseStage = 'Beta';
const monthLabels = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep']; const monthLabels = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep'];
@@ -353,10 +352,7 @@
<span class="powered-by-label">Powered by Lean 101</span> <span class="powered-by-label">Powered by Lean 101</span>
</div> </div>
<div class="auth-meta"> <div class="auth-meta">
<span class="version-badge"> <span class="version-badge">{appVersion}</span>
<span>{appVersion}</span>
<span class="release-pill">{releaseStage}</span>
</span>
<span>&copy; {currentYear} Hunter Premium Produce</span> <span>&copy; {currentYear} Hunter Premium Produce</span>
</div> </div>
</div> </div>
@@ -375,13 +371,12 @@
</div> </div>
<div class="auth-status-row"> <div class="auth-status-row">
<span class="auth-status-pill">Secure Workspace Access</span> <span class="auth-status-pill">Secure Workspace Access</span>
<span class="release-pill">{releaseStage}</span>
</div> </div>
</div> </div>
<div class="auth-copy"> <div class="auth-copy">
<h2>Login</h2> <h2>Welcome back</h2>
<p>Enter your username & password below:</p> <p>Sign in with your email and password to continue.</p>
</div> </div>
<form class="signin-form auth-form" onsubmit={handleLogin}> <form class="signin-form auth-form" onsubmit={handleLogin}>
@@ -422,10 +417,7 @@
<span class="powered-by-label">Powered by Lean 101</span> <span class="powered-by-label">Powered by Lean 101</span>
</div> </div>
<div class="auth-meta"> <div class="auth-meta">
<span class="version-badge"> <span class="version-badge">{appVersion}</span>
<span>{appVersion}</span>
<span class="release-pill">{releaseStage}</span>
</span>
<span>&copy; {currentYear} Lean 101</span> <span>&copy; {currentYear} Lean 101</span>
</div> </div>
</div> </div>
@@ -806,25 +798,23 @@
width: min(100%, 38rem); width: min(100%, 38rem);
display: grid; display: grid;
gap: 1.35rem; gap: 1.35rem;
padding: 1.5rem; padding: 2.1rem 2rem 1.6rem;
border: 1px solid rgba(212, 226, 218, 0.95); border: 1px solid var(--color-border);
border-radius: 1.7rem; border-radius: 1.25rem;
background: background: var(--color-bg-surface);
radial-gradient(circle at top left, rgba(115, 197, 146, 0.16), transparent 32%),
radial-gradient(circle at bottom right, rgba(33, 94, 60, 0.1), transparent 30%),
rgba(255, 255, 255, 0.96);
box-shadow: none;
backdrop-filter: blur(14px);
overflow: hidden; overflow: hidden;
} }
/* Crisp brand accent along the top edge, clipped to the card radius.
Replaces the old green radial glow, which read as a muddy shadow. */
.auth-card::before { .auth-card::before {
content: ''; content: '';
position: absolute; position: absolute;
inset: 0; top: 0;
background: left: 0;
linear-gradient(135deg, rgba(255, 255, 255, 0.42), transparent 35%), right: 0;
linear-gradient(180deg, transparent, rgba(238, 248, 242, 0.55)); height: 4px;
background: var(--color-brand);
pointer-events: none; pointer-events: none;
} }
@@ -913,21 +903,6 @@
flex-wrap: wrap; flex-wrap: wrap;
} }
.release-pill {
display: inline-flex;
align-items: center;
justify-content: center;
padding: 0.28rem 0.62rem;
border: 1px solid color-mix(in srgb, var(--color-brand) 16%, transparent);
border-radius: 999px;
background: var(--color-brand-tint);
color: var(--color-success);
font-size: 0.72rem;
font-weight: 700;
letter-spacing: 0.08em;
text-transform: uppercase;
}
.auth-copy { .auth-copy {
display: grid; display: grid;
gap: 0.55rem; gap: 0.55rem;
@@ -994,20 +969,25 @@
} }
.auth-form input { .auth-form input {
padding: 1rem 1.05rem; padding: 0.95rem 1rem;
border: 1px solid #d6e3db; border: 1px solid var(--color-border);
border-radius: 1rem; border-radius: 0.7rem;
background: rgba(248, 251, 249, 0.94); background: var(--color-bg-app);
color: var(--color-text-primary);
transition: transition:
border-color 160ms ease, border-color 160ms ease,
box-shadow 160ms ease, box-shadow 160ms ease,
background-color 160ms ease; background-color 160ms ease;
} }
.auth-form input::placeholder {
color: var(--color-text-muted);
}
.auth-form input:focus { .auth-form input:focus {
outline: none; outline: none;
border-color: var(--color-brand); border-color: var(--color-brand);
box-shadow: 0 0 0 0.24rem color-mix(in srgb, var(--color-brand) 12%, transparent); box-shadow: 0 0 0 3px color-mix(in srgb, var(--color-brand) 22%, transparent);
background: #fff; background: #fff;
} }
@@ -1021,6 +1001,26 @@
width: 100%; width: 100%;
min-height: 3.35rem; min-height: 3.35rem;
margin-top: 0.2rem; margin-top: 0.2rem;
font-size: 1.02rem;
transition: background-color 160ms cubic-bezier(0.22, 1, 0.36, 1);
}
.auth-submit:hover:not(:disabled) {
background: #126a33;
}
.auth-submit:active:not(:disabled) {
background: #0f5a2b;
}
.auth-submit:focus-visible {
outline: 3px solid color-mix(in srgb, var(--color-brand) 45%, transparent);
outline-offset: 2px;
}
.auth-submit:disabled {
opacity: 0.7;
cursor: progress;
} }
.auth-footer { .auth-footer {
@@ -1221,23 +1221,20 @@
.signin-form { .signin-form {
display: grid; display: grid;
grid-template-columns: repeat(3, minmax(0, 1fr)); grid-template-columns: 1fr;
gap: 0.75rem; gap: 0.95rem;
width: min(100%, 38rem);
}
.signin-form input {
width: 100%; width: 100%;
padding: 0.9rem 0.95rem;
border: 1px solid var(--line-strong);
border-radius: 0.95rem;
background: var(--panel-soft);
color: var(--text);
} }
.login-error { .login-error {
color: #a03737; margin: 0;
padding: 0.75rem 0.95rem;
border: 1px solid rgba(160, 55, 55, 0.3);
border-radius: 0.7rem;
background: #fdf2f2;
color: #8a1622;
font-weight: 600; font-weight: 600;
font-size: 0.95rem;
} }
.focus-card { .focus-card {
@@ -1905,8 +1902,8 @@
} }
.auth-card { .auth-card {
padding: 1.15rem; padding: 1.5rem 1.15rem 1.15rem;
border-radius: 1.35rem; border-radius: 1.1rem;
} }
.auth-header, .auth-header,
+2 -2
View File
@@ -118,10 +118,10 @@ describe('route loaders use the SvelteKit fetch argument', () => {
expect(apiMocks.productCosts).toHaveBeenCalledWith(fetcher); expect(apiMocks.productCosts).toHaveBeenCalledWith(fetcher);
}); });
it('passes fetch through the mix calculator history loader', async () => { it('passes fetch through the mix calculator loader', async () => {
await mixCalculatorLoad({ fetch: fetcher } as never); await mixCalculatorLoad({ fetch: fetcher } as never);
expect(apiMocks.mixCalculatorSessions).toHaveBeenCalledWith(fetcher); expect(apiMocks.mixCalculatorOptions).toHaveBeenCalledWith(fetcher);
}); });
it('passes fetch through the new mix calculator loader', async () => { it('passes fetch through the new mix calculator loader', async () => {
+37 -15
View File
@@ -1,9 +1,13 @@
<script lang="ts"> <script lang="ts">
import { clientSession, hasModuleAccess } from '$lib/session'; import { clientSession, hasModuleAccess } from '$lib/session';
import type { MixCalculatorSession } from '$lib/types'; import { featureFlags } from '$lib/features';
import MixCalculatorEditor from '$lib/components/mix-calculator/MixCalculatorEditor.svelte';
import type { MixCalculatorOptions, MixCalculatorSession } from '$lib/types';
let { data }: { data: { sessions: MixCalculatorSession[] } } = $props(); let { data }: { data: { sessions?: MixCalculatorSession[]; options?: MixCalculatorOptions } } =
$props();
const sessions = $derived(data.sessions ?? []);
const canEdit = $derived(hasModuleAccess($clientSession, 'mix_calculator', 'edit')); const canEdit = $derived(hasModuleAccess($clientSession, 'mix_calculator', 'edit'));
function formatDate(value: string) { function formatDate(value: string) {
@@ -18,6 +22,9 @@
} }
</script> </script>
{#if !featureFlags.mixCalculatorSessionHistory}
<MixCalculatorEditor options={data.options} />
{:else}
{#if canEdit} {#if canEdit}
<section class="page-actions"> <section class="page-actions">
<a class="primary-button" href="/mix-calculator/new">New mix session</a> <a class="primary-button" href="/mix-calculator/new">New mix session</a>
@@ -27,17 +34,17 @@
<section class="metric-row"> <section class="metric-row">
<article class="metric-card"> <article class="metric-card">
<span>Saved Sessions</span> <span>Saved Sessions</span>
<strong>{data.sessions.length}</strong> <strong>{sessions.length}</strong>
<p>Visible under your access scope</p> <p>Visible under your access scope</p>
</article> </article>
<article class="metric-card"> <article class="metric-card">
<span>Total Planned Kg</span> <span>Total Planned Kg</span>
<strong>{formatNumber(data.sessions.reduce((sum, session) => sum + session.total_kg, 0), 2)}</strong> <strong>{formatNumber(sessions.reduce((sum, session) => sum + session.total_kg, 0), 2)}</strong>
<p>Across the visible history</p> <p>Across the visible history</p>
</article> </article>
<article class="metric-card"> <article class="metric-card">
<span>Sessions With Warnings</span> <span>Sessions With Warnings</span>
<strong>{data.sessions.filter((session) => session.warnings.length).length}</strong> <strong>{sessions.filter((session) => session.warnings.length).length}</strong>
<p>Fractional bag outputs need review</p> <p>Fractional bag outputs need review</p>
</article> </article>
</section> </section>
@@ -50,7 +57,7 @@
</div> </div>
</div> </div>
{#if data.sessions.length} {#if sessions.length}
<div class="table-wrap"> <div class="table-wrap">
<table> <table>
<thead> <thead>
@@ -65,7 +72,7 @@
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
{#each data.sessions as session} {#each sessions as session}
<tr> <tr>
<td data-label="Session"> <td data-label="Session">
<strong>{session.session_number}</strong> <strong>{session.session_number}</strong>
@@ -102,6 +109,7 @@
</div> </div>
{/if} {/if}
</section> </section>
{/if}
<style> <style>
h2, h2,
@@ -140,10 +148,24 @@
align-items: center; align-items: center;
justify-content: center; justify-content: center;
padding: 0.78rem 0.96rem; padding: 0.78rem 0.96rem;
border-radius: 0.9rem; border-radius: 0.6rem;
background: var(--color-brand); background: var(--color-brand);
color: #fff; color: #fff;
font-weight: 600; font-weight: 600;
transition: background-color 160ms ease;
}
.primary-button:hover {
background: #126a33;
}
.primary-button:active {
background: #0f5a2b;
}
.primary-button:focus-visible {
outline: 3px solid color-mix(in srgb, var(--color-brand) 45%, transparent);
outline-offset: 2px;
} }
.metric-row { .metric-row {
@@ -155,7 +177,7 @@
.metric-card, .metric-card,
.table-card { .table-card {
border: 1px solid var(--line); border: 1px solid var(--line);
border-radius: 1.3rem; border-radius: 0.8rem;
background: var(--panel); background: var(--panel);
box-shadow: var(--shadow); box-shadow: var(--shadow);
} }
@@ -222,12 +244,12 @@
tbody td:first-child { tbody td:first-child {
border-left: 1px solid var(--line); border-left: 1px solid var(--line);
border-radius: 1rem 0 0 1rem; border-radius: 0.65rem 0 0 0.65rem;
} }
tbody td:last-child { tbody td:last-child {
border-right: 1px solid var(--line); border-right: 1px solid var(--line);
border-radius: 0 1rem 1rem 0; border-radius: 0 0.65rem 0.65rem 0;
} }
.row-actions { .row-actions {
@@ -245,8 +267,8 @@
margin-left: 0.55rem; margin-left: 0.55rem;
padding: 0.25rem 0.5rem; padding: 0.25rem 0.5rem;
border-radius: 999px; border-radius: 999px;
background: #fff6e6; background: #fdf6e9;
color: #8b5b1e; color: #8a5a00;
font-size: 0.72rem; font-size: 0.72rem;
font-weight: 700; font-weight: 700;
text-transform: uppercase; text-transform: uppercase;
@@ -257,7 +279,7 @@
display: grid; display: grid;
gap: 0.2rem; gap: 0.2rem;
padding: 1rem; padding: 1rem;
border-radius: 1rem; border-radius: 0.65rem;
background: var(--panel-soft); background: var(--panel-soft);
} }
@@ -299,7 +321,7 @@
tbody tr { tbody tr {
padding: 0.3rem; padding: 0.3rem;
border: 1px solid var(--line); border: 1px solid var(--line);
border-radius: 1rem; border-radius: 0.65rem;
background: var(--panel-soft); background: var(--panel-soft);
} }
+27 -8
View File
@@ -2,17 +2,35 @@ import { redirect } from '@sveltejs/kit';
import { api } from '$lib/api'; import { api } from '$lib/api';
import { featureFlags } from '$lib/features'; import { featureFlags } from '$lib/features';
import { getStoredClientSession, hasModuleAccess, hasStoredClientSession } from '$lib/session'; import { getStoredClientSession, hasModuleAccess, hasStoredClientSession } from '$lib/session';
import { canOpenMixCalculator, getWorkspaceHomeHref } from '$lib/workspace-access'; import { canCreateMixSession, canOpenMixCalculator, getWorkspaceHomeHref } from '$lib/workspace-access';
export async function load({ fetch }) { export async function load({ fetch }) {
// Single-page mode (session history disabled): this route IS the calculator.
if (!featureFlags.mixCalculatorSessionHistory) { if (!featureFlags.mixCalculatorSessionHistory) {
throw redirect(307, '/mix-calculator/new'); if (!hasStoredClientSession()) {
return { options: { clients: [], products: [] } };
} }
if (!hasStoredClientSession()) { const session = getStoredClientSession();
if (!canCreateMixSession(session)) {
throw redirect(307, getWorkspaceHomeHref(session));
}
try {
return { return {
sessions: [] options:
hasModuleAccess(session, 'mix_calculator', 'edit') || session?.role === 'internal'
? await api.mixCalculatorOptions(fetch)
: { clients: [], products: [] }
}; };
} catch {
return { options: { clients: [], products: [] } };
}
}
// History mode: list saved sessions.
if (!hasStoredClientSession()) {
return { sessions: [] };
} }
const session = getStoredClientSession(); const session = getStoredClientSession();
@@ -22,11 +40,12 @@ export async function load({ fetch }) {
try { try {
return { return {
sessions: hasModuleAccess(session, 'mix_calculator') || session?.role === 'internal' ? await api.mixCalculatorSessions(fetch) : [] sessions:
hasModuleAccess(session, 'mix_calculator') || session?.role === 'internal'
? await api.mixCalculatorSessions(fetch)
: []
}; };
} catch { } catch {
return { return { sessions: [] };
sessions: []
};
} }
} }
@@ -15,7 +15,7 @@
{#if featureFlags.mixCalculatorSessionHistory} {#if featureFlags.mixCalculatorSessionHistory}
<a class="secondary-button" href="/mix-calculator">Back to session history</a> <a class="secondary-button" href="/mix-calculator">Back to session history</a>
{:else} {:else}
<a class="secondary-button" href="/mix-calculator/new">New mix session</a> <a class="secondary-button" href="/mix-calculator">New mix session</a>
{/if} {/if}
</section> </section>
{/if} {/if}
@@ -38,7 +38,7 @@
max-width: 42rem; max-width: 42rem;
padding: 1.25rem; padding: 1.25rem;
border: 1px solid var(--line); border: 1px solid var(--line);
border-radius: 1.25rem; border-radius: 0.8rem;
background: var(--panel); background: var(--panel);
box-shadow: var(--shadow); box-shadow: var(--shadow);
} }
@@ -58,7 +58,7 @@
margin-top: 1rem; margin-top: 1rem;
padding: 0.78rem 0.92rem; padding: 0.78rem 0.92rem;
border: 1px solid var(--line-strong); border: 1px solid var(--line-strong);
border-radius: 0.88rem; border-radius: 0.6rem;
background: #fff; background: #fff;
color: #304038; color: #304038;
font-weight: 600; font-weight: 600;
@@ -15,7 +15,7 @@
{#if featureFlags.mixCalculatorSessionHistory} {#if featureFlags.mixCalculatorSessionHistory}
<a class="secondary-button" href="/mix-calculator">Back to session history</a> <a class="secondary-button" href="/mix-calculator">Back to session history</a>
{:else} {:else}
<a class="secondary-button" href="/mix-calculator/new">New mix session</a> <a class="secondary-button" href="/mix-calculator">New mix session</a>
{/if} {/if}
</section> </section>
{/if} {/if}
+1 -1
View File
@@ -31,7 +31,7 @@ describe('root route access', () => {
sessionMocks.hasModuleAccess.mockReturnValue(false); sessionMocks.hasModuleAccess.mockReturnValue(false);
expect(() => load({ fetch: vi.fn() as typeof fetch })).toThrow( expect(() => load({ fetch: vi.fn() as typeof fetch })).toThrow(
expect.objectContaining({ status: 307, location: '/mix-calculator/new' }) expect.objectContaining({ status: 307, location: '/mix-calculator' })
); );
}); });
@@ -30,6 +30,6 @@ describe('settings route access', () => {
sessionMocks.hasPermission.mockImplementation((session, key) => session?.permissions?.includes(key) ?? false); sessionMocks.hasPermission.mockImplementation((session, key) => session?.permissions?.includes(key) ?? false);
sessionMocks.hasModuleAccess.mockReturnValue(false); sessionMocks.hasModuleAccess.mockReturnValue(false);
expect(() => load()).toThrow(expect.objectContaining({ status: 307, location: '/mix-calculator/new' })); expect(() => load()).toThrow(expect.objectContaining({ status: 307, location: '/mix-calculator' }));
}); });
}); });
File diff suppressed because it is too large Load Diff
+30
View File
@@ -0,0 +1,30 @@
import { redirect } from '@sveltejs/kit';
import { api } from '$lib/api';
import { getStoredClientSession, hasStoredClientSession } from '$lib/session';
import { canOpenThroughput, getWorkspaceHomeHref } from '$lib/workspace-access';
export async function load({ fetch }) {
if (!hasStoredClientSession()) {
return { entries: [], products: [] };
}
const session = getStoredClientSession();
if (!canOpenThroughput(session)) {
throw redirect(307, getWorkspaceHomeHref(session));
}
// Default view shows the last 30 days; "Find past entries" surfaces older ones.
const recentFrom = new Date();
recentFrom.setDate(recentFrom.getDate() - 30);
const dateFrom = recentFrom.toISOString().slice(0, 10);
try {
const [entries, products] = await Promise.all([
api.throughputEntries({ date_from: dateFrom, limit: 200 }, fetch),
api.throughputProducts(fetch)
]);
return { entries, products };
} catch {
return { entries: [], products: [] };
}
}
@@ -0,0 +1,389 @@
<script lang="ts">
import { goto } from '$app/navigation';
import { api } from '$lib/api';
import type {
ThroughputEntryCreateInput,
ThroughputProduct,
ThroughputQuantityType
} from '$lib/types';
import { CheckCircle2, AlertTriangle, ArrowLeft } from 'lucide-svelte';
let { data } = $props<{ data: { products: ThroughputProduct[] } }>();
const products = $derived(data.products ?? []);
const today = new Date().toISOString().slice(0, 10);
let productionDate = $state(today);
let productId = $state<string>('');
let bagSize = $state<string>('');
let quantity = $state<string>('');
let quantityType = $state<ThroughputQuantityType>('bags');
let scalesChecked = $state(true);
let labelCorrect = $state(true);
let bagSealed = $state(true);
let palletGood = $state(true);
let sampleBoxNo = $state('');
let tw1 = $state('');
let tw2 = $state('');
let tw3 = $state('');
let tw4 = $state('');
let tw5 = $state('');
let staffName = $state('');
let notes = $state('');
let saving = $state(false);
let successMessage = $state('');
let errorMessage = $state('');
const selectedProduct = $derived(
productId ? products.find((p: ThroughputProduct) => String(p.id) === productId) ?? null : null
);
$effect(() => {
if (selectedProduct) {
if (!bagSize && selectedProduct.default_bag_size != null) {
bagSize = String(selectedProduct.default_bag_size);
}
if (selectedProduct.is_bulka_default) {
quantityType = 'kg';
}
}
});
const qaWarning = $derived(!(scalesChecked && labelCorrect && bagSealed && palletGood));
function toNum(value: string): number | null {
const trimmed = value.trim();
if (!trimmed) return null;
const n = Number(trimmed);
return Number.isFinite(n) ? n : null;
}
function resetExceptDateAndStaff() {
productId = '';
bagSize = '';
quantity = '';
quantityType = 'bags';
scalesChecked = true;
labelCorrect = true;
bagSealed = true;
palletGood = true;
sampleBoxNo = '';
tw1 = '';
tw2 = '';
tw3 = '';
tw4 = '';
tw5 = '';
notes = '';
}
function buildPayload(): ThroughputEntryCreateInput | null {
const qty = toNum(quantity);
if (qty === null || qty < 0) {
errorMessage = 'Quantity is required and must be 0 or greater.';
return null;
}
if (!productionDate) {
errorMessage = 'Date is required.';
return null;
}
if (!productId) {
errorMessage = 'Product is required.';
return null;
}
const bag = toNum(bagSize);
if (quantityType === 'bags' && (bag === null || bag <= 0)) {
errorMessage = 'Bag size is required when quantity type is "bags".';
return null;
}
return {
production_date: productionDate,
product_id: Number(productId),
product_name_snapshot: selectedProduct?.name ?? '',
bag_size: bag,
scales_checked: scalesChecked,
label_correct: labelCorrect,
bag_sealed: bagSealed,
pallet_good_condition: palletGood,
sample_box_no: sampleBoxNo.trim() || null,
test_weight_1: toNum(tw1),
test_weight_2: toNum(tw2),
test_weight_3: toNum(tw3),
test_weight_4: toNum(tw4),
test_weight_5: toNum(tw5),
quantity: qty,
quantity_type: quantityType,
staff_name: staffName.trim() || null,
notes: notes.trim() || null
};
}
async function submit(mode: 'save' | 'save-and-add') {
errorMessage = '';
successMessage = '';
const payload = buildPayload();
if (!payload) return;
saving = true;
try {
await api.createThroughputEntry(payload);
if (mode === 'save') {
await goto('/throughput');
return;
}
successMessage = 'Entry saved. Ready for the next one.';
resetExceptDateAndStaff();
} catch (err) {
errorMessage = err instanceof Error ? err.message : 'Failed to save entry';
} finally {
saving = false;
}
}
</script>
<section class="add-entry">
<header class="page-header">
<a class="back-link" href="/throughput"><ArrowLeft size={16} /> Back to log</a>
<h1>Add Throughput Entry</h1>
</header>
{#if successMessage}
<div class="banner banner-ok"><CheckCircle2 size={16} /> {successMessage}</div>
{/if}
{#if errorMessage}
<div class="banner banner-error"><AlertTriangle size={16} /> {errorMessage}</div>
{/if}
<form onsubmit={(e) => { e.preventDefault(); submit('save'); }}>
<div class="grid">
<label>
<span>Date *</span>
<input type="date" bind:value={productionDate} required />
</label>
<label>
<span>Product *</span>
<select bind:value={productId} required>
<option value="">Select product…</option>
{#each products as p (p.id)}
<option value={String(p.id)}>{p.name}{p.item_id ? ` · ${p.item_id}` : ''}</option>
{/each}
</select>
</label>
<label>
<span>Bag size (kg)</span>
<input type="number" min="0" step="0.01" bind:value={bagSize} placeholder="e.g. 20" />
</label>
<label>
<span>Quantity *</span>
<input type="number" min="0" step="0.01" bind:value={quantity} required />
</label>
<label>
<span>Quantity type *</span>
<select bind:value={quantityType}>
<option value="bags">Bags</option>
<option value="kg">Kilograms (bulka / bulk)</option>
</select>
</label>
<label>
<span>Sample box no.</span>
<input type="text" bind:value={sampleBoxNo} />
</label>
</div>
<fieldset class="qa">
<legend>QA checklist</legend>
<label class="check"><input type="checkbox" bind:checked={scalesChecked} /> Scales checked</label>
<label class="check"><input type="checkbox" bind:checked={labelCorrect} /> Label correct</label>
<label class="check"><input type="checkbox" bind:checked={bagSealed} /> Bag sealed</label>
<label class="check"><input type="checkbox" bind:checked={palletGood} /> Pallet in good condition</label>
{#if qaWarning}
<p class="qa-warning"><AlertTriangle size={14} /> One or more QA checks failed — this entry will be flagged.</p>
{/if}
</fieldset>
<fieldset class="weights">
<legend>Test weights (optional, RCP 001)</legend>
<div class="weight-grid">
<label><span>1</span><input type="number" step="0.01" bind:value={tw1} /></label>
<label><span>2</span><input type="number" step="0.01" bind:value={tw2} /></label>
<label><span>3</span><input type="number" step="0.01" bind:value={tw3} /></label>
<label><span>4</span><input type="number" step="0.01" bind:value={tw4} /></label>
<label><span>5</span><input type="number" step="0.01" bind:value={tw5} /></label>
</div>
</fieldset>
<div class="grid">
<label>
<span>Staff name</span>
<input type="text" bind:value={staffName} placeholder="e.g. Jake" />
</label>
<label class="full">
<span>Notes</span>
<textarea rows="2" bind:value={notes}></textarea>
</label>
</div>
<div class="actions">
<button type="submit" class="primary-button" disabled={saving}>Save</button>
<button type="button" class="secondary-button" disabled={saving} onclick={() => submit('save-and-add')}>
Save and add another
</button>
<a href="/throughput" class="ghost-button">Cancel</a>
</div>
</form>
</section>
<style>
.add-entry {
display: flex;
flex-direction: column;
gap: 1.25rem;
padding: 1rem 0;
max-width: 980px;
}
.page-header {
display: flex;
flex-direction: column;
gap: 0.4rem;
}
.page-header h1 {
margin: 0;
font-size: 1.5rem;
}
.back-link {
display: inline-flex;
align-items: center;
gap: 0.3rem;
color: var(--text-muted, #6b7280);
font-size: 0.85rem;
text-decoration: none;
}
form {
display: flex;
flex-direction: column;
gap: 1rem;
padding: 1.25rem;
background: var(--surface, #fff);
border: 1px solid var(--border, #e5e7eb);
border-radius: 0.65rem;
}
.grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 0.75rem;
}
label {
display: flex;
flex-direction: column;
gap: 0.3rem;
font-size: 0.85rem;
}
label.full {
grid-column: 1 / -1;
}
input,
select,
textarea {
padding: 0.5rem 0.65rem;
border: 1px solid var(--border, #d1d5db);
border-radius: 0.4rem;
font: inherit;
}
fieldset {
border: 1px solid var(--border, #e5e7eb);
border-radius: 0.5rem;
padding: 0.75rem 1rem;
display: flex;
flex-direction: column;
gap: 0.5rem;
}
legend {
padding: 0 0.4rem;
font-weight: 600;
font-size: 0.85rem;
}
.qa {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 0.4rem 1rem;
}
.check {
flex-direction: row;
align-items: center;
gap: 0.45rem;
}
.qa-warning {
grid-column: 1 / -1;
margin: 0;
color: #92400e;
display: inline-flex;
align-items: center;
gap: 0.35rem;
font-size: 0.85rem;
}
.weight-grid {
display: grid;
grid-template-columns: repeat(5, minmax(0, 1fr));
gap: 0.5rem;
}
.weight-grid label {
flex-direction: column;
gap: 0.25rem;
}
.weight-grid label span {
font-size: 0.75rem;
color: var(--text-muted, #6b7280);
}
.actions {
display: flex;
gap: 0.5rem;
flex-wrap: wrap;
}
.primary-button {
padding: 0.6rem 1rem;
background: var(--accent, #1f2937);
color: white;
border-radius: 0.5rem;
border: 0;
font-weight: 600;
cursor: pointer;
}
.secondary-button {
padding: 0.6rem 1rem;
background: white;
border: 1px solid var(--border, #d1d5db);
border-radius: 0.5rem;
font-weight: 600;
cursor: pointer;
}
.ghost-button {
padding: 0.6rem 1rem;
border-radius: 0.5rem;
text-decoration: none;
color: var(--text-muted, #6b7280);
align-self: center;
}
.banner {
display: flex;
align-items: center;
gap: 0.5rem;
padding: 0.65rem 0.85rem;
border-radius: 0.5rem;
}
.banner-ok {
background: #ecfdf5;
color: #065f46;
border: 1px solid #a7f3d0;
}
.banner-error {
background: #fef2f2;
color: #991b1b;
border: 1px solid #fecaca;
}
</style>
@@ -0,0 +1,22 @@
import { redirect } from '@sveltejs/kit';
import { api } from '$lib/api';
import { getStoredClientSession, hasStoredClientSession } from '$lib/session';
import { canEditThroughput, getWorkspaceHomeHref } from '$lib/workspace-access';
export async function load({ fetch }) {
if (!hasStoredClientSession()) {
return { products: [] };
}
const session = getStoredClientSession();
if (!canEditThroughput(session)) {
throw redirect(307, getWorkspaceHomeHref(session));
}
try {
const products = await api.throughputProducts(fetch);
return { products };
} catch {
return { products: [] };
}
}
BIN
View File
Binary file not shown.