tweaks
This commit is contained in:
@@ -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
|
||||||
|
|||||||
@@ -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 product’s 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)
|
||||||
@@ -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 repo’s 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.
|
||||||
|
|||||||
@@ -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
@@ -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(
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
@@ -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
|
||||||
@@ -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
|
||||||
|
|||||||
@@ -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"),
|
||||||
|
|||||||
@@ -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(
|
||||||
|
|||||||
+9
-3
@@ -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")
|
||||||
info("Docs available at /docs", logger_name="data_entry_app.services")
|
if settings.docs_enabled:
|
||||||
|
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",
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -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",
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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")
|
||||||
@@ -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
@@ -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()
|
||||||
|
|||||||
@@ -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"
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -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,12 +264,22 @@ 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
|
||||||
|
|
||||||
mix_result = calculate_mix_cost(db, product.mix_id, overrides=overrides)
|
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)
|
||||||
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)
|
||||||
|
|||||||
@@ -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} · {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),
|
logo_reader = ImageReader(logo_source)
|
||||||
Paragraph(session_record.mix_date.strftime("%d %b %Y"), value),
|
logo_width, logo_height = logo_reader.getSize()
|
||||||
],
|
aspect_ratio = logo_height / max(logo_width, 1)
|
||||||
[
|
draw_width = 108
|
||||||
Paragraph("Prepared by", label),
|
draw_height = draw_width * aspect_ratio
|
||||||
Paragraph(session_record.prepared_by_name, value),
|
pdf.drawImage(
|
||||||
],
|
logo_reader,
|
||||||
[
|
margin,
|
||||||
Paragraph("Status", label),
|
current_y - draw_height,
|
||||||
Paragraph(session_record.status.title(), value),
|
width=draw_width,
|
||||||
],
|
height=draw_height,
|
||||||
]
|
preserveAspectRatio=True,
|
||||||
],
|
mask="auto",
|
||||||
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),
|
|
||||||
]
|
|
||||||
)
|
)
|
||||||
)
|
current_y -= draw_height + 20
|
||||||
story.extend([header_table, Spacer(1, 10)])
|
|
||||||
|
|
||||||
summary_table = Table(
|
pdf.setFillColor(palette["text"])
|
||||||
[
|
pdf.setFont("Helvetica-Bold", 15)
|
||||||
[
|
pdf.drawString(margin, current_y, "Calculated Output")
|
||||||
[Paragraph("Batch size", label), Paragraph(f"{_fmt_number(session_record.batch_size_kg)}kg", card_value)],
|
current_y -= 16
|
||||||
[Paragraph("Total output", label), Paragraph(f"{_fmt_number(session_record.total_kg)}kg", card_value)],
|
|
||||||
[Paragraph("Bags", label), Paragraph(_fmt_number(session_record.total_bags), card_value)],
|
|
||||||
[Paragraph("Unit pack", label), Paragraph(f"{_fmt_number(session_record.product_unit_size_kg)}kg", card_value)],
|
|
||||||
]
|
|
||||||
],
|
|
||||||
colWidths=[43 * mm, 43 * mm, 43 * mm, 43 * mm],
|
|
||||||
)
|
|
||||||
summary_table.setStyle(
|
|
||||||
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["muted"])
|
||||||
[
|
pdf.setFont("Helvetica", 10)
|
||||||
[
|
pdf.drawString(margin, current_y, "Snapshot of the scaled raw material requirements.")
|
||||||
[Paragraph("Mix source", label), Paragraph(session_record.mix_name, value), Paragraph(f"Saved against {session_record.product_unit_of_measure} units.", body)],
|
current_y -= 20
|
||||||
[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)],
|
stat_height = 66
|
||||||
]
|
stat_width = (content_width - (gutter * 2)) / 3
|
||||||
],
|
draw_label_value_card(
|
||||||
colWidths=[60 * mm, 60 * mm, 52 * mm],
|
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,
|
||||||
)
|
|
||||||
)
|
)
|
||||||
story.extend([detail_table, Spacer(1, 10)])
|
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,
|
||||||
|
)
|
||||||
|
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()
|
||||||
|
|||||||
@@ -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,10 +28,44 @@ 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:
|
||||||
rounded_bags = round(total_bags)
|
rounded_bags = round(total_bags)
|
||||||
if abs(total_bags - rounded_bags) < 1e-9:
|
if abs(total_bags - rounded_bags) < 1e-9:
|
||||||
@@ -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
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -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
|
||||||
@@ -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 = [
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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():
|
||||||
|
|||||||
@@ -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
|
||||||
+443
-40
@@ -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] $EnvFile = ".env.production",
|
[string] $AppSlug = "clients",
|
||||||
|
[string] $RemotePath,
|
||||||
|
[string] $BackendContainer,
|
||||||
|
[string] $PortEnvKey,
|
||||||
|
[string] $EnvFile = ".env.production",
|
||||||
[string] $SshKey,
|
[string] $SshKey,
|
||||||
[string] $ComposeFile = "docker-compose.production.yml",
|
[string] $Password,
|
||||||
|
[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
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
Generated
+12
-2
@@ -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",
|
||||||
|
|||||||
@@ -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
@@ -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),
|
||||||
|
|||||||
@@ -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;
|
|
||||||
}
|
|
||||||
|
|
||||||
previewModalOpen = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
function closePreviewModal() {
|
|
||||||
previewModalOpen = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function saveSession(mode: 'update' | 'create', destination: 'detail' | 'print' = 'detail') {
|
|
||||||
const payload = buildPayload();
|
|
||||||
if (!payload) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
saveLoading = true;
|
|
||||||
const tid = toast.loading(mode === 'update' ? 'Saving session…' : 'Creating session…');
|
|
||||||
try {
|
try {
|
||||||
const saved =
|
const payload = buildPayload();
|
||||||
mode === 'update' && initialSession
|
if (!payload) {
|
||||||
? await api.updateMixCalculatorSession(initialSession.id, payload)
|
toast.dismiss(tid);
|
||||||
: await api.createMixCalculatorSession(payload);
|
toast.error(formError || 'Complete the mix details first.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const blob = await api.previewMixCalculatorPdf(payload);
|
||||||
|
const url = URL.createObjectURL(blob);
|
||||||
|
window.open(url, '_blank', 'noopener,noreferrer');
|
||||||
|
setTimeout(() => URL.revokeObjectURL(url), 60_000);
|
||||||
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>© {currentYear} Hunter Premium Produce</small>
|
<small>© {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) }];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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>;
|
||||||
|
|||||||
@@ -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', () => {
|
||||||
|
|||||||
@@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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 {
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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>© {currentYear} Hunter Premium Produce</span>
|
<span>© {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>© {currentYear} Lean 101</span>
|
<span>© {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,
|
||||||
|
|||||||
@@ -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 () => {
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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: [] } };
|
||||||
|
}
|
||||||
|
|
||||||
|
const session = getStoredClientSession();
|
||||||
|
if (!canCreateMixSession(session)) {
|
||||||
|
throw redirect(307, getWorkspaceHomeHref(session));
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
return {
|
||||||
|
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()) {
|
if (!hasStoredClientSession()) {
|
||||||
return {
|
return { sessions: [] };
|
||||||
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}
|
||||||
|
|||||||
@@ -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
@@ -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: [] };
|
||||||
|
}
|
||||||
|
}
|
||||||
Binary file not shown.
Reference in New Issue
Block a user