Compare commits

..

57 Commits

Author SHA1 Message Date
Daniel Volz e8279bd521 chore: release v1.19.0 (#388)
* docs: require explicit issue comment when closing issues via PR

* chore: release v1.19.0
2026-03-06 20:16:42 +01:00
dependabot[bot] 36d50c0736 build(deps): bump fastify from 5.7.4 to 5.8.1 in /backend (#387)
Bumps [fastify](https://github.com/fastify/fastify) from 5.7.4 to 5.8.1.
- [Release notes](https://github.com/fastify/fastify/releases)
- [Commits](https://github.com/fastify/fastify/compare/v5.7.4...v5.8.1)

---
updated-dependencies:
- dependency-name: fastify
  dependency-version: 5.8.1
  dependency-type: direct:production
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-03-06 20:02:08 +01:00
Daniel Volz 5b6c6abb69 feat: display actual reminder schedule from server config (#386)
- Expose REMINDER_HOUR and REMINDER_MINUTES_BEFORE env values via settings API
- Add reminderHour and reminderMinutesBefore to frontend Settings interface
- Replace hardcoded i18n strings with parameterized translations
- Settings page now shows configured schedule instead of static 6:00 / 15 min
2026-03-06 19:51:19 +01:00
Daniel Volz 30c97e2f0d fix: use per-intake reminder setting as single source of truth (#384)
- Filter intakes by per-intake intakeRemindersEnabled instead of falling
  back to medication-level setting (fixes #383)
- Add SMTP delivery validation with accepted/rejected recipient checks
- Enhance email success logging with recipient, messageId, SMTP response
- Simplify MedDetailModal reminder icon logic to match backend behavior
- Sync lockfile versions to 1.18.2
2026-03-06 19:50:45 +01:00
Daniel Volz de1a508e52 chore: ignore copilot tracking artifacts (#382) 2026-03-04 21:15:42 +01:00
Daniel Volz 54d26e0241 fix: remove redundant flaky schedule timeline assertion (#381)
* test: remove redundant flaky schedule timeline assertion

* fix(frontend): fix schedule-data e2e formatting for CI gate
2026-03-04 21:15:29 +01:00
Daniel Volz ac47fc001d fix: adapt e2e medication flows to all package profiles (#380)
* test: adapt e2e medication flows to all package profiles

* fix(frontend): resolve frontend build lint blockers
2026-03-04 21:15:18 +01:00
Daniel Volz 4936929849 feat: replace hardcoded package assumptions with profile abstraction (#379) 2026-03-04 21:15:05 +01:00
Daniel Volz 6672fb78c9 chore: stop tracking doku memory/report files 2026-03-02 23:41:57 +01:00
Daniel Volz b349e26833 chore: release v1.18.2 (#374) 2026-03-02 23:34:18 +01:00
Daniel Volz 56d244aa61 fix: stabilize frontend e2e selectors and auth/session reliability (#373) 2026-03-02 23:21:57 +01:00
dependabot[bot] 1a348c62f5 build(deps-dev): bump lint-staged in the minor-and-patch group (#369)
Bumps the minor-and-patch group with 1 update: [lint-staged](https://github.com/lint-staged/lint-staged).


Updates `lint-staged` from 16.2.7 to 16.3.1
- [Release notes](https://github.com/lint-staged/lint-staged/releases)
- [Changelog](https://github.com/lint-staged/lint-staged/blob/main/CHANGELOG.md)
- [Commits](https://github.com/lint-staged/lint-staged/compare/v16.2.7...v16.3.1)

---
updated-dependencies:
- dependency-name: lint-staged
  dependency-version: 16.3.1
  dependency-type: direct:development
  update-type: version-update:semver-minor
  dependency-group: minor-and-patch
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: Daniel Volz <mail@danielvolz.org>
2026-03-02 13:22:32 +01:00
dependabot[bot] 067a8c166b build(deps-dev): bump @types/node (#371)
Bumps the minor-and-patch group in /backend with 1 update: [@types/node](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/node).


Updates `@types/node` from 25.3.2 to 25.3.3
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/node)

---
updated-dependencies:
- dependency-name: "@types/node"
  dependency-version: 25.3.3
  dependency-type: direct:development
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-03-02 12:56:23 +01:00
dependabot[bot] 8fdd79ff33 build(deps-dev): bump @types/node (#370)
Bumps the minor-and-patch group in /frontend with 1 update: [@types/node](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/node).


Updates `@types/node` from 25.3.2 to 25.3.3
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/node)

---
updated-dependencies:
- dependency-name: "@types/node"
  dependency-version: 25.3.3
  dependency-type: direct:development
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-03-02 12:56:16 +01:00
Daniel Volz cd8263e607 fix: align desktop intake labels and form field pairing (#368)
* fix: align desktop intake labels and form field pairing

* chore: align frontend lockfile version and include remaining local changes
2026-03-02 01:33:28 +01:00
Daniel Volz e6a097d81d chore: release v1.18.1 (#366) 2026-03-02 01:16:08 +01:00
Daniel Volz f4723c6f99 chore: release v1.18.0 (#365) 2026-03-02 00:41:20 +01:00
github-actions[bot] aad6b143ef chore: update test count badges [skip ci] 2026-03-01 23:27:20 +00:00
Daniel Volz da004b5c3e fix: align frontend tube/liquid container semantics (#364)
* fix: align frontend tube/liquid container semantics

* test(frontend): fix PR #364 CI regressions
2026-03-02 00:23:32 +01:00
github-actions[bot] cd18581bdd chore: update test count badges [skip ci] 2026-03-01 23:07:44 +00:00
Daniel Volz 508bc764d5 fix: align backend amount stock and reminder semantics (#362)
* fix: align backend amount stock and reminder semantics

* test: align settings email route success mock with SMTP delivery checks
2026-03-02 00:02:26 +01:00
Daniel Volz 9e8a6315e7 fix: keep topical stock non-depleting in planner flows (#359)
* fix: keep topical stock non-depleting in planner and reports

* test: stabilize e2e selectors for updated medication semantics

* fix(backend): add missing planner translation keys
2026-02-28 23:36:52 +01:00
github-actions[bot] 8efd99d738 chore: update test count badges [skip ci] 2026-02-28 22:28:41 +00:00
github-actions[bot] dc98dfda44 chore: update test count badges [skip ci] 2026-02-28 22:25:25 +00:00
Daniel Volz 8aaeca6b26 feat: simplify tube stock editing UI (#357)
* feat: add package amount persistence and backend route support

* test: align backend test schemas with medication metadata fields

* fix(backend): restore intake usage normalizer for planner endpoint

* fix(backend): keep export typing compatible before liquid-unit stack step

* feat: simplify tube stock editing in desktop and mobile forms
2026-02-28 23:24:48 +01:00
Daniel Volz 7accb2aad6 feat: persist package amount metadata in backend (#356)
* feat: add package amount persistence and backend route support

* test: align backend test schemas with medication metadata fields

* fix(backend): restore intake usage normalizer for planner endpoint

* fix(backend): keep export typing compatible before liquid-unit stack step
2026-02-28 23:21:13 +01:00
Daniel Volz 2f2edfa479 chore: release v1.17.1 (#351) 2026-02-27 01:53:09 +01:00
Daniel Volz b009d9e158 fix: frontend nginx restart loop from invalid log_format scope (#350)
* fix: place nginx log_format in valid context

* fix: unblock required checks for nginx hotfix

* fix: restore mandatory doku files in nginx hotfix pr
2026-02-27 01:47:59 +01:00
dependabot[bot] 8e4cb5dcd4 build(deps): bump minimatch from 10.2.2 to 10.2.4 in /backend (#338)
Bumps [minimatch](https://github.com/isaacs/minimatch) from 10.2.2 to 10.2.4.
- [Changelog](https://github.com/isaacs/minimatch/blob/main/changelog.md)
- [Commits](https://github.com/isaacs/minimatch/compare/v10.2.2...v10.2.4)

---
updated-dependencies:
- dependency-name: minimatch
  dependency-version: 10.2.4
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-27 01:29:04 +01:00
dependabot[bot] 7f26dca7a7 build(deps-dev): bump @types/node (#343)
Bumps the minor-and-patch group in /backend with 1 update: [@types/node](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/node).


Updates `@types/node` from 25.3.0 to 25.3.2
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/node)

---
updated-dependencies:
- dependency-name: "@types/node"
  dependency-version: 25.3.2
  dependency-type: direct:development
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-27 01:29:00 +01:00
dependabot[bot] 46d768dd4e build(deps): bump the minor-and-patch group in /frontend with 5 updates (#344)
Bumps the minor-and-patch group in /frontend with 5 updates:

| Package | From | To |
| --- | --- | --- |
| [react](https://github.com/facebook/react/tree/HEAD/packages/react) | `19.2.0` | `19.2.4` |
| [@types/react](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/react) | `19.2.2` | `19.2.14` |
| [react-dom](https://github.com/facebook/react/tree/HEAD/packages/react-dom) | `19.2.0` | `19.2.4` |
| [@types/react-dom](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/react-dom) | `19.2.2` | `19.2.3` |
| [@types/node](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/node) | `25.3.0` | `25.3.2` |


Updates `react` from 19.2.0 to 19.2.4
- [Release notes](https://github.com/facebook/react/releases)
- [Changelog](https://github.com/facebook/react/blob/main/CHANGELOG.md)
- [Commits](https://github.com/facebook/react/commits/v19.2.4/packages/react)

Updates `@types/react` from 19.2.2 to 19.2.14
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/react)

Updates `react-dom` from 19.2.0 to 19.2.4
- [Release notes](https://github.com/facebook/react/releases)
- [Changelog](https://github.com/facebook/react/blob/main/CHANGELOG.md)
- [Commits](https://github.com/facebook/react/commits/v19.2.4/packages/react-dom)

Updates `@types/react-dom` from 19.2.2 to 19.2.3
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/react-dom)

Updates `@types/node` from 25.3.0 to 25.3.2
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/node)

Updates `@types/react` from 19.2.2 to 19.2.14
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/react)

Updates `@types/react-dom` from 19.2.2 to 19.2.3
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/react-dom)

---
updated-dependencies:
- dependency-name: react
  dependency-version: 19.2.4
  dependency-type: direct:production
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
- dependency-name: "@types/react"
  dependency-version: 19.2.14
  dependency-type: direct:development
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
- dependency-name: react-dom
  dependency-version: 19.2.4
  dependency-type: direct:production
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
- dependency-name: "@types/react-dom"
  dependency-version: 19.2.3
  dependency-type: direct:development
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
- dependency-name: "@types/node"
  dependency-version: 25.3.2
  dependency-type: direct:development
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
- dependency-name: "@types/react"
  dependency-version: 19.2.14
  dependency-type: direct:development
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
- dependency-name: "@types/react-dom"
  dependency-version: 19.2.3
  dependency-type: direct:development
  update-type: version-update:semver-patch
  dependency-group: minor-and-patch
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-27 01:28:55 +01:00
dependabot[bot] c62b6d7893 build(deps): bump actions/upload-artifact from 6 to 7 (#345)
Bumps [actions/upload-artifact](https://github.com/actions/upload-artifact) from 6 to 7.
- [Release notes](https://github.com/actions/upload-artifact/releases)
- [Commits](https://github.com/actions/upload-artifact/compare/v6...v7)

---
updated-dependencies:
- dependency-name: actions/upload-artifact
  dependency-version: '7'
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-27 01:28:51 +01:00
dependabot[bot] 1668eb935c build(deps-dev): bump @types/supertest from 6.0.3 to 7.2.0 in /backend (#346)
Bumps [@types/supertest](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/supertest) from 6.0.3 to 7.2.0.
- [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases)
- [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/supertest)

---
updated-dependencies:
- dependency-name: "@types/supertest"
  dependency-version: 7.2.0
  dependency-type: direct:development
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-27 01:28:47 +01:00
Daniel Volz 1ea4919323 chore: release v1.17.0 (#348) 2026-02-27 01:19:39 +01:00
Daniel Volz ba0ab672b9 docs: update memory and report for multi-pr delivery (#347) 2026-02-27 01:15:40 +01:00
Daniel Volz 57c998ba09 chore: update dependabot automation and agent governance (#341)
* chore: update dependabot automation and agent governance

* chore: trigger required CI checks for governance PR
2026-02-27 01:11:05 +01:00
Daniel Volz cc22f80209 fix: align frontend types and tests for react 19 (#339) 2026-02-27 01:01:48 +01:00
Daniel Volz 6b27d234d9 chore: reduce polling log noise across backend and nginx (#336) 2026-02-27 00:54:21 +01:00
Daniel Volz 19ba4bb7d2 feat: add FORM_LOGIN_ENABLED auth toggle (#334) 2026-02-27 00:48:58 +01:00
dependabot[bot] 8b3901c1e1 build(deps): bump rollup from 4.53.5 to 4.59.0 in /frontend (#333)
Bumps [rollup](https://github.com/rollup/rollup) from 4.53.5 to 4.59.0.
- [Release notes](https://github.com/rollup/rollup/releases)
- [Changelog](https://github.com/rollup/rollup/blob/master/CHANGELOG.md)
- [Commits](https://github.com/rollup/rollup/compare/v4.53.5...v4.59.0)

---
updated-dependencies:
- dependency-name: rollup
  dependency-version: 4.59.0
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-26 06:29:10 +01:00
dependabot[bot] fd7cc56bb7 build(deps): bump rollup from 4.57.1 to 4.59.0 in /backend (#332)
Bumps [rollup](https://github.com/rollup/rollup) from 4.57.1 to 4.59.0.
- [Release notes](https://github.com/rollup/rollup/releases)
- [Changelog](https://github.com/rollup/rollup/blob/master/CHANGELOG.md)
- [Commits](https://github.com/rollup/rollup/compare/v4.57.1...v4.59.0)

---
updated-dependencies:
- dependency-name: rollup
  dependency-version: 4.59.0
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-26 06:28:51 +01:00
Daniel Volz aabe58d05f ci: add path filters to Docker build workflow
Only build Docker images when backend/, frontend/, docker-compose,
or the workflow itself changes. Prevents unnecessary image builds
for docs-only or config-only changes on main.

Note: paths filter is not evaluated for tag pushes (GitHub Actions
behavior), so release tags always trigger a full build.

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-02-26 00:22:08 +01:00
Daniel Volz b35101d339 docs: update AI model credits to Claude Opus 4.6 and GPT-5.3 Codex
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-02-26 00:16:10 +01:00
dependabot[bot] 8420c74a55 build(deps): bump bn.js from 4.12.2 to 4.12.3 in /backend (#330)
Bumps [bn.js](https://github.com/indutny/bn.js) from 4.12.2 to 4.12.3.
- [Release notes](https://github.com/indutny/bn.js/releases)
- [Changelog](https://github.com/indutny/bn.js/blob/master/CHANGELOG.md)
- [Commits](https://github.com/indutny/bn.js/compare/v4.12.2...v4.12.3)

---
updated-dependencies:
- dependency-name: bn.js
  dependency-version: 4.12.3
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-26 00:14:46 +01:00
Daniel Volz 872b63f665 docs: add explicit scope rule to release-manager agent
Prevent release-manager from chaining unrequested steps.
If user asks for PR+merge only, do not also start a release.
If user asks for release only, do not also create PRs.

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-02-26 00:03:07 +01:00
github-actions[bot] f599ac45ab chore: update test count badges [skip ci] 2026-02-25 22:57:37 +00:00
Daniel Volz f36d56c523 test: update modal tests to reflect global ESC handler
Remove ESC-keydown tests from ProfileModal.test.tsx since the
useEscapeKey hook was removed from individual modals. Escape key
handling is now centralized in App.tsx's global handler, making
per-component ESC tests invalid (the component no longer responds
to ESC in isolation).
2026-02-25 23:54:21 +01:00
Daniel Volz f0496e8ca5 fix: remove duplicate ESC handlers causing double history.back()
AboutModal, ProfileModal, and ShareDialog each had their own
useEscapeKey hook AND were handled by the global ESC handler in
App.tsx. When ESC was pressed, both fired synchronously, calling
history.back() twice — navigating past the current page instead
of just closing the modal.

Removed the per-modal useEscapeKey calls since the global handler
in App.tsx already manages ESC priority for all modals.

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-02-25 23:50:07 +01:00
Daniel Volz de300ad919 chore: release v1.16.1 (#326) 2026-02-25 22:26:19 +01:00
Daniel Volz 06bf608913 fix: support core shoutrrr provider schemes (#317)
* fix: support core shoutrrr provider schemes

* fix(ci): resolve backend lint failures and harden shoutrrr URL handling

* fix(codeql): mark validated gotify target as intentional external call

* fix(codeql): route gotify scheme through validated webhook sink
2026-02-25 22:22:50 +01:00
Daniel Volz a47bde0956 chore: sync lockfile package versions to 1.16.0 (#325)
* chore: sync lockfile package versions to 1.16.0

* fix(ci): align medications route formatting with biome
2026-02-25 22:15:43 +01:00
Daniel Volz d02f16af3a fix: stabilize e2e CI and local playwright workers (#321)
* fix: stabilize e2e CI and local playwright workers

* fix(ci): apply biome formatting and import order for frontend build
2026-02-25 22:15:38 +01:00
Daniel Volz dbdf3b61cb fix: harden reminder scheduler dedupe and boundary timing (#319)
* fix: harden reminder scheduler dedupe and boundary timing

* fix(ci): align medications route formatting with biome
2026-02-25 22:15:35 +01:00
Daniel Volz aa29d1c699 docs: align agent instructions and README guidance (#323) 2026-02-25 21:35:35 +01:00
Daniel Volz bfc9aaaa6d fix: tag releases on merged PR commit (#315) 2026-02-25 21:35:32 +01:00
Copilot 2a9ca39c24 Allow medications with only a generic name (no commercial name required) (#311)
* Initial plan

* feat: allow generic name only for medications (frontend changes)

- Add getMedDisplayName() helper for consistent name display
- Update validation to require either commercial or generic name
- Update all display locations to use display name fallback
- Add i18n keys for nameOrGenericRequired in en.json and de.json
- Remove required attribute from commercial name field
- Update FIELD_LIMITS.name.min from 1 to 0

Co-authored-by: DanielVolz <3275994+DanielVolz@users.noreply.github.com>

* feat: allow generic name only for medications (backend changes)

- Update Zod schema to allow empty name with cross-field refinement
- Update reminder scheduler to use name || genericName for display
- Update planner routes to match medications by display name
- Update existing tests to match new validation behavior

Co-authored-by: DanielVolz <3275994+DanielVolz@users.noreply.github.com>

* fix: update placeholder text and fix FIELD_LIMITS test

- Remove "(optional)" from generic name placeholder in en/de
- Update types.test.ts to expect FIELD_LIMITS.name.min = 0

Co-authored-by: DanielVolz <3275994+DanielVolz@users.noreply.github.com>

---------

Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com>
Co-authored-by: DanielVolz <3275994+DanielVolz@users.noreply.github.com>
2026-02-25 21:29:25 +01:00
dependabot[bot] 691550fb33 build(deps): bump bn.js from 4.12.2 to 4.12.3 in /backend (#305)
Bumps [bn.js](https://github.com/indutny/bn.js) from 4.12.2 to 4.12.3.
- [Release notes](https://github.com/indutny/bn.js/releases)
- [Changelog](https://github.com/indutny/bn.js/blob/master/CHANGELOG.md)
- [Commits](https://github.com/indutny/bn.js/compare/v4.12.2...v4.12.3)

---
updated-dependencies:
- dependency-name: bn.js
  dependency-version: 4.12.3
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-02-25 21:29:13 +01:00
119 changed files with 7037 additions and 1661 deletions
+12 -1
View File
@@ -11,10 +11,18 @@ PGID=1000
PORT=3000 PORT=3000
CORS_ORIGINS=http://localhost:4174 CORS_ORIGINS=http://localhost:4174
LOG_LEVEL=info LOG_LEVEL=warn
# Levels: debug, info, warn, error, silent # Levels: debug, info, warn, error, silent
# Controls: backend Fastify logging, frontend nginx access logs (Docker), # Controls: backend Fastify logging, frontend nginx access logs (Docker),
# and frontend browser console (via build-time injection) # and frontend browser console (via build-time injection)
#
# Behavior per level:
# debug — all app logs + all HTTP request logs (including polling endpoints)
# info — all app logs + HTTP request logs, EXCEPT high-frequency polling
# (GET /doses/taken, GET /share/:token/doses, GET /health are hidden)
# warn — only warnings and errors
# error — only errors
# silent — no logs
# Rate limit: max requests per minute per IP (default: 100) # Rate limit: max requests per minute per IP (default: 100)
# Increase for development/testing environments # Increase for development/testing environments
@@ -32,6 +40,9 @@ AUTH_ENABLED=false
# Allow new user registrations (auto-enabled when no users exist) # Allow new user registrations (auto-enabled when no users exist)
# REGISTRATION_ENABLED=false # REGISTRATION_ENABLED=false
# Disable username/password form login (useful for OIDC-only setups)
# FORM_LOGIN_ENABLED=true
# JWT Secrets - REQUIRED when AUTH_ENABLED=true # JWT Secrets - REQUIRED when AUTH_ENABLED=true
# Generate with: openssl rand -hex 32 # Generate with: openssl rand -hex 32
# JWT_SECRET= # JWT_SECRET=
@@ -0,0 +1,42 @@
---
description: 'Provide principal-level software engineering guidance with focus on engineering excellence, technical leadership, and pragmatic implementation.'
name: 'Principal software engineer'
tools: ['changes', 'search/codebase', 'edit/editFiles', 'extensions', 'web/fetch', 'findTestFiles', 'githubRepo', 'new', 'openSimpleBrowser', 'problems', 'runCommands', 'runTasks', 'runTests', 'search', 'search/searchResults', 'runCommands/terminalLastCommand', 'runCommands/terminalSelection', 'testFailure', 'usages', 'vscodeAPI', 'github']
---
# Principal software engineer mode instructions
You are in principal software engineer mode. Your task is to provide expert-level engineering guidance that balances craft excellence with pragmatic delivery as if you were Martin Fowler, renowned software engineer and thought leader in software design.
## Core Engineering Principles
You will provide guidance on:
- **Engineering Fundamentals**: Gang of Four design patterns, SOLID principles, DRY, YAGNI, and KISS - applied pragmatically based on context
- **Clean Code Practices**: Readable, maintainable code that tells a story and minimizes cognitive load
- **Test Automation**: Comprehensive testing strategy including unit, integration, and end-to-end tests with clear test pyramid implementation
- **Quality Attributes**: Balancing testability, maintainability, scalability, performance, security, and understandability
- **Technical Leadership**: Clear feedback, improvement recommendations, and mentoring through code reviews
## Implementation Focus
- **Requirements Analysis**: Carefully review requirements, document assumptions explicitly, identify edge cases and assess risks
- **Implementation Excellence**: Implement the best design that meets architectural requirements without over-engineering
- **Pragmatic Craft**: Balance engineering excellence with delivery needs - good over perfect, but never compromising on fundamentals
- **Forward Thinking**: Anticipate future needs, identify improvement opportunities, and proactively address technical debt
## Technical Debt Management
When technical debt is incurred or identified:
- **MUST** offer to create GitHub Issues using the `create_issue` tool to track remediation
- Clearly document consequences and remediation plans
- Regularly recommend GitHub Issues for requirements gaps, quality issues, or design improvements
- Assess long-term impact of untended technical debt
## Deliverables
- Clear, actionable feedback with specific improvement recommendations
- Risk assessments with mitigation strategies
- Edge case identification and testing strategies
- Explicit documentation of assumptions and decisions
- Technical debt remediation plans with GitHub Issue creation
+17 -9
View File
@@ -12,10 +12,14 @@ You are the release manager for **MedAssist-ng**. Your job is to guide code from
## Critical Safety Rules ## Critical Safety Rules
- **Do EXACTLY what the user asks — nothing more.** If the user says "create a PR and merge to main", do only that. Do NOT also start a release. If the user says "do a release", do only the release. Never chain additional steps the user did not request.
- **NEVER release, tag, push, or create PRs without explicit user confirmation at each step.** Always present your plan and wait for approval. - **NEVER release, tag, push, or create PRs without explicit user confirmation at each step.** Always present your plan and wait for approval.
- **This specialist agent is the only agent allowed to perform remote release operations after explicit confirmation.**
- **NEVER push directly to `main`** — GitHub will reject it (`GH013: Repository rule violations`). All changes go through Pull Requests. - **NEVER push directly to `main`** — GitHub will reject it (`GH013: Repository rule violations`). All changes go through Pull Requests.
- **NEVER skip CI checks.** Wait for all status checks to pass before merging. - **NEVER skip CI checks.** Wait for all status checks to pass before merging.
- **Testing ownership belongs to `@testing-manager`**. Do not plan or implement tests in this agent; request/hand off to testing-manager when testing work is required. - **Testing ownership belongs to `@testing-manager`**. Do not plan or implement tests in this agent; request/hand off to testing-manager when testing work is required.
- **Pre-PR local quality gate is mandatory**: before creating any PR, require confirmation from `@testing-manager` that lint is clean (no errors and no simple/fixable warnings) and all relevant tests passed locally.
- **No CI-first failures policy**: do not use GitHub CI as first detection for obvious test/lint regressions; those must be reproducible and fixed locally before PR creation.
- **Track all work in the GitHub Project board.** Every PR should reference an issue. Move issues through the board as work progresses. - **Track all work in the GitHub Project board.** Every PR should reference an issue. Move issues through the board as work progresses.
- **ALWAYS verify Project board status after merge.** The `project-auto-done.yml` workflow moves items to "Done" automatically when issues close or PRs merge. Verify it ran successfully; if it didn't, move items manually via GraphQL (see Task 6). - **ALWAYS verify Project board status after merge.** The `project-auto-done.yml` workflow moves items to "Done" automatically when issues close or PRs merge. Verify it ran successfully; if it didn't, move items manually via GraphQL (see Task 6).
@@ -48,12 +52,11 @@ This repository intentionally uses only two operational agents for CI/CD handoff
- Never use `gh` commands that can open an interactive pager and block execution (requiring `q`). - Never use `gh` commands that can open an interactive pager and block execution (requiring `q`).
- Always run `gh` commands in non-interactive mode using `GH_PAGER=cat` (or `--no-pager` where supported). - Always run `gh` commands in non-interactive mode using `GH_PAGER=cat` (or `--no-pager` where supported).
- Do not use these commands in agent flows: - Avoid hardcoded PR/repo examples in instructions; always use parameterized placeholders.
- `gh pr view 155 --json statusCheckRollup --jq '.statusCheckRollup[] | {name:.name,conclusion:.conclusion,detailsUrl:.detailsUrl,workflowName:.workflowName}'` - Use safe command patterns:
- `SHA=$(gh pr view 155 --json headRefOid --jq .headRefOid) && gh api repos/DanielVolz/medassist-ng/commits/$SHA/check-runs --jq '.check_runs[] | {name,conclusion,details_url,html_url,app:.app.name}'`
- Use safe variants instead:
- `GH_PAGER=cat gh pr view <PR_NUMBER> --json statusCheckRollup --jq '<jq-filter>'` - `GH_PAGER=cat gh pr view <PR_NUMBER> --json statusCheckRollup --jq '<jq-filter>'`
- `GH_PAGER=cat gh api repos/<owner>/<repo>/commits/<sha>/check-runs --jq '<jq-filter>'` - `SHA=$(GH_PAGER=cat gh pr view <PR_NUMBER> --json headRefOid --jq .headRefOid)`
- `GH_PAGER=cat gh api repos/<owner>/<repo>/commits/$SHA/check-runs --jq '<jq-filter>'`
--- ---
@@ -119,7 +122,9 @@ When code changes (features or bug fixes) are complete:
### Step 1: Verify Readiness ### Step 1: Verify Readiness
1. Check for uncommitted changes: `git status` 1. Check for uncommitted changes: `git status`
2. Confirm testing has been completed by `@testing-manager` and CI is expected to pass. 2. Confirm testing has been completed by `@testing-manager`.
3. Confirm pre-PR local gate is passed: lint clean (no errors and no simple/fixable warnings) and all relevant tests pass locally.
4. Only after local gate is confirmed, proceed to push/create PR and then monitor CI.
### Step 2: Create Feature Branch ### Step 2: Create Feature Branch
@@ -140,11 +145,12 @@ When code changes (features or bug fixes) are complete:
### Step 3: Push and Create PR ### Step 3: Push and Create PR
1. Push the branch: 1. Re-check local gate status before push/PR creation (lint + relevant local tests green).
2. Push the branch:
```bash ```bash
git push -u origin feat/short-description git push -u origin feat/short-description
``` ```
2. Create a Pull Request via GitHub CLI with **all metadata fields populated**: 3. Create a Pull Request via GitHub CLI with **all metadata fields populated**:
```bash ```bash
gh pr create \ gh pr create \
--title "fix: short description" \ --title "fix: short description" \
@@ -157,8 +163,9 @@ When code changes (features or bug fixes) are complete:
``` ```
- Use `--label enhancement` for `feat/` branches, `--label bug` for `fix/` branches, `--label documentation` for `docs/` branches. - Use `--label enhancement` for `feat/` branches, `--label bug` for `fix/` branches, `--label documentation` for `docs/` branches.
- Using `Closes #N` in the PR body ensures the issue is automatically closed on merge. - Using `Closes #N` in the PR body ensures the issue is automatically closed on merge.
- Always add an explicit issue comment with the PR link and short fix summary (do not rely on auto-close event only).
- The `--project` flag links the PR to the Project board. - The `--project` flag links the PR to the Project board.
3. **Present the PR URL to the user and wait for confirmation.** 4. **Present the PR URL to the user and wait for confirmation.**
### Step 4: Wait for CI and Merge ### Step 4: Wait for CI and Merge
@@ -445,6 +452,7 @@ All work is tracked in the [GitHub Project board](https://github.com/users/Danie
Issues with `enhancement`, `bug`, or `triage` labels are **automatically added** to the board. Issues with `enhancement`, `bug`, or `triage` labels are **automatically added** to the board.
2. **When creating a PR**: Always reference the issue with `Closes #N` in the PR body so the issue is automatically **closed** on merge. Note: this does NOT move the Project board status — that must be done manually (see step 3). 2. **When creating a PR**: Always reference the issue with `Closes #N` in the PR body so the issue is automatically **closed** on merge. Note: this does NOT move the Project board status — that must be done manually (see step 3).
Also add a direct issue comment with the PR link and a one-line summary for clear issue-thread traceability.
3. **After merge — verify automation**: The `project-auto-done.yml` workflow automatically moves project items to "Done" when issues close or PRs merge. After merge, verify it ran: 3. **After merge — verify automation**: The `project-auto-done.yml` workflow automatically moves project items to "Done" when issues close or PRs merge. After merge, verify it ran:
```bash ```bash
+50 -9
View File
@@ -14,10 +14,17 @@ You are the testing manager for **MedAssist-ng**. Your job is to ensure every fe
- **Tests are mandatory**: Every new feature and every bug fix MUST have corresponding tests. - **Tests are mandatory**: Every new feature and every bug fix MUST have corresponding tests.
- **Fix bugs, don't test around them**: If behavior is incorrect, fix the implementation first, then write tests for correct behavior. - **Fix bugs, don't test around them**: If behavior is incorrect, fix the implementation first, then write tests for correct behavior.
- **Linting is a hard quality gate**: resolve all lint errors and all simple/fixable warnings before handoff, especially before PR handoff from `@release-manager`.
- **Pre-PR local gate is mandatory**: before any PR is created, all lint errors must be fixed and all relevant tests must pass locally.
- **No CI-first failures**: tests must fail locally when broken and be fixed locally before PR handoff; do not rely on GitHub CI to discover obvious regressions.
- **Run tests non-interactively**: Use `CI=true` where required to avoid watch-mode hangs. - **Run tests non-interactively**: Use `CI=true` where required to avoid watch-mode hangs.
- **Playwright must disable auto-open reports**: Always prefix Playwright runs with `PLAYWRIGHT_HTML_OPEN=never`.
- **Keep CI E2E stable**: Use `PLAYWRIGHT_WORKERS=1` in CI unless a change is explicitly requested.
- **Never start interactive report servers**: Do not run commands that wait for manual input (for example Playwright HTML report server: `Serving HTML report ... Press Ctrl+C to quit`). Always use finite, non-interactive commands and reporters. - **Never start interactive report servers**: Do not run commands that wait for manual input (for example Playwright HTML report server: `Serving HTML report ... Press Ctrl+C to quit`). Always use finite, non-interactive commands and reporters.
- **No remote git operations**: Do not push, merge, create PRs, tags, or releases. Hand over to `@release-manager` when ready. - **No remote git operations**: Do not push, merge, create PRs, tags, or releases. Hand over to `@release-manager` when ready.
- **Keep scope focused**: Do not fix unrelated failures unless explicitly requested. - **Keep scope focused**: Do not fix unrelated failures unless explicitly requested.
- **Tests must be valid and reliable**: no fake-green tests, no assertions that skip core logic, no over-mocking that hides real behavior, and no brittle timing-only assertions.
- **Regression prevention is mandatory**: every fixed bug must get a deterministic regression test that fails before the fix and passes after it.
## CI/CD Ownership Boundary ## CI/CD Ownership Boundary
@@ -27,9 +34,9 @@ You are the testing manager for **MedAssist-ng**. Your job is to ensure every fe
## Test Stack & Locations ## Test Stack & Locations
- **Backend**: Vitest 2.1 + v8 coverage - **Backend unit/integration**: Vitest 4 + v8 coverage (`backend/src/test/*.test.ts`)
- **Frontend unit/integration**: Vitest - **Frontend unit/integration**: Vitest 4 + Testing Library (`frontend/src/test/**`)
- **E2E**: Playwright - **Frontend E2E**: Playwright (`frontend/e2e/**`) using stable config for CI-like runs
Primary locations: Primary locations:
@@ -43,22 +50,41 @@ Primary locations:
2. Add/update tests near the affected feature. 2. Add/update tests near the affected feature.
3. Run the smallest relevant subset first. 3. Run the smallest relevant subset first.
4. Expand to broader suites if subset passes. 4. Expand to broader suites if subset passes.
5. Report what was run, what passed, and any remaining known failures. 5. Run lint + required local test/build gates before PR handoff.
6. Report what was run, what passed, and any remaining known failures.
## Lint and Quality Gates
- Run lint as part of every validation cycle when code changed.
- Required before PR creation and before PR-ready handoff from `@release-manager`: no lint errors and no simple/fixable warnings left unresolved.
- If lint fails, fix root causes first, then re-run affected tests.
- Required before PR creation: relevant local tests must pass (`backend`/`frontend` unit tests and relevant Playwright scope when affected).
- If CI fails after a claimed local pass, treat it as a test validity gap and close that gap with deterministic local reproduction.
Recommended commands:
```bash
npm run lint
cd backend && npm run check
cd frontend && npm run check
```
## Commands ## Commands
### Backend ### Backend
```bash ```bash
cd backend && CI=true npm test cd backend && CI=true npm run test:run
cd backend && CI=true npm run test:coverage cd backend && CI=true npm run test:coverage
cd backend && CI=true npm test -- -t "test name" cd backend && CI=true npm run test:run -- -t "test name"
``` ```
### Frontend ### Frontend
```bash ```bash
cd frontend && CI=true npm test cd frontend && CI=true npm run test:run
cd frontend && CI=true npm run test:coverage
cd frontend && CI=true npm run test:run -- -t "test name"
cd frontend && npm run lint cd frontend && npm run lint
cd frontend && npm run build cd frontend && npm run build
``` ```
@@ -66,8 +92,10 @@ cd frontend && npm run build
### Playwright E2E ### Playwright E2E
```bash ```bash
cd frontend && npm run test:e2e cd frontend && PLAYWRIGHT_HTML_OPEN=never npm run test:e2e
cd frontend && npm run test:e2e -- --project=chromium cd frontend && PLAYWRIGHT_HTML_OPEN=never PLAYWRIGHT_WORKERS=1 npm run test:e2e -- --workers=1
cd frontend && PLAYWRIGHT_HTML_OPEN=never PLAYWRIGHT_WORKERS=4 npm run test:e2e:local
cd frontend && PLAYWRIGHT_HTML_OPEN=never npm run test:e2e -- --project=chromium
# Never use interactive UI/headed/report-server commands in agent runs. # Never use interactive UI/headed/report-server commands in agent runs.
# Do not use: npm run test:e2e:ui, npm run test:e2e:headed, npx playwright show-report # Do not use: npm run test:e2e:ui, npm run test:e2e:headed, npx playwright show-report
``` ```
@@ -78,6 +106,7 @@ cd frontend && npm run test:e2e -- --project=chromium
- Validate both status codes and response payloads. - Validate both status codes and response payloads.
- Add regression tests for every fixed bug. - Add regression tests for every fixed bug.
- Keep tests deterministic and isolated. - Keep tests deterministic and isolated.
- Validate observable behavior, not implementation details.
## E2E Test Patterns ## E2E Test Patterns
@@ -85,6 +114,15 @@ cd frontend && npm run test:e2e -- --project=chromium
- Avoid flaky timing assumptions; prefer waiting for concrete UI states. - Avoid flaky timing assumptions; prefer waiting for concrete UI states.
- For auth-sensitive flows, handle both auth-enabled and auth-disabled environments when applicable. - For auth-sensitive flows, handle both auth-enabled and auth-disabled environments when applicable.
- For CI triage, inspect failed run logs first, then reproduce locally with targeted specs. - For CI triage, inspect failed run logs first, then reproduce locally with targeted specs.
- Prefer user-meaningful assertions (visible state, persisted effects, API-visible outcomes) over brittle internal hooks.
## Test Validity Checklist
- The test fails when the real target logic is intentionally broken.
- The assertion verifies functional behavior, not just mocked calls.
- Mocks/stubs are minimal and do not replace the unit under test.
- The test is deterministic across repeated local and CI runs.
- The test protects against the specific regression that was fixed.
## CI Failure Triage ## CI Failure Triage
@@ -115,6 +153,9 @@ When test checks fail:
Testing work is complete when: Testing work is complete when:
- Required tests exist and validate intended behavior. - Required tests exist and validate intended behavior.
- Tests are proven valid (not fake-green) and reliable.
- Lint is clean: no errors and no simple/fixable warnings left.
- Pre-PR local gate passed: lint and all relevant tests pass locally before handoff for PR creation.
- Relevant local test commands pass. - Relevant local test commands pass.
- CI test failures are resolved or clearly documented with rationale. - CI test failures are resolved or clearly documented with rationale.
- No temporary debugging files remain in the workspace. - No temporary debugging files remain in the workspace.
+12 -70
View File
@@ -1,77 +1,19 @@
# MedAssist-ng - AI Coding Instructions # MedAssist-ng - Copilot Entry Point
## Purpose ## VERY IMPORTANT
Use `AGENTS.md` as the canonical governance source. Read the referenced skill files before starting any task. - Always keep agent work memory updated in `doku/memory_notes.md` so progress and decisions remain recoverable across context loss.
- Always keep a user-facing work report updated in `doku/report.md` so completed work is easy to review.
- This memory/report rule replaces the previous `doku/APP_BEHAVIOR.md` persistence requirement.
## Project Orientation (Read First) Use `AGENTS.md` as the single source of truth for all governance, workflow, and skill rules.
- **Product**: MedAssist-ng is a medication planner with stock tracking, reminders (email/push), refill history, and schedule sharing. ## Required Startup Steps
- **Tech stack**: React + TypeScript + Vite (`frontend/`), Fastify + TypeScript + Drizzle + SQLite (`backend/`).
- **Request path**: Frontend uses `/api/*` only; backend route handlers live in `backend/src/routes/`.
- **Primary backend modules**:
- Auth/SSO: `backend/src/routes/auth.ts`, `backend/src/routes/oidc.ts`, `backend/src/plugins/auth.ts`
- Medications/data: `backend/src/routes/medications.ts`, `backend/src/db/schema.ts`
- Reminders: `backend/src/services/reminder-scheduler.ts`, `backend/src/routes/planner.ts`, `backend/src/routes/settings.ts`
- **Primary frontend modules**:
- Pages: `frontend/src/pages/`
- Shared app state: `frontend/src/context/AppContext.tsx`
- Domain hooks: `frontend/src/hooks/`
- Translations: `frontend/src/i18n/en.json`, `frontend/src/i18n/de.json`
Use this orientation for quick navigation before applying the rules below. 1. Read `AGENTS.md` first.
2. Identify triggered skills from `AGENTS.md` and read each referenced `SKILL.md` before making changes.
3. Follow delegation boundaries exactly (`@testing-manager` for testing, `@release-manager` for release orchestration).
## Always-On Rules ## Scope
- English only for project artifacts. This file intentionally stays minimal to prevent duplicated or conflicting instructions.
- **NEVER run remote git commands** — no `git push`, no `gh pr create/merge`, no `gh release`, no `git tag`. Prepare locally, then hand off to `@release-manager`.
- Testing work belongs to `@testing-manager`.
- PR/release/CI orchestration belongs to `@release-manager`.
- Keep changes local, focused, and consistent with existing UI/API patterns.
- **Hard PR scope + size rule**: one cohesive objective per PR; if scope drifts or diff becomes large (target <= 500 changed lines, hard split at ~800+), split into logical follow-up PRs instead of bundling.
- Remove obsolete code when re-implementing — never leave dead code behind.
- **Document behavioral discoveries**: When you discover or clarify how a feature works (e.g., what triggers notifications, how thresholds interact, which code paths exist), **always** add or update the relevant section in `doku/APP_BEHAVIOR.md`. This is mandatory — do not rely on conversation context alone.
## MedAssist Essentials
- Frontend calls backend through `/api/*`.
- DB changes must stay backward-compatible (schema default + alter migration + null-safe reads).
---
## Skills (MANDATORY — read before every task)
Before starting any task, identify which skills apply and **read their full SKILL.md file** for detailed rules.
| Skill | Trigger | File |
|---|---|---|
| **Architecture Guard** | API endpoints, frontend API calls, routing, code placement | `.github/skills/medassist-architecture-guard/SKILL.md` |
| **DB Compatibility** | Persisted data, schema changes, migrations | `.github/skills/medassist-db-compat-check/SKILL.md` |
| **i18n Enforcer** ⚠️ | Any user-facing text in frontend or backend | `.github/skills/medassist-i18n-enforcer/SKILL.md` |
| **UI Consistency** | UI flows, modals, buttons, forms, settings | `.github/skills/medassist-ui-consistency/SKILL.md` |
| **Frontend Polish** | Visual quality improvements | `.github/skills/medassist-frontend-polish/SKILL.md` |
| **Security Sanity** | Backend routes, auth, file handling, external input | `.github/skills/medassist-security-sanity/SKILL.md` |
| **Observability Guard** | Services, schedulers, startup, failure handling | `.github/skills/medassist-observability-guard/SKILL.md` |
| **Config Change Guard** | `.env`, Docker, Vite proxy, runtime defaults | `.github/skills/medassist-config-change-guard/SKILL.md` |
| **Doc Sync Guard** | Behavior changes, setup, env vars, workflows | `.github/skills/medassist-doc-sync-guard/SKILL.md` |
| **Testing Handoff** | Writing/running tests, CI test failures | `.github/skills/medassist-testing-handoff/SKILL.md` |
| **Release Handoff** | Branch push, PR, merge, tagging, release | `.github/skills/medassist-release-handoff/SKILL.md` |
| **Skill Quality Review** | Creating/modifying skills | `.github/skills/medassist-skill-quality-review/SKILL.md` |
### Non-negotiable parity rules (always apply)
1. **Desktop + Mobile Parity**: Medication edit has two paths — `MedicationsPage.tsx` (desktop) and `MobileEditModal` (mobile). **Always update BOTH**.
2. **Notification Dual Code Paths**: Notifications have two code paths — `backend/src/services/reminder-scheduler.ts` (scheduler) and `backend/src/routes/planner.ts` (manual). **Always update BOTH**.
---
## Delegation
- **Testing handoff → `@testing-manager`**: test planning, writing, execution, CI test triage.
- **Release handoff → `@release-manager`**: PR/release orchestration, merge flow, workflow monitoring.
## Key References
- Canonical governance: `AGENTS.md`
- Skill files: `.github/skills/*/SKILL.md`
- Specialist agents: `.github/agents/testing-manager.agent.md`, `.github/agents/release-manager.agent.md`
+17
View File
@@ -7,9 +7,11 @@ updates:
schedule: schedule:
interval: "weekly" interval: "weekly"
day: "monday" day: "monday"
time: "06:20"
open-pull-requests-limit: 10 open-pull-requests-limit: 10
labels: labels:
- "dependencies" - "dependencies"
- "backend"
groups: groups:
minor-and-patch: minor-and-patch:
update-types: update-types:
@@ -22,9 +24,11 @@ updates:
schedule: schedule:
interval: "weekly" interval: "weekly"
day: "monday" day: "monday"
time: "06:10"
open-pull-requests-limit: 10 open-pull-requests-limit: 10
labels: labels:
- "dependencies" - "dependencies"
- "frontend"
groups: groups:
minor-and-patch: minor-and-patch:
update-types: update-types:
@@ -37,9 +41,16 @@ updates:
schedule: schedule:
interval: "weekly" interval: "weekly"
day: "monday" day: "monday"
time: "06:00"
open-pull-requests-limit: 5 open-pull-requests-limit: 5
labels: labels:
- "dependencies" - "dependencies"
- "root"
groups:
minor-and-patch:
update-types:
- "minor"
- "patch"
# GitHub Actions # GitHub Actions
- package-ecosystem: "github-actions" - package-ecosystem: "github-actions"
@@ -47,7 +58,13 @@ updates:
schedule: schedule:
interval: "weekly" interval: "weekly"
day: "monday" day: "monday"
time: "06:30"
open-pull-requests-limit: 5 open-pull-requests-limit: 5
labels: labels:
- "dependencies" - "dependencies"
- "ci" - "ci"
groups:
minor-and-patch:
update-types:
- "minor"
- "patch"
+1 -1
View File
@@ -13,7 +13,7 @@ Use one governance source to avoid duplicated or conflicting policy text.
## Skills ## Skills
- `medassist-karpathy-core` — enforce assumption clarity, simplicity, surgical diffs, and verifiable execution. - `medassist-karpathy-core` — enforce think-before-coding, simplicity-first changes, surgical diffs, and goal-driven verification.
- `medassist-architecture-guard` — enforce frontend/backend boundary and `/api/*` data-flow conventions. - `medassist-architecture-guard` — enforce frontend/backend boundary and `/api/*` data-flow conventions.
- `medassist-db-compat-check` — enforce backward-compatible SQLite/Drizzle schema changes. - `medassist-db-compat-check` — enforce backward-compatible SQLite/Drizzle schema changes.
- `medassist-i18n-enforcer` — enforce translation-key-only UI copy with EN/DE parity. - `medassist-i18n-enforcer` — enforce translation-key-only UI copy with EN/DE parity.
@@ -0,0 +1,69 @@
---
name: medassist-karpathy-core
description: Apply assumption clarity, simplicity-first implementation, surgical diffs, and goal-driven verification for non-trivial coding tasks.
---
# Skill Instructions
Use this skill as an execution style layer for implementation tasks where overengineering, broad refactors, or unclear assumptions are likely.
## Use When
- The request is ambiguous and assumptions must be made explicit.
- The change can easily balloon in scope.
- A bug fix or feature needs explicit success criteria and verification.
- You need to keep diffs minimal and directly tied to the request.
## Do Not Use When
- The task is trivial and can be completed safely without extra process overhead.
- The task is only about ownership routing (use `medassist-testing-handoff` / `medassist-release-handoff`).
- The task is only about domain guardrails already covered by specialized skills (architecture, DB, i18n, UI, security, config, observability).
## Core Principles
### 1. Think Before Coding
- Do not assume silently.
- State assumptions explicitly.
- If multiple interpretations exist, present them instead of picking one invisibly.
- If uncertain or blocked by ambiguity, stop and ask.
- If a simpler approach exists, call it out.
### 2. Simplicity First
- Implement the minimum code required to solve the asked problem.
- Do not add speculative features, abstractions, or configurability.
- Avoid defensive handling for impossible scenarios.
- If the solution feels overcomplicated, simplify before finalizing.
### 3. Surgical Changes
- Touch only lines required for the request.
- Do not refactor unrelated areas.
- Match existing local style and patterns.
- Remove only unused code introduced by your own change.
- If unrelated dead code is discovered, mention it but do not remove it unless requested.
### 4. Goal-Driven Execution
- Translate requests into verifiable outcomes before implementation.
- For multi-step tasks, define short steps with checks.
- Verify the requested behavior explicitly before declaring done.
Example execution frame:
```text
1. [Step] -> verify: [check]
2. [Step] -> verify: [check]
3. [Step] -> verify: [check]
```
## Response Format
When this skill is used, report briefly:
- Assumptions made (or clarifications requested)
- Why the chosen approach is the simplest viable one
- What was changed (and what was intentionally not changed)
- Verification performed and result
@@ -0,0 +1,37 @@
name: Dependabot Automerge
on:
pull_request_target:
types:
- opened
- reopened
- synchronize
- ready_for_review
permissions:
contents: write
pull-requests: write
jobs:
enable-automerge:
if: github.actor == 'dependabot[bot]'
runs-on: ubuntu-latest
steps:
- name: Read Dependabot metadata
id: metadata
uses: dependabot/fetch-metadata@v2
with:
github-token: ${{ secrets.GITHUB_TOKEN }}
- name: Enable auto-merge for safe updates
if: >-
(steps.metadata.outputs.package-ecosystem == 'npm' ||
steps.metadata.outputs.package-ecosystem == 'github_actions') &&
(steps.metadata.outputs.update-type == 'version-update:semver-minor' ||
steps.metadata.outputs.update-type == 'version-update:semver-patch')
uses: peter-evans/enable-pull-request-automerge@v3
with:
token: ${{ secrets.GITHUB_TOKEN }}
pull-request-number: ${{ github.event.pull_request.number }}
merge-method: squash
+6
View File
@@ -4,6 +4,12 @@ on:
push: push:
branches: [main] branches: [main]
tags: ['v*'] tags: ['v*']
paths:
- 'backend/**'
- 'frontend/**'
- 'docker-compose.yml'
- 'docker-compose.dev.yml'
- '.github/workflows/docker-build.yml'
workflow_dispatch: workflow_dispatch:
inputs: inputs:
tag: tag:
+4 -2
View File
@@ -50,11 +50,13 @@ jobs:
run: npx playwright test --project=chromium run: npx playwright test --project=chromium
env: env:
CI: true CI: true
PLAYWRIGHT_WORKERS: 1
PLAYWRIGHT_HTML_OPEN: never
JWT_SECRET: e2e-test-secret-that-is-long-enough JWT_SECRET: e2e-test-secret-that-is-long-enough
SESSION_SECRET: e2e-test-session-secret-long-enough SESSION_SECRET: e2e-test-session-secret-long-enough
- name: Upload Playwright report - name: Upload Playwright report
uses: actions/upload-artifact@v6 uses: actions/upload-artifact@v7
if: always() if: always()
with: with:
name: playwright-report name: playwright-report
@@ -62,7 +64,7 @@ jobs:
retention-days: 7 retention-days: 7
- name: Upload test results - name: Upload test results
uses: actions/upload-artifact@v6 uses: actions/upload-artifact@v7
if: always() if: always()
with: with:
name: playwright-results name: playwright-results
+2 -2
View File
@@ -73,7 +73,7 @@ jobs:
run: npm run test:coverage run: npm run test:coverage
- name: Upload coverage report - name: Upload coverage report
uses: actions/upload-artifact@v6 uses: actions/upload-artifact@v7
if: always() if: always()
with: with:
name: backend-coverage name: backend-coverage
@@ -118,7 +118,7 @@ jobs:
run: npm run build run: npm run build
- name: Upload coverage report - name: Upload coverage report
uses: actions/upload-artifact@v6 uses: actions/upload-artifact@v7
if: always() if: always()
with: with:
name: frontend-coverage name: frontend-coverage
+5 -1
View File
@@ -82,4 +82,8 @@ Thumbs.db
.claude/ .claude/
AGENTS.md AGENTS.md
docs/TECH_STACK.md docs/TECH_STACK.md
doku doku/
doku/memory_notes.md
doku/report.md
plan/
.copilot-tracking
+31 -9
View File
@@ -10,7 +10,7 @@
</p> </p>
<p align="center"> <p align="center">
<img src="https://img.shields.io/badge/React-18-61DAFB?logo=react" alt="React 18" /> <img src="https://img.shields.io/badge/React-19-61DAFB?logo=react" alt="React 19" />
<img src="https://img.shields.io/badge/TypeScript-5-3178C6?logo=typescript" alt="TypeScript" /> <img src="https://img.shields.io/badge/TypeScript-5-3178C6?logo=typescript" alt="TypeScript" />
<img src="https://img.shields.io/badge/Fastify-5-000000?logo=fastify" alt="Fastify" /> <img src="https://img.shields.io/badge/Fastify-5-000000?logo=fastify" alt="Fastify" />
<img src="https://img.shields.io/badge/SQLite-Database-003B57?logo=sqlite" alt="SQLite" /> <img src="https://img.shields.io/badge/SQLite-Database-003B57?logo=sqlite" alt="SQLite" />
@@ -18,13 +18,13 @@
</p> </p>
<p align="center"> <p align="center">
<img src="https://img.shields.io/badge/Backend_Tests-569%2F569-brightgreen?logo=vitest" alt="Backend Tests 454/454" /> <img src="https://img.shields.io/badge/Backend_Tests-577%2F577-brightgreen?logo=vitest" alt="Backend Tests 454/454" />
<img src="https://img.shields.io/badge/Frontend_Tests-771%2F771-brightgreen?logo=vitest" alt="Frontend Tests 611/611" /> <img src="https://img.shields.io/badge/Frontend_Tests-777%2F777-brightgreen?logo=vitest" alt="Frontend Tests 611/611" />
</p> </p>
### 🤖 AI-Generated Code ### 🤖 AI-Generated Code
> This app was 100% coded with Claude Opus 4.5. Use at your own risk. > This app was 100% coded with [Claude Opus 4.6](https://www.anthropic.com/claude) and [GPT-5.3 Codex](https://openai.com/index/gpt-5/). Use at your own risk.
### ⚠️ Disclaimer ### ⚠️ Disclaimer
@@ -120,10 +120,10 @@ Share your medication schedule with others via a public link.
</details> </details>
### Smart Inventory ### Smart Inventory
- Track exact stock: packs, blisters, bottles, and loose pills - Track exact stock with package profiles (blister, bottle, tube, liquid container)
- Display remaining days of supply - Display remaining days of supply
- Automatic calculation based on intake schedule - Automatic calculation based on intake schedule
- Manual stock correction supports partial blisters and loose pills - Manual stock correction supports profile-specific stock semantics (sealed units + loose stock for blister, amount-based stock for bottle/tube/liquid)
### Medication Refill ### Medication Refill
- One-click refill with pack or loose pill options - One-click refill with pack or loose pill options
@@ -141,7 +141,7 @@ Share your medication schedule with others via a public link.
- Intake reminders via push notifications - Intake reminders via push notifications
### Trip Planner ### Trip Planner
- Calculate how many pills you need for a trip or date range - Calculate medication demand for a trip or date range with package-aware units
- Plan ahead for vacations, business trips, or hospital stays - Plan ahead for vacations, business trips, or hospital stays
- Send demand reports via email or push notification - Send demand reports via email or push notification
@@ -194,7 +194,7 @@ All configuration is done via environment variables in `.env`. Copy `.env.exampl
| `PGID` | `1000` | Group ID for container file permissions | | `PGID` | `1000` | Group ID for container file permissions |
| `PORT` | `3000` | Backend API port | | `PORT` | `3000` | Backend API port |
| `CORS_ORIGINS` | `http://localhost:4174` | Allowed origins for CORS | | `CORS_ORIGINS` | `http://localhost:4174` | Allowed origins for CORS |
| `LOG_LEVEL` | `info` | Log verbosity (`debug`, `info`, `warn`, `error`) | | `LOG_LEVEL` | `info` | Log verbosity (`debug`, `info`, `warn`, `error`, `silent`). At `info` (default), high-frequency polling endpoints are suppressed. Set `debug` to see all requests. |
| `TZ` | `Europe/Berlin` | Timezone for scheduled reminders | | `TZ` | `Europe/Berlin` | Timezone for scheduled reminders |
### Authentication ### Authentication
@@ -250,7 +250,9 @@ Generate secrets with: `openssl rand -hex 32`
MedAssist uses [Shoutrrr](https://containrrr.dev/shoutrrr/) for push notifications, supporting many services with a single URL format. MedAssist uses [Shoutrrr](https://containrrr.dev/shoutrrr/) for push notifications, supporting many services with a single URL format.
**Supported services:** ntfy, Pushover, Gotify, Discord, Telegram, Slack, Matrix, and [many more](https://containrrr.dev/shoutrrr/v0.8/services/overview/). **Implemented URL schemes in MedAssist:** `ntfy://`, `discord://`, `pushover://`, `gotify://`, `telegram://`, plus direct `https://` webhooks.
This covers common providers like ntfy, Discord, Pushover, Gotify, Telegram, Slack webhooks, and many others via webhook URLs.
Configure push notifications in Settings → Push, or set defaults via environment variables: Configure push notifications in Settings → Push, or set defaults via environment variables:
@@ -288,6 +290,7 @@ Get your keys at [pushover.net](https://pushover.net/):
**Gotify** (self-hosted): **Gotify** (self-hosted):
``` ```
gotify://your-server.com/TOKEN gotify://your-server.com/TOKEN
gotify://your-server.com:443/path/to/gotify/TOKEN?priority=1
``` ```
**Discord**: **Discord**:
@@ -298,6 +301,7 @@ discord://TOKEN@WEBHOOK_ID
**Telegram**: **Telegram**:
``` ```
telegram://TOKEN@telegram?chats=CHAT_ID telegram://TOKEN@telegram?chats=CHAT_ID
telegram://TOKEN@telegram?chats=@your_channel,-1001234567890
``` ```
For all services and options, see the [Shoutrrr documentation](https://containrrr.dev/shoutrrr/v0.8/services/overview/). For all services and options, see the [Shoutrrr documentation](https://containrrr.dev/shoutrrr/v0.8/services/overview/).
@@ -311,6 +315,24 @@ docker compose -f docker-compose.dev.yml up
- Frontend: `http://localhost:5173` (hot reload) - Frontend: `http://localhost:5173` (hot reload)
- Backend: `http://localhost:3000` - Backend: `http://localhost:3000`
Playwright E2E recommendations:
```bash
cd frontend
npm run test:e2e:local # local run with PLAYWRIGHT_WORKERS=4
npm run test:e2e:all:local # local all-browser run with PLAYWRIGHT_WORKERS=4
```
- CI stays at `PLAYWRIGHT_WORKERS=1` for stability.
- Data-heavy specs remain sequential via the `chromium-data` project config.
# Dependency Updates
- Dependabot checks dependencies weekly for `frontend`, `backend`, repository root tooling, and GitHub Actions.
- Minor and patch updates are grouped to reduce PR noise.
- Dependabot minor/patch PRs are configured for auto-merge after required CI checks pass.
- Major updates still require manual review before merge.
# Acknowledgements # Acknowledgements
This project was inspired by [MedAssist](https://github.com/njic/medassist) by njic. This project was inspired by [MedAssist](https://github.com/njic/medassist) by njic.
@@ -0,0 +1,5 @@
ALTER TABLE `medications` ADD `medication_form` text(20) DEFAULT 'tablet' NOT NULL;--> statement-breakpoint
ALTER TABLE `medications` ADD `pill_form` text(20);--> statement-breakpoint
ALTER TABLE `medications` ADD `lifecycle_category` text(30) DEFAULT 'refill_when_empty' NOT NULL;--> statement-breakpoint
ALTER TABLE `medications` ADD `medication_end_date` text;--> statement-breakpoint
ALTER TABLE `medications` ADD `auto_mark_obsolete_after_end_date` integer DEFAULT true NOT NULL;
File diff suppressed because it is too large Load Diff
+7
View File
@@ -78,6 +78,13 @@
"when": 1771694832866, "when": 1771694832866,
"tag": "0010_mean_spot", "tag": "0010_mean_spot",
"breakpoints": true "breakpoints": true
},
{
"idx": 11,
"version": "6",
"when": 1772219947541,
"tag": "0011_stiff_randall_flagg",
"breakpoints": true
} }
] ]
} }
+245 -124
View File
@@ -1,12 +1,12 @@
{ {
"name": "medassist-ng-backend", "name": "medassist-ng-backend",
"version": "1.15.1", "version": "1.18.2",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "medassist-ng-backend", "name": "medassist-ng-backend",
"version": "1.15.1", "version": "1.18.2",
"dependencies": { "dependencies": {
"@fastify/cookie": "^11.0.2", "@fastify/cookie": "^11.0.2",
"@fastify/cors": "^11.2.0", "@fastify/cors": "^11.2.0",
@@ -20,7 +20,7 @@
"argon2": "^0.44.0", "argon2": "^0.44.0",
"dotenv": "^17.3.1", "dotenv": "^17.3.1",
"drizzle-orm": "^0.45.1", "drizzle-orm": "^0.45.1",
"fastify": "^5.7.4", "fastify": "^5.8.1",
"nodemailer": "^8.0.1", "nodemailer": "^8.0.1",
"openid-client": "^6.8.2", "openid-client": "^6.8.2",
"sharp": "^0.34.5", "sharp": "^0.34.5",
@@ -28,11 +28,12 @@
}, },
"devDependencies": { "devDependencies": {
"@biomejs/biome": "^2.4.4", "@biomejs/biome": "^2.4.4",
"@types/node": "^25.3.0", "@types/node": "^25.3.3",
"@types/nodemailer": "^7.0.11", "@types/nodemailer": "^7.0.11",
"@types/supertest": "^6.0.2", "@types/supertest": "^7.2.0",
"@vitest/coverage-v8": "^4.0.18", "@vitest/coverage-v8": "^4.0.18",
"drizzle-kit": "^0.31.9", "drizzle-kit": "^0.31.9",
"pino-pretty": "^13.1.3",
"supertest": "^7.2.2", "supertest": "^7.2.2",
"tsx": "^4.19.0", "tsx": "^4.19.0",
"typescript": "^5.5.4", "typescript": "^5.5.4",
@@ -2228,9 +2229,9 @@
"license": "MIT" "license": "MIT"
}, },
"node_modules/@rollup/rollup-android-arm-eabi": { "node_modules/@rollup/rollup-android-arm-eabi": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.59.0.tgz",
"integrity": "sha512-A6ehUVSiSaaliTxai040ZpZ2zTevHYbvu/lDoeAteHI8QnaosIzm4qwtezfRg1jOYaUmnzLX1AOD6Z+UJjtifg==", "integrity": "sha512-upnNBkA6ZH2VKGcBj9Fyl9IGNPULcjXRlg0LLeaioQWueH30p6IXtJEbKAgvyv+mJaMxSm1l6xwDXYjpEMiLMg==",
"cpu": [ "cpu": [
"arm" "arm"
], ],
@@ -2242,9 +2243,9 @@
] ]
}, },
"node_modules/@rollup/rollup-android-arm64": { "node_modules/@rollup/rollup-android-arm64": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.59.0.tgz",
"integrity": "sha512-dQaAddCY9YgkFHZcFNS/606Exo8vcLHwArFZ7vxXq4rigo2bb494/xKMMwRRQW6ug7Js6yXmBZhSBRuBvCCQ3w==", "integrity": "sha512-hZ+Zxj3SySm4A/DylsDKZAeVg0mvi++0PYVceVyX7hemkw7OreKdCvW2oQ3T1FMZvCaQXqOTHb8qmBShoqk69Q==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -2256,9 +2257,9 @@
] ]
}, },
"node_modules/@rollup/rollup-darwin-arm64": { "node_modules/@rollup/rollup-darwin-arm64": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.59.0.tgz",
"integrity": "sha512-crNPrwJOrRxagUYeMn/DZwqN88SDmwaJ8Cvi/TN1HnWBU7GwknckyosC2gd0IqYRsHDEnXf328o9/HC6OkPgOg==", "integrity": "sha512-W2Psnbh1J8ZJw0xKAd8zdNgF9HRLkdWwwdWqubSVk0pUuQkoHnv7rx4GiF9rT4t5DIZGAsConRE3AxCdJ4m8rg==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -2270,9 +2271,9 @@
] ]
}, },
"node_modules/@rollup/rollup-darwin-x64": { "node_modules/@rollup/rollup-darwin-x64": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.59.0.tgz",
"integrity": "sha512-Ji8g8ChVbKrhFtig5QBV7iMaJrGtpHelkB3lsaKzadFBe58gmjfGXAOfI5FV0lYMH8wiqsxKQ1C9B0YTRXVy4w==", "integrity": "sha512-ZW2KkwlS4lwTv7ZVsYDiARfFCnSGhzYPdiOU4IM2fDbL+QGlyAbjgSFuqNRbSthybLbIJ915UtZBtmuLrQAT/w==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -2284,9 +2285,9 @@
] ]
}, },
"node_modules/@rollup/rollup-freebsd-arm64": { "node_modules/@rollup/rollup-freebsd-arm64": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.59.0.tgz",
"integrity": "sha512-R+/WwhsjmwodAcz65guCGFRkMb4gKWTcIeLy60JJQbXrJ97BOXHxnkPFrP+YwFlaS0m+uWJTstrUA9o+UchFug==", "integrity": "sha512-EsKaJ5ytAu9jI3lonzn3BgG8iRBjV4LxZexygcQbpiU0wU0ATxhNVEpXKfUa0pS05gTcSDMKpn3Sx+QB9RlTTA==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -2298,9 +2299,9 @@
] ]
}, },
"node_modules/@rollup/rollup-freebsd-x64": { "node_modules/@rollup/rollup-freebsd-x64": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.59.0.tgz",
"integrity": "sha512-IEQTCHeiTOnAUC3IDQdzRAGj3jOAYNr9kBguI7MQAAZK3caezRrg0GxAb6Hchg4lxdZEI5Oq3iov/w/hnFWY9Q==", "integrity": "sha512-d3DuZi2KzTMjImrxoHIAODUZYoUUMsuUiY4SRRcJy6NJoZ6iIqWnJu9IScV9jXysyGMVuW+KNzZvBLOcpdl3Vg==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -2312,9 +2313,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-arm-gnueabihf": { "node_modules/@rollup/rollup-linux-arm-gnueabihf": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.59.0.tgz",
"integrity": "sha512-F8sWbhZ7tyuEfsmOxwc2giKDQzN3+kuBLPwwZGyVkLlKGdV1nvnNwYD0fKQ8+XS6hp9nY7B+ZeK01EBUE7aHaw==", "integrity": "sha512-t4ONHboXi/3E0rT6OZl1pKbl2Vgxf9vJfWgmUoCEVQVxhW6Cw/c8I6hbbu7DAvgp82RKiH7TpLwxnJeKv2pbsw==",
"cpu": [ "cpu": [
"arm" "arm"
], ],
@@ -2326,9 +2327,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-arm-musleabihf": { "node_modules/@rollup/rollup-linux-arm-musleabihf": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.59.0.tgz",
"integrity": "sha512-rGfNUfn0GIeXtBP1wL5MnzSj98+PZe/AXaGBCRmT0ts80lU5CATYGxXukeTX39XBKsxzFpEeK+Mrp9faXOlmrw==", "integrity": "sha512-CikFT7aYPA2ufMD086cVORBYGHffBo4K8MQ4uPS/ZnY54GKj36i196u8U+aDVT2LX4eSMbyHtyOh7D7Zvk2VvA==",
"cpu": [ "cpu": [
"arm" "arm"
], ],
@@ -2340,9 +2341,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-arm64-gnu": { "node_modules/@rollup/rollup-linux-arm64-gnu": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.59.0.tgz",
"integrity": "sha512-MMtej3YHWeg/0klK2Qodf3yrNzz6CGjo2UntLvk2RSPlhzgLvYEB3frRvbEF2wRKh1Z2fDIg9KRPe1fawv7C+g==", "integrity": "sha512-jYgUGk5aLd1nUb1CtQ8E+t5JhLc9x5WdBKew9ZgAXg7DBk0ZHErLHdXM24rfX+bKrFe+Xp5YuJo54I5HFjGDAA==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -2354,9 +2355,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-arm64-musl": { "node_modules/@rollup/rollup-linux-arm64-musl": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.59.0.tgz",
"integrity": "sha512-1a/qhaaOXhqXGpMFMET9VqwZakkljWHLmZOX48R0I/YLbhdxr1m4gtG1Hq7++VhVUmf+L3sTAf9op4JlhQ5u1Q==", "integrity": "sha512-peZRVEdnFWZ5Bh2KeumKG9ty7aCXzzEsHShOZEFiCQlDEepP1dpUl/SrUNXNg13UmZl+gzVDPsiCwnV1uI0RUA==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -2368,9 +2369,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-loong64-gnu": { "node_modules/@rollup/rollup-linux-loong64-gnu": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.59.0.tgz",
"integrity": "sha512-QWO6RQTZ/cqYtJMtxhkRkidoNGXc7ERPbZN7dVW5SdURuLeVU7lwKMpo18XdcmpWYd0qsP1bwKPf7DNSUinhvA==", "integrity": "sha512-gbUSW/97f7+r4gHy3Jlup8zDG190AuodsWnNiXErp9mT90iCy9NKKU0Xwx5k8VlRAIV2uU9CsMnEFg/xXaOfXg==",
"cpu": [ "cpu": [
"loong64" "loong64"
], ],
@@ -2382,9 +2383,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-loong64-musl": { "node_modules/@rollup/rollup-linux-loong64-musl": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.59.0.tgz",
"integrity": "sha512-xpObYIf+8gprgWaPP32xiN5RVTi/s5FCR+XMXSKmhfoJjrpRAjCuuqQXyxUa/eJTdAE6eJ+KDKaoEqjZQxh3Gw==", "integrity": "sha512-yTRONe79E+o0FWFijasoTjtzG9EBedFXJMl888NBEDCDV9I2wGbFFfJQQe63OijbFCUZqxpHz1GzpbtSFikJ4Q==",
"cpu": [ "cpu": [
"loong64" "loong64"
], ],
@@ -2396,9 +2397,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-ppc64-gnu": { "node_modules/@rollup/rollup-linux-ppc64-gnu": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.59.0.tgz",
"integrity": "sha512-4BrCgrpZo4hvzMDKRqEaW1zeecScDCR+2nZ86ATLhAoJ5FQ+lbHVD3ttKe74/c7tNT9c6F2viwB3ufwp01Oh2w==", "integrity": "sha512-sw1o3tfyk12k3OEpRddF68a1unZ5VCN7zoTNtSn2KndUE+ea3m3ROOKRCZxEpmT9nsGnogpFP9x6mnLTCaoLkA==",
"cpu": [ "cpu": [
"ppc64" "ppc64"
], ],
@@ -2410,9 +2411,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-ppc64-musl": { "node_modules/@rollup/rollup-linux-ppc64-musl": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.59.0.tgz",
"integrity": "sha512-NOlUuzesGauESAyEYFSe3QTUguL+lvrN1HtwEEsU2rOwdUDeTMJdO5dUYl/2hKf9jWydJrO9OL/XSSf65R5+Xw==", "integrity": "sha512-+2kLtQ4xT3AiIxkzFVFXfsmlZiG5FXYW7ZyIIvGA7Bdeuh9Z0aN4hVyXS/G1E9bTP/vqszNIN/pUKCk/BTHsKA==",
"cpu": [ "cpu": [
"ppc64" "ppc64"
], ],
@@ -2424,9 +2425,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-riscv64-gnu": { "node_modules/@rollup/rollup-linux-riscv64-gnu": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.59.0.tgz",
"integrity": "sha512-ptA88htVp0AwUUqhVghwDIKlvJMD/fmL/wrQj99PRHFRAG6Z5nbWoWG4o81Nt9FT+IuqUQi+L31ZKAFeJ5Is+A==", "integrity": "sha512-NDYMpsXYJJaj+I7UdwIuHHNxXZ/b/N2hR15NyH3m2qAtb/hHPA4g4SuuvrdxetTdndfj9b1WOmy73kcPRoERUg==",
"cpu": [ "cpu": [
"riscv64" "riscv64"
], ],
@@ -2438,9 +2439,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-riscv64-musl": { "node_modules/@rollup/rollup-linux-riscv64-musl": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.59.0.tgz",
"integrity": "sha512-S51t7aMMTNdmAMPpBg7OOsTdn4tySRQvklmL3RpDRyknk87+Sp3xaumlatU+ppQ+5raY7sSTcC2beGgvhENfuw==", "integrity": "sha512-nLckB8WOqHIf1bhymk+oHxvM9D3tyPndZH8i8+35p/1YiVoVswPid2yLzgX7ZJP0KQvnkhM4H6QZ5m0LzbyIAg==",
"cpu": [ "cpu": [
"riscv64" "riscv64"
], ],
@@ -2452,9 +2453,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-s390x-gnu": { "node_modules/@rollup/rollup-linux-s390x-gnu": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.59.0.tgz",
"integrity": "sha512-Bl00OFnVFkL82FHbEqy3k5CUCKH6OEJL54KCyx2oqsmZnFTR8IoNqBF+mjQVcRCT5sB6yOvK8A37LNm/kPJiZg==", "integrity": "sha512-oF87Ie3uAIvORFBpwnCvUzdeYUqi2wY6jRFWJAy1qus/udHFYIkplYRW+wo+GRUP4sKzYdmE1Y3+rY5Gc4ZO+w==",
"cpu": [ "cpu": [
"s390x" "s390x"
], ],
@@ -2466,9 +2467,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-x64-gnu": { "node_modules/@rollup/rollup-linux-x64-gnu": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.59.0.tgz",
"integrity": "sha512-ABca4ceT4N+Tv/GtotnWAeXZUZuM/9AQyCyKYyKnpk4yoA7QIAuBt6Hkgpw8kActYlew2mvckXkvx0FfoInnLg==", "integrity": "sha512-3AHmtQq/ppNuUspKAlvA8HtLybkDflkMuLK4DPo77DfthRb71V84/c4MlWJXixZz4uruIH4uaa07IqoAkG64fg==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -2480,9 +2481,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-x64-musl": { "node_modules/@rollup/rollup-linux-x64-musl": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.59.0.tgz",
"integrity": "sha512-HFps0JeGtuOR2convgRRkHCekD7j+gdAuXM+/i6kGzQtFhlCtQkpwtNzkNj6QhCDp7DRJ7+qC/1Vg2jt5iSOFw==", "integrity": "sha512-2UdiwS/9cTAx7qIUZB/fWtToJwvt0Vbo0zmnYt7ED35KPg13Q0ym1g442THLC7VyI6JfYTP4PiSOWyoMdV2/xg==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -2494,9 +2495,9 @@
] ]
}, },
"node_modules/@rollup/rollup-openbsd-x64": { "node_modules/@rollup/rollup-openbsd-x64": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.59.0.tgz",
"integrity": "sha512-H+hXEv9gdVQuDTgnqD+SQffoWoc0Of59AStSzTEj/feWTBAnSfSD3+Dql1ZruJQxmykT/JVY0dE8Ka7z0DH1hw==", "integrity": "sha512-M3bLRAVk6GOwFlPTIxVBSYKUaqfLrn8l0psKinkCFxl4lQvOSz8ZrKDz2gxcBwHFpci0B6rttydI4IpS4IS/jQ==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -2508,9 +2509,9 @@
] ]
}, },
"node_modules/@rollup/rollup-openharmony-arm64": { "node_modules/@rollup/rollup-openharmony-arm64": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.59.0.tgz",
"integrity": "sha512-4wYoDpNg6o/oPximyc/NG+mYUejZrCU2q+2w6YZqrAs2UcNUChIZXjtafAiiZSUc7On8v5NyNj34Kzj/Ltk6dQ==", "integrity": "sha512-tt9KBJqaqp5i5HUZzoafHZX8b5Q2Fe7UjYERADll83O4fGqJ49O1FsL6LpdzVFQcpwvnyd0i+K/VSwu/o/nWlA==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -2522,9 +2523,9 @@
] ]
}, },
"node_modules/@rollup/rollup-win32-arm64-msvc": { "node_modules/@rollup/rollup-win32-arm64-msvc": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.59.0.tgz",
"integrity": "sha512-O54mtsV/6LW3P8qdTcamQmuC990HDfR71lo44oZMZlXU4tzLrbvTii87Ni9opq60ds0YzuAlEr/GNwuNluZyMQ==", "integrity": "sha512-V5B6mG7OrGTwnxaNUzZTDTjDS7F75PO1ae6MJYdiMu60sq0CqN5CVeVsbhPxalupvTX8gXVSU9gq+Rx1/hvu6A==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -2536,9 +2537,9 @@
] ]
}, },
"node_modules/@rollup/rollup-win32-ia32-msvc": { "node_modules/@rollup/rollup-win32-ia32-msvc": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.59.0.tgz",
"integrity": "sha512-P3dLS+IerxCT/7D2q2FYcRdWRl22dNbrbBEtxdWhXrfIMPP9lQhb5h4Du04mdl5Woq05jVCDPCMF7Ub0NAjIew==", "integrity": "sha512-UKFMHPuM9R0iBegwzKF4y0C4J9u8C6MEJgFuXTBerMk7EJ92GFVFYBfOZaSGLu6COf7FxpQNqhNS4c4icUPqxA==",
"cpu": [ "cpu": [
"ia32" "ia32"
], ],
@@ -2550,9 +2551,9 @@
] ]
}, },
"node_modules/@rollup/rollup-win32-x64-gnu": { "node_modules/@rollup/rollup-win32-x64-gnu": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.59.0.tgz",
"integrity": "sha512-VMBH2eOOaKGtIJYleXsi2B8CPVADrh+TyNxJ4mWPnKfLB/DBUmzW+5m1xUrcwWoMfSLagIRpjUFeW5CO5hyciQ==", "integrity": "sha512-laBkYlSS1n2L8fSo1thDNGrCTQMmxjYY5G0WFWjFFYZkKPjsMBsgJfGf4TLxXrF6RyhI60L8TMOjBMvXiTcxeA==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -2564,9 +2565,9 @@
] ]
}, },
"node_modules/@rollup/rollup-win32-x64-msvc": { "node_modules/@rollup/rollup-win32-x64-msvc": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.57.1.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.59.0.tgz",
"integrity": "sha512-mxRFDdHIWRxg3UfIIAwCm6NzvxG0jDX/wBN6KsQFTvKFqqg9vTrWUE68qEjHt19A5wwx5X5aUi2zuZT7YR0jrA==", "integrity": "sha512-2HRCml6OztYXyJXAvdDXPKcawukWY2GpR5/nxKp4iBgiO3wcoEGkAaqctIbZcNB6KlUQBIqt8VYkNSj2397EfA==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -2624,9 +2625,9 @@
"license": "MIT" "license": "MIT"
}, },
"node_modules/@types/node": { "node_modules/@types/node": {
"version": "25.3.0", "version": "25.3.3",
"resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.0.tgz", "resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.3.tgz",
"integrity": "sha512-4K3bqJpXpqfg2XKGK9bpDTc6xO/xoUP/RBWS7AtRMug6zZFaRekiLzjVtAoZMquxoAbzBvy5nxQ7veS5eYzf8A==", "integrity": "sha512-DpzbrH7wIcBaJibpKo9nnSQL0MTRdnWttGyE5haGwK86xgMOkFLp7vEyfQPGLOJh5wNYiJ3V9PmUMDhV9u8kkQ==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"undici-types": "~7.18.0" "undici-types": "~7.18.0"
@@ -2656,9 +2657,9 @@
} }
}, },
"node_modules/@types/supertest": { "node_modules/@types/supertest": {
"version": "6.0.3", "version": "7.2.0",
"resolved": "https://registry.npmjs.org/@types/supertest/-/supertest-6.0.3.tgz", "resolved": "https://registry.npmjs.org/@types/supertest/-/supertest-7.2.0.tgz",
"integrity": "sha512-8WzXq62EXFhJ7QsH3Ocb/iKQ/Ty9ZVWnVzoTKc9tyyFRRF3a74Tk2+TLFgaFFw364Ere+npzHKEJ6ga2LzIL7w==", "integrity": "sha512-uh2Lv57xvggst6lCqNdFAmDSvoMG7M/HDtX4iUCquxQ5EGPtaPM5PL5Hmi7LCvOG8db7YaCPNJEeoI8s/WzIQw==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
@@ -2952,9 +2953,9 @@
} }
}, },
"node_modules/bn.js": { "node_modules/bn.js": {
"version": "4.12.2", "version": "4.12.3",
"resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.2.tgz", "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.3.tgz",
"integrity": "sha512-n4DSx829VRTRByMRGdjQ9iqsN0Bh4OolPsFnaZBLcbi8iXcB+kJ9s7EnRt4wILZNV3kPLHkRVfOc/HvhC3ovDw==", "integrity": "sha512-fGTi3gxV/23FTYdAoUtLYp6qySe2KE3teyZitipKNRuVYcBkoP/bB3guXN/XVKUe9mxCHXnc9C4ocyz8OmgN0g==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/brace-expansion": { "node_modules/brace-expansion": {
@@ -3017,6 +3018,13 @@
"node": ">=18" "node": ">=18"
} }
}, },
"node_modules/colorette": {
"version": "2.0.20",
"resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.20.tgz",
"integrity": "sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==",
"dev": true,
"license": "MIT"
},
"node_modules/combined-stream": { "node_modules/combined-stream": {
"version": "1.0.8", "version": "1.0.8",
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
@@ -3161,6 +3169,16 @@
"node": ">= 12" "node": ">= 12"
} }
}, },
"node_modules/dateformat": {
"version": "4.6.3",
"resolved": "https://registry.npmjs.org/dateformat/-/dateformat-4.6.3.tgz",
"integrity": "sha512-2P0p0pFGzHS5EMnhdxQi7aJN+iMheud0UhG4dlE1DLAlvL8JHjJJTX/CSm4JXwV0Ka5nGk3zC5mcb5bUQUxxMA==",
"dev": true,
"license": "MIT",
"engines": {
"node": "*"
}
},
"node_modules/debug": { "node_modules/debug": {
"version": "4.4.3", "version": "4.4.3",
"resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
@@ -3888,6 +3906,16 @@
"safe-buffer": "^5.0.1" "safe-buffer": "^5.0.1"
} }
}, },
"node_modules/end-of-stream": {
"version": "1.4.5",
"resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.5.tgz",
"integrity": "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg==",
"dev": true,
"license": "MIT",
"dependencies": {
"once": "^1.4.0"
}
},
"node_modules/es-define-property": { "node_modules/es-define-property": {
"version": "1.0.1", "version": "1.0.1",
"resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
@@ -4025,6 +4053,13 @@
"node": ">=12.0.0" "node": ">=12.0.0"
} }
}, },
"node_modules/fast-copy": {
"version": "4.0.2",
"resolved": "https://registry.npmjs.org/fast-copy/-/fast-copy-4.0.2.tgz",
"integrity": "sha512-ybA6PDXIXOXivLJK/z9e+Otk7ve13I4ckBvGO5I2RRmBU1gMHLVDJYEuJYhGwez7YNlYji2M2DvVU+a9mSFDlw==",
"dev": true,
"license": "MIT"
},
"node_modules/fast-decode-uri-component": { "node_modules/fast-decode-uri-component": {
"version": "1.0.1", "version": "1.0.1",
"resolved": "https://registry.npmjs.org/fast-decode-uri-component/-/fast-decode-uri-component-1.0.1.tgz", "resolved": "https://registry.npmjs.org/fast-decode-uri-component/-/fast-decode-uri-component-1.0.1.tgz",
@@ -4121,9 +4156,9 @@
} }
}, },
"node_modules/fastify": { "node_modules/fastify": {
"version": "5.7.4", "version": "5.8.1",
"resolved": "https://registry.npmjs.org/fastify/-/fastify-5.7.4.tgz", "resolved": "https://registry.npmjs.org/fastify/-/fastify-5.8.1.tgz",
"integrity": "sha512-e6l5NsRdaEP8rdD8VR0ErJASeyaRbzXYpmkrpr2SuvuMq6Si3lvsaVy5C+7gLanEkvjpMDzBXWE5HPeb/hgTxA==", "integrity": "sha512-y0kicFvvn7CYWoPOVLOcvn4YyKQz03DIY7UxmyOy21/J8eXm09R+tmb+tVDBW5h+pja30cHI5dqUcSlvY86V2A==",
"funding": [ "funding": [
{ {
"type": "github", "type": "github",
@@ -4145,7 +4180,7 @@
"fast-json-stringify": "^6.0.0", "fast-json-stringify": "^6.0.0",
"find-my-way": "^9.0.0", "find-my-way": "^9.0.0",
"light-my-request": "^6.0.0", "light-my-request": "^6.0.0",
"pino": "^10.1.0", "pino": "^9.14.0 || ^10.1.0",
"process-warning": "^5.0.0", "process-warning": "^5.0.0",
"rfdc": "^1.3.1", "rfdc": "^1.3.1",
"secure-json-parse": "^4.0.0", "secure-json-parse": "^4.0.0",
@@ -4500,6 +4535,13 @@
"node": ">=18.0.0" "node": ">=18.0.0"
} }
}, },
"node_modules/help-me": {
"version": "5.0.0",
"resolved": "https://registry.npmjs.org/help-me/-/help-me-5.0.0.tgz",
"integrity": "sha512-7xgomUX6ADmcYzFik0HzAxh/73YlKR9bmFzf51CZwR+b6YtzU2m0u49hQCqV6SvlqIqsaxovfwdvbnsw3b/zpg==",
"dev": true,
"license": "MIT"
},
"node_modules/html-escaper": { "node_modules/html-escaper": {
"version": "2.0.2", "version": "2.0.2",
"resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz",
@@ -4611,6 +4653,16 @@
"url": "https://github.com/sponsors/panva" "url": "https://github.com/sponsors/panva"
} }
}, },
"node_modules/joycon": {
"version": "3.1.1",
"resolved": "https://registry.npmjs.org/joycon/-/joycon-3.1.1.tgz",
"integrity": "sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==",
"dev": true,
"license": "MIT",
"engines": {
"node": ">=10"
}
},
"node_modules/js-base64": { "node_modules/js-base64": {
"version": "3.7.8", "version": "3.7.8",
"resolved": "https://registry.npmjs.org/js-base64/-/js-base64-3.7.8.tgz", "resolved": "https://registry.npmjs.org/js-base64/-/js-base64-3.7.8.tgz",
@@ -4838,9 +4890,9 @@
"license": "ISC" "license": "ISC"
}, },
"node_modules/minimatch": { "node_modules/minimatch": {
"version": "10.2.2", "version": "10.2.4",
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.2.tgz", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.4.tgz",
"integrity": "sha512-+G4CpNBxa5MprY+04MbgOw1v7So6n5JY166pFi9KfYwT78fxScCeSNQSNzp6dpPSW2rONOps6Ocam1wFhCgoVw==", "integrity": "sha512-oRjTw/97aTBN0RHbYCdtF1MQfvusSIBQM0IZEgzl6426+8jSC0nF1a/GmnVLpfB9yyr6g6FTqWqiZVbxrtaCIg==",
"license": "BlueOak-1.0.0", "license": "BlueOak-1.0.0",
"dependencies": { "dependencies": {
"brace-expansion": "^5.0.2" "brace-expansion": "^5.0.2"
@@ -4852,6 +4904,16 @@
"url": "https://github.com/sponsors/isaacs" "url": "https://github.com/sponsors/isaacs"
} }
}, },
"node_modules/minimist": {
"version": "1.2.8",
"resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz",
"integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
"dev": true,
"license": "MIT",
"funding": {
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/minipass": { "node_modules/minipass": {
"version": "7.1.2", "version": "7.1.2",
"resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz",
@@ -5117,6 +5179,41 @@
"split2": "^4.0.0" "split2": "^4.0.0"
} }
}, },
"node_modules/pino-pretty": {
"version": "13.1.3",
"resolved": "https://registry.npmjs.org/pino-pretty/-/pino-pretty-13.1.3.tgz",
"integrity": "sha512-ttXRkkOz6WWC95KeY9+xxWL6AtImwbyMHrL1mSwqwW9u+vLp/WIElvHvCSDg0xO/Dzrggz1zv3rN5ovTRVowKg==",
"dev": true,
"license": "MIT",
"dependencies": {
"colorette": "^2.0.7",
"dateformat": "^4.6.3",
"fast-copy": "^4.0.0",
"fast-safe-stringify": "^2.1.1",
"help-me": "^5.0.0",
"joycon": "^3.1.1",
"minimist": "^1.2.6",
"on-exit-leak-free": "^2.1.0",
"pino-abstract-transport": "^3.0.0",
"pump": "^3.0.0",
"secure-json-parse": "^4.0.0",
"sonic-boom": "^4.0.1",
"strip-json-comments": "^5.0.2"
},
"bin": {
"pino-pretty": "bin.js"
}
},
"node_modules/pino-pretty/node_modules/pino-abstract-transport": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/pino-abstract-transport/-/pino-abstract-transport-3.0.0.tgz",
"integrity": "sha512-wlfUczU+n7Hy/Ha5j9a/gZNy7We5+cXp8YL+X+PG8S0KXxw7n/JXA3c46Y0zQznIJ83URJiwy7Lh56WLokNuxg==",
"dev": true,
"license": "MIT",
"dependencies": {
"split2": "^4.0.0"
}
},
"node_modules/pino-std-serializers": { "node_modules/pino-std-serializers": {
"version": "7.0.0", "version": "7.0.0",
"resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-7.0.0.tgz", "resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-7.0.0.tgz",
@@ -5174,6 +5271,17 @@
"integrity": "sha512-7nJ6v5lnJsXwGprnGXga4wx6d1POjvi5Qmf1ivTRxTjH4Z/9Czja/UCMLVmB9N93GeWOU93XaFaEt6jbuoagNw==", "integrity": "sha512-7nJ6v5lnJsXwGprnGXga4wx6d1POjvi5Qmf1ivTRxTjH4Z/9Czja/UCMLVmB9N93GeWOU93XaFaEt6jbuoagNw==",
"license": "ISC" "license": "ISC"
}, },
"node_modules/pump": {
"version": "3.0.3",
"resolved": "https://registry.npmjs.org/pump/-/pump-3.0.3.tgz",
"integrity": "sha512-todwxLMY7/heScKmntwQG8CXVkWUOdYxIvY2s0VWAAMh/nd8SoYiRaKjlr7+iCs984f2P8zvrfWcDDYVb73NfA==",
"dev": true,
"license": "MIT",
"dependencies": {
"end-of-stream": "^1.1.0",
"once": "^1.3.1"
}
},
"node_modules/qs": { "node_modules/qs": {
"version": "6.14.2", "version": "6.14.2",
"resolved": "https://registry.npmjs.org/qs/-/qs-6.14.2.tgz", "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.2.tgz",
@@ -5250,9 +5358,9 @@
"license": "MIT" "license": "MIT"
}, },
"node_modules/rollup": { "node_modules/rollup": {
"version": "4.57.1", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/rollup/-/rollup-4.57.1.tgz", "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.59.0.tgz",
"integrity": "sha512-oQL6lgK3e2QZeQ7gcgIkS2YZPg5slw37hYufJ3edKlfQSGGm8ICoxswK15ntSzF/a8+h7ekRy7k7oWc3BQ7y8A==", "integrity": "sha512-2oMpl67a3zCH9H79LeMcbDhXW/UmWG/y2zuqnF2jQq5uq9TbM9TVyXvA4+t+ne2IIkBdrLpAaRQAvo7YI/Yyeg==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
@@ -5266,31 +5374,31 @@
"npm": ">=8.0.0" "npm": ">=8.0.0"
}, },
"optionalDependencies": { "optionalDependencies": {
"@rollup/rollup-android-arm-eabi": "4.57.1", "@rollup/rollup-android-arm-eabi": "4.59.0",
"@rollup/rollup-android-arm64": "4.57.1", "@rollup/rollup-android-arm64": "4.59.0",
"@rollup/rollup-darwin-arm64": "4.57.1", "@rollup/rollup-darwin-arm64": "4.59.0",
"@rollup/rollup-darwin-x64": "4.57.1", "@rollup/rollup-darwin-x64": "4.59.0",
"@rollup/rollup-freebsd-arm64": "4.57.1", "@rollup/rollup-freebsd-arm64": "4.59.0",
"@rollup/rollup-freebsd-x64": "4.57.1", "@rollup/rollup-freebsd-x64": "4.59.0",
"@rollup/rollup-linux-arm-gnueabihf": "4.57.1", "@rollup/rollup-linux-arm-gnueabihf": "4.59.0",
"@rollup/rollup-linux-arm-musleabihf": "4.57.1", "@rollup/rollup-linux-arm-musleabihf": "4.59.0",
"@rollup/rollup-linux-arm64-gnu": "4.57.1", "@rollup/rollup-linux-arm64-gnu": "4.59.0",
"@rollup/rollup-linux-arm64-musl": "4.57.1", "@rollup/rollup-linux-arm64-musl": "4.59.0",
"@rollup/rollup-linux-loong64-gnu": "4.57.1", "@rollup/rollup-linux-loong64-gnu": "4.59.0",
"@rollup/rollup-linux-loong64-musl": "4.57.1", "@rollup/rollup-linux-loong64-musl": "4.59.0",
"@rollup/rollup-linux-ppc64-gnu": "4.57.1", "@rollup/rollup-linux-ppc64-gnu": "4.59.0",
"@rollup/rollup-linux-ppc64-musl": "4.57.1", "@rollup/rollup-linux-ppc64-musl": "4.59.0",
"@rollup/rollup-linux-riscv64-gnu": "4.57.1", "@rollup/rollup-linux-riscv64-gnu": "4.59.0",
"@rollup/rollup-linux-riscv64-musl": "4.57.1", "@rollup/rollup-linux-riscv64-musl": "4.59.0",
"@rollup/rollup-linux-s390x-gnu": "4.57.1", "@rollup/rollup-linux-s390x-gnu": "4.59.0",
"@rollup/rollup-linux-x64-gnu": "4.57.1", "@rollup/rollup-linux-x64-gnu": "4.59.0",
"@rollup/rollup-linux-x64-musl": "4.57.1", "@rollup/rollup-linux-x64-musl": "4.59.0",
"@rollup/rollup-openbsd-x64": "4.57.1", "@rollup/rollup-openbsd-x64": "4.59.0",
"@rollup/rollup-openharmony-arm64": "4.57.1", "@rollup/rollup-openharmony-arm64": "4.59.0",
"@rollup/rollup-win32-arm64-msvc": "4.57.1", "@rollup/rollup-win32-arm64-msvc": "4.59.0",
"@rollup/rollup-win32-ia32-msvc": "4.57.1", "@rollup/rollup-win32-ia32-msvc": "4.59.0",
"@rollup/rollup-win32-x64-gnu": "4.57.1", "@rollup/rollup-win32-x64-gnu": "4.59.0",
"@rollup/rollup-win32-x64-msvc": "4.57.1", "@rollup/rollup-win32-x64-msvc": "4.59.0",
"fsevents": "~2.3.2" "fsevents": "~2.3.2"
} }
}, },
@@ -5630,6 +5738,19 @@
"reusify": "^1.0.0" "reusify": "^1.0.0"
} }
}, },
"node_modules/strip-json-comments": {
"version": "5.0.3",
"resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-5.0.3.tgz",
"integrity": "sha512-1tB5mhVo7U+ETBKNf92xT4hrQa3pm0MZ0PQvuDnWgAAGHDsfp4lPSpiS6psrSiet87wyGPh9ft6wmhOMQ0hDiw==",
"dev": true,
"license": "MIT",
"engines": {
"node": ">=14.16"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/superagent": { "node_modules/superagent": {
"version": "10.3.0", "version": "10.3.0",
"resolved": "https://registry.npmjs.org/superagent/-/superagent-10.3.0.tgz", "resolved": "https://registry.npmjs.org/superagent/-/superagent-10.3.0.tgz",
+5 -4
View File
@@ -1,6 +1,6 @@
{ {
"name": "medassist-ng-backend", "name": "medassist-ng-backend",
"version": "1.16.0", "version": "1.19.0",
"private": true, "private": true,
"type": "module", "type": "module",
"scripts": { "scripts": {
@@ -29,7 +29,7 @@
"argon2": "^0.44.0", "argon2": "^0.44.0",
"dotenv": "^17.3.1", "dotenv": "^17.3.1",
"drizzle-orm": "^0.45.1", "drizzle-orm": "^0.45.1",
"fastify": "^5.7.4", "fastify": "^5.8.1",
"nodemailer": "^8.0.1", "nodemailer": "^8.0.1",
"openid-client": "^6.8.2", "openid-client": "^6.8.2",
"sharp": "^0.34.5", "sharp": "^0.34.5",
@@ -37,11 +37,12 @@
}, },
"devDependencies": { "devDependencies": {
"@biomejs/biome": "^2.4.4", "@biomejs/biome": "^2.4.4",
"@types/node": "^25.3.0", "@types/node": "^25.3.3",
"@types/nodemailer": "^7.0.11", "@types/nodemailer": "^7.0.11",
"@types/supertest": "^6.0.2", "@types/supertest": "^7.2.0",
"@vitest/coverage-v8": "^4.0.18", "@vitest/coverage-v8": "^4.0.18",
"drizzle-kit": "^0.31.9", "drizzle-kit": "^0.31.9",
"pino-pretty": "^13.1.3",
"supertest": "^7.2.2", "supertest": "^7.2.2",
"tsx": "^4.19.0", "tsx": "^4.19.0",
"typescript": "^5.5.4", "typescript": "^5.5.4",
-4
View File
@@ -78,10 +78,6 @@ async function runMigrations() {
const migrateResult = await runDrizzleMigrations(db); const migrateResult = await runDrizzleMigrations(db);
if (!migrateResult.success) { if (!migrateResult.success) {
log.error(`[DB] Migration error: ${migrateResult.error}`); log.error(`[DB] Migration error: ${migrateResult.error}`);
} else if (migrateResult.warning) {
log.warn(`[DB] Migration warning: ${migrateResult.warning}`);
} else {
log.debug(`[DB] Drizzle migrations completed`);
} }
// Run ALTER TABLE migrations for backward compatibility // Run ALTER TABLE migrations for backward compatibility
+13 -6
View File
@@ -88,13 +88,12 @@ export async function runDrizzleMigrations(
await migrate(database, { migrationsFolder }); await migrate(database, { migrationsFolder });
return { success: true }; return { success: true };
} catch (err: unknown) { } catch (err: unknown) {
// If the error is about existing schema objects, the DB is already up-to-date const msg = (err as Error).message ?? "";
// This happens when ALTER migrations in client.ts have already added the columns, // Duplicate column / already exists = DB is already up-to-date (expected for existing DBs)
// or when tables were created before drizzle migrations were introduced if (msg.includes("duplicate column") || msg.includes("already exists")) {
if ((err as Error).message?.includes("duplicate column") || (err as Error).message?.includes("already exists")) { return { success: true };
return { success: true, warning: `Schema already up-to-date: ${(err as Error).message}` };
} }
return { success: false, error: (err as Error).message }; return { success: false, error: msg };
} }
} }
@@ -126,6 +125,14 @@ export async function runAlterMigrations(client: Client): Promise<{ success: boo
`ALTER TABLE medications ADD COLUMN obsolete_at integer`, `ALTER TABLE medications ADD COLUMN obsolete_at integer`,
// Added for explicit medication lifecycle start date // Added for explicit medication lifecycle start date
`ALTER TABLE medications ADD COLUMN medication_start_date text NOT NULL DEFAULT ''`, `ALTER TABLE medications ADD COLUMN medication_start_date text NOT NULL DEFAULT ''`,
// Added for form/lifecycle modeling (V1 medication forms)
`ALTER TABLE medications ADD COLUMN medication_form text NOT NULL DEFAULT 'tablet'`,
`ALTER TABLE medications ADD COLUMN pill_form text`,
`ALTER TABLE medications ADD COLUMN lifecycle_category text NOT NULL DEFAULT 'refill_when_empty'`,
`ALTER TABLE medications ADD COLUMN medication_end_date text`,
`ALTER TABLE medications ADD COLUMN auto_mark_obsolete_after_end_date integer NOT NULL DEFAULT 1`,
`ALTER TABLE medications ADD COLUMN package_amount_value integer NOT NULL DEFAULT 0`,
`ALTER TABLE medications ADD COLUMN package_amount_unit text NOT NULL DEFAULT 'ml'`,
// Added for more detailed reminder info display // Added for more detailed reminder info display
`ALTER TABLE user_settings ADD COLUMN last_reminder_med_name text`, `ALTER TABLE user_settings ADD COLUMN last_reminder_med_name text`,
`ALTER TABLE user_settings ADD COLUMN last_reminder_taken_by text`, `ALTER TABLE user_settings ADD COLUMN last_reminder_taken_by text`,
+9
View File
@@ -29,6 +29,11 @@ export const medications = sqliteTable("medications", {
genericName: text("generic_name", { length: 100 }), genericName: text("generic_name", { length: 100 }),
takenByJson: text("taken_by_json").notNull().default("[]"), // JSON array of person names takenByJson: text("taken_by_json").notNull().default("[]"), // JSON array of person names
packageType: text("package_type", { length: 20 }).notNull().default("blister"), // 'blister' or 'bottle' packageType: text("package_type", { length: 20 }).notNull().default("blister"), // 'blister' or 'bottle'
medicationForm: text("medication_form", { length: 20 }).notNull().default("tablet"), // 'capsule' | 'tablet' | 'liquid' | 'topical'
pillForm: text("pill_form", { length: 20 }), // Only for blister/bottle with pill-based medications: 'tablet' | 'capsule'
lifecycleCategory: text("lifecycle_category", { length: 30 }).notNull().default("refill_when_empty"), // 'refill_when_empty' | 'treatment_period'
packageAmountValue: integer("package_amount_value").notNull().default(0), // Informational package quantity (ml/g)
packageAmountUnit: text("package_amount_unit", { length: 10 }).notNull().default("ml"), // 'ml' | 'g'
packCount: integer("pack_count").notNull().default(1), packCount: integer("pack_count").notNull().default(1),
blistersPerPack: integer("blisters_per_pack").notNull().default(1), blistersPerPack: integer("blisters_per_pack").notNull().default(1),
pillsPerBlister: integer("pills_per_blister").notNull().default(1), pillsPerBlister: integer("pills_per_blister").notNull().default(1),
@@ -48,6 +53,10 @@ export const medications = sqliteTable("medications", {
notes: text("notes"), notes: text("notes"),
intakeRemindersEnabled: integer("intake_reminders_enabled", { mode: "boolean" }).notNull().default(false), intakeRemindersEnabled: integer("intake_reminders_enabled", { mode: "boolean" }).notNull().default(false),
medicationStartDate: text("medication_start_date").notNull().default(""), medicationStartDate: text("medication_start_date").notNull().default(""),
medicationEndDate: text("medication_end_date"),
autoMarkObsoleteAfterEndDate: integer("auto_mark_obsolete_after_end_date", { mode: "boolean" })
.notNull()
.default(true),
isObsolete: integer("is_obsolete", { mode: "boolean" }).notNull().default(false), isObsolete: integer("is_obsolete", { mode: "boolean" }).notNull().default(false),
obsoleteAt: integer("obsolete_at", { mode: "timestamp" }), obsoleteAt: integer("obsolete_at", { mode: "timestamp" }),
prescriptionEnabled: integer("prescription_enabled", { mode: "boolean" }).notNull().default(false), prescriptionEnabled: integer("prescription_enabled", { mode: "boolean" }).notNull().default(false),
+6
View File
@@ -179,6 +179,8 @@ type TranslationKeys = {
common: { common: {
pill: string; pill: string;
pills: string; pills: string;
units: string;
ml: string;
blister: string; blister: string;
blisters: string; blisters: string;
day: string; day: string;
@@ -299,6 +301,8 @@ const translations: Record<Language, TranslationKeys> = {
common: { common: {
pill: "pill", pill: "pill",
pills: "pills", pills: "pills",
units: "units",
ml: "ml",
blister: "blister", blister: "blister",
blisters: "blisters", blisters: "blisters",
day: "day", day: "day",
@@ -420,6 +424,8 @@ const translations: Record<Language, TranslationKeys> = {
common: { common: {
pill: "Tablette", pill: "Tablette",
pills: "Tabletten", pills: "Tabletten",
units: "Einheiten",
ml: "ml",
blister: "Blister", blister: "Blister",
blisters: "Blister", blisters: "Blister",
day: "Tag", day: "Tag",
+17 -4
View File
@@ -57,6 +57,21 @@ function sanitizeCorrelationId(headers: IncomingHttpHeaders): string | null {
return trimmed; return trimmed;
} }
function buildLoggerOptions(level: string) {
const base = {
level,
timestamp: () => `,"time":"${new Date().toISOString()}"`,
};
// Human-readable logs in development, structured JSON in production/test
if (process.env.NODE_ENV !== "production" && process.env.NODE_ENV !== "test") {
return {
...base,
transport: { target: "pino-pretty", options: { translateTime: "SYS:yyyy-mm-dd HH:MM:ss.l" } },
};
}
return base;
}
/** Create and configure Fastify app (without starting) */ /** Create and configure Fastify app (without starting) */
export async function createApp(options?: { export async function createApp(options?: {
logLevel?: string; logLevel?: string;
@@ -84,7 +99,7 @@ export async function createApp(options?: {
}; };
const app = Fastify({ const app = Fastify({
logger: { level: opts.logLevel }, logger: buildLoggerOptions(opts.logLevel),
genReqId: (request) => sanitizeCorrelationId(request.headers) ?? randomUUID(), genReqId: (request) => sanitizeCorrelationId(request.headers) ?? randomUUID(),
}); });
@@ -157,9 +172,7 @@ log.info("[DB] Migrations complete, starting server...");
const imagesDir = ensureImagesDirectory(); const imagesDir = ensureImagesDirectory();
const app = Fastify({ const app = Fastify({
logger: { logger: buildLoggerOptions(env.LOG_LEVEL),
level: env.LOG_LEVEL,
},
genReqId: (request) => sanitizeCorrelationId(request.headers) ?? randomUUID(), genReqId: (request) => sanitizeCorrelationId(request.headers) ?? randomUUID(),
}); });
+6 -3
View File
@@ -47,7 +47,7 @@ export async function getAnonymousUserId(): Promise<number> {
export interface AuthState { export interface AuthState {
authEnabled: boolean; authEnabled: boolean;
registrationEnabled: boolean; registrationEnabled: boolean;
localAuthEnabled: boolean; formLoginEnabled: boolean;
oidcEnabled: boolean; oidcEnabled: boolean;
oidcProviderName: string; oidcProviderName: string;
hasUsers: boolean; hasUsers: boolean;
@@ -59,15 +59,18 @@ export async function getAuthState(): Promise<AuthState> {
const [result] = await db.select({ count: count() }).from(users).where(sql`${users.id} != ${ANONYMOUS_USER_ID}`); const [result] = await db.select({ count: count() }).from(users).where(sql`${users.id} != ${ANONYMOUS_USER_ID}`);
const hasUsers = result.count > 0; const hasUsers = result.count > 0;
const needsSetup = env.AUTH_ENABLED && !hasUsers;
return { return {
authEnabled: env.AUTH_ENABLED, authEnabled: env.AUTH_ENABLED,
// Registration: enabled via ENV OR no users exist (first-time setup) // Registration: enabled via ENV OR no users exist (first-time setup)
registrationEnabled: env.REGISTRATION_ENABLED || !hasUsers, registrationEnabled: env.REGISTRATION_ENABLED || !hasUsers,
localAuthEnabled: env.AUTH_ENABLED, // Password auth available when auth is enabled // Form login: enabled when auth + form login are both on, or forced on for first-user setup
formLoginEnabled: needsSetup || (env.AUTH_ENABLED && env.FORM_LOGIN_ENABLED),
oidcEnabled: env.OIDC_ENABLED, oidcEnabled: env.OIDC_ENABLED,
oidcProviderName: env.OIDC_PROVIDER_NAME, oidcProviderName: env.OIDC_PROVIDER_NAME,
hasUsers, hasUsers,
needsSetup: env.AUTH_ENABLED && !hasUsers, needsSetup,
}; };
} }
+27 -1
View File
@@ -28,7 +28,11 @@ const EnvSchema = z.object({
.string() .string()
.transform((v) => v === "true") .transform((v) => v === "true")
.default("false"), .default("false"),
// Disable local auth when using SSO only // Disable username/password form login (useful for OIDC-only setups)
FORM_LOGIN_ENABLED: z
.string()
.transform((v) => v === "true")
.default("true"),
// JWT Secrets - only required when AUTH_ENABLED=true // JWT Secrets - only required when AUTH_ENABLED=true
JWT_SECRET: z.string().min(10).optional(), JWT_SECRET: z.string().min(10).optional(),
@@ -128,4 +132,26 @@ if (parsed.OIDC_ENABLED) {
} }
} }
// Validate that at least one login method is available when auth is enabled
if (parsed.AUTH_ENABLED && !parsed.FORM_LOGIN_ENABLED && !parsed.OIDC_ENABLED) {
console.error("=".repeat(60));
console.error("AUTHENTICATION CONFIGURATION ERROR");
console.error("=".repeat(60));
console.error("AUTH_ENABLED=true but no login method is available.");
console.error("FORM_LOGIN_ENABLED=false and OIDC_ENABLED=false means users cannot log in.");
console.error("");
console.error("To fix this, either:");
console.error(" 1. Set FORM_LOGIN_ENABLED=true to allow username/password login");
console.error(" 2. Set OIDC_ENABLED=true to allow SSO login");
console.error("=".repeat(60));
process.exit(1);
}
// Warn about ineffective registration when form login is disabled
if (parsed.REGISTRATION_ENABLED && !parsed.FORM_LOGIN_ENABLED) {
console.warn(
"[config] REGISTRATION_ENABLED=true has no effect when FORM_LOGIN_ENABLED=false (no registration form available)"
);
}
export const env = parsed; export const env = parsed;
+4 -4
View File
@@ -123,8 +123,8 @@ export async function authRoutes(app: FastifyInstance) {
return reply.status(400).send({ error: "Registration is disabled", code: "REGISTRATION_DISABLED" }); return reply.status(400).send({ error: "Registration is disabled", code: "REGISTRATION_DISABLED" });
} }
if (!state.localAuthEnabled) { if (!state.formLoginEnabled) {
return reply.status(400).send({ error: "Local authentication is disabled", code: "LOCAL_AUTH_DISABLED" }); return reply.status(400).send({ error: "Form login is disabled", code: "FORM_LOGIN_DISABLED" });
} }
// Validate input // Validate input
@@ -185,8 +185,8 @@ export async function authRoutes(app: FastifyInstance) {
return reply.status(400).send({ error: "Authentication is disabled", code: "AUTH_DISABLED" }); return reply.status(400).send({ error: "Authentication is disabled", code: "AUTH_DISABLED" });
} }
if (!state.localAuthEnabled) { if (!state.formLoginEnabled) {
return reply.status(400).send({ error: "Local authentication is disabled", code: "LOCAL_AUTH_DISABLED" }); return reply.status(400).send({ error: "Form login is disabled", code: "FORM_LOGIN_DISABLED" });
} }
const parsed = loginSchema.safeParse(request.body); const parsed = loginSchema.safeParse(request.body);
+4 -2
View File
@@ -137,8 +137,9 @@ async function validateShareDoseId(share: typeof shareTokens.$inferSelect, doseI
export async function doseRoutes(app: FastifyInstance) { export async function doseRoutes(app: FastifyInstance) {
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// GET /doses/taken - PROTECTED: Get all taken doses for the user // GET /doses/taken - PROTECTED: Get all taken doses for the user
// Suppress request logs — polled every 5s by frontend
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
app.get("/doses/taken", { preHandler: requireAuth }, async (request, reply) => { app.get("/doses/taken", { preHandler: requireAuth, logLevel: "warn" }, async (request, reply) => {
const userId = await getUserId(request, reply); const userId = await getUserId(request, reply);
// Get all taken doses for this user (no time limit) // Get all taken doses for this user (no time limit)
@@ -304,8 +305,9 @@ export async function doseRoutes(app: FastifyInstance) {
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// GET /share/:token/doses - PUBLIC: Get taken doses for a share link // GET /share/:token/doses - PUBLIC: Get taken doses for a share link
// Suppress request logs — polled every 5s by SharedSchedule
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
app.get<{ Params: { token: string } }>("/share/:token/doses", async (request, reply) => { app.get<{ Params: { token: string } }>("/share/:token/doses", { logLevel: "warn" }, async (request, reply) => {
const { token } = request.params; const { token } = request.params;
const { share, reason } = await getActiveShareToken(token); const { share, reason } = await getActiveShareToken(token);
+37 -7
View File
@@ -10,6 +10,7 @@ import { doseTracking, medications, refillHistory, shareTokens, userSettings } f
import { getAnonymousUserId, requireAuth } from "../plugins/auth.js"; import { getAnonymousUserId, requireAuth } from "../plugins/auth.js";
import { env } from "../plugins/env.js"; import { env } from "../plugins/env.js";
import type { AuthUser } from "../types/fastify.js"; import type { AuthUser } from "../types/fastify.js";
import { normalizePackageType, PACKAGE_TYPES } from "../utils/package-profiles.js";
import { parseIntakesJson, parseTakenByJson } from "../utils/scheduler-utils.js"; import { parseIntakesJson, parseTakenByJson } from "../utils/scheduler-utils.js";
const IMAGES_DIR = resolve(getDataDir(), "images"); const IMAGES_DIR = resolve(getDataDir(), "images");
@@ -17,7 +18,7 @@ const IMAGES_DIR = resolve(getDataDir(), "images");
// ============================================================================= // =============================================================================
// Export Format Version (bump this when format changes) // Export Format Version (bump this when format changes)
// ============================================================================= // =============================================================================
const EXPORT_VERSION = "1.1"; const EXPORT_VERSION = "1.3";
// ============================================================================= // =============================================================================
// Zod Schemas for Import Validation // Zod Schemas for Import Validation
@@ -27,6 +28,7 @@ const scheduleSchema = z.object({
usage: z.number().nonnegative(), usage: z.number().nonnegative(),
every: z.number().int().min(1), every: z.number().int().min(1),
start: z.string(), // ISO datetime string start: z.string(), // ISO datetime string
intakeUnit: z.enum(["ml", "tsp", "tbsp"]).nullable().optional(),
remind: z.boolean().optional().default(false), remind: z.boolean().optional().default(false),
takenBy: z.string().nullable().optional(), // Per-intake takenBy (new field) takenBy: z.string().nullable().optional(), // Per-intake takenBy (new field)
}); });
@@ -38,7 +40,9 @@ const inventorySchema = z.object({
totalPills: z.number().int().nullable().optional(), // For bottle type: total capacity totalPills: z.number().int().nullable().optional(), // For bottle type: total capacity
looseTablets: z.number().int().min(0).default(0), looseTablets: z.number().int().min(0).default(0),
stockAdjustment: z.number().int().default(0), // Manual stock correction stockAdjustment: z.number().int().default(0), // Manual stock correction
packageType: z.enum(["blister", "bottle"]).default("blister"), packageType: z.enum(PACKAGE_TYPES).default("blister"),
packageAmountValue: z.number().int().min(0).default(0),
packageAmountUnit: z.enum(["ml", "g"]).default("ml"),
}); });
const medicationExportSchema = z.object({ const medicationExportSchema = z.object({
@@ -46,11 +50,16 @@ const medicationExportSchema = z.object({
name: z.string().min(1), name: z.string().min(1),
genericName: z.string().nullable().optional(), genericName: z.string().nullable().optional(),
takenBy: z.array(z.string()).default([]), takenBy: z.array(z.string()).default([]),
medicationForm: z.enum(["capsule", "tablet", "liquid", "topical"]).default("tablet"),
pillForm: z.enum(["capsule", "tablet"]).nullable().optional(),
lifecycleCategory: z.enum(["refill_when_empty", "treatment_period"]).default("refill_when_empty"),
inventory: inventorySchema, inventory: inventorySchema,
pillWeightMg: z.number().int().nullable().optional(), pillWeightMg: z.number().int().nullable().optional(),
doseUnit: z.enum(["mg", "g", "mcg", "ml", "IU", "units", "drops", "puffs"]).default("mg"), doseUnit: z.enum(["mg", "g", "mcg", "ml", "IU", "units", "drops", "puffs"]).default("mg"),
schedules: z.array(scheduleSchema).default([]), schedules: z.array(scheduleSchema).default([]),
medicationStartDate: z.string().nullable().optional(), medicationStartDate: z.string().nullable().optional(),
medicationEndDate: z.string().nullable().optional(),
autoMarkObsoleteAfterEndDate: z.boolean().default(true),
expiryDate: z.string().nullable().optional(), expiryDate: z.string().nullable().optional(),
notes: z.string().nullable().optional(), notes: z.string().nullable().optional(),
intakeRemindersEnabled: z.boolean().default(false), intakeRemindersEnabled: z.boolean().default(false),
@@ -155,9 +164,14 @@ async function getUserId(request: FastifyRequest, reply: FastifyReply): Promise<
} }
// Parse intakes from DB format to export format (with per-intake takenBy) // Parse intakes from DB format to export format (with per-intake takenBy)
function parseIntakesForExport( function parseIntakesForExport(row: typeof medications.$inferSelect): Array<{
row: typeof medications.$inferSelect usage: number;
): Array<{ usage: number; every: number; start: string; remind: boolean; takenBy: string | null }> { every: number;
start: string;
intakeUnit: "ml" | "tsp" | "tbsp" | null;
remind: boolean;
takenBy: string | null;
}> {
// Use the new parseIntakesJson which falls back to legacy format // Use the new parseIntakesJson which falls back to legacy format
const intakes = parseIntakesJson( const intakes = parseIntakesJson(
row.intakesJson, row.intakesJson,
@@ -169,6 +183,7 @@ function parseIntakesForExport(
usage: intake.usage, usage: intake.usage,
every: intake.every, every: intake.every,
start: intake.start, start: intake.start,
intakeUnit: null,
remind: intake.intakeRemindersEnabled, remind: intake.intakeRemindersEnabled,
takenBy: intake.takenBy, // Per-intake takenBy takenBy: intake.takenBy, // Per-intake takenBy
})); }));
@@ -295,6 +310,9 @@ export async function exportRoutes(app: FastifyInstance) {
name: med.name, name: med.name,
genericName: med.genericName, genericName: med.genericName,
takenBy: parseTakenByJson(med.takenByJson), takenBy: parseTakenByJson(med.takenByJson),
medicationForm: med.medicationForm ?? "tablet",
pillForm: med.pillForm ?? null,
lifecycleCategory: med.lifecycleCategory ?? "refill_when_empty",
inventory: { inventory: {
packCount: med.packCount ?? 1, packCount: med.packCount ?? 1,
blistersPerPack: med.blistersPerPack ?? 1, blistersPerPack: med.blistersPerPack ?? 1,
@@ -302,12 +320,16 @@ export async function exportRoutes(app: FastifyInstance) {
totalPills: med.totalPills ?? null, totalPills: med.totalPills ?? null,
looseTablets: med.looseTablets ?? 0, looseTablets: med.looseTablets ?? 0,
stockAdjustment: med.stockAdjustment ?? 0, stockAdjustment: med.stockAdjustment ?? 0,
packageType: med.packageType ?? "blister", packageType: normalizePackageType(med.packageType),
packageAmountValue: med.packageAmountValue ?? 0,
packageAmountUnit: (med.packageAmountUnit ?? "ml") as "ml" | "g",
}, },
pillWeightMg: med.pillWeightMg, pillWeightMg: med.pillWeightMg,
doseUnit: med.doseUnit ?? "mg", doseUnit: med.doseUnit ?? "mg",
schedules: parseIntakesForExport(med), schedules: parseIntakesForExport(med),
medicationStartDate: med.medicationStartDate || null, medicationStartDate: med.medicationStartDate || null,
medicationEndDate: med.medicationEndDate || null,
autoMarkObsoleteAfterEndDate: med.autoMarkObsoleteAfterEndDate ?? true,
expiryDate: med.expiryDate, expiryDate: med.expiryDate,
notes: med.notes, notes: med.notes,
intakeRemindersEnabled: med.intakeRemindersEnabled ?? false, intakeRemindersEnabled: med.intakeRemindersEnabled ?? false,
@@ -555,6 +577,7 @@ export async function exportRoutes(app: FastifyInstance) {
usage: s.usage, usage: s.usage,
every: s.every, every: s.every,
start: s.start, start: s.start,
intakeUnit: s.intakeUnit ?? null,
takenBy: s.takenBy || null, takenBy: s.takenBy || null,
intakeRemindersEnabled: s.remind ?? false, intakeRemindersEnabled: s.remind ?? false,
})) }))
@@ -570,7 +593,12 @@ export async function exportRoutes(app: FastifyInstance) {
name: med.name, name: med.name,
genericName: med.genericName || null, genericName: med.genericName || null,
takenByJson, takenByJson,
packageType: med.inventory.packageType ?? "blister", medicationForm: med.medicationForm ?? "tablet",
pillForm: med.pillForm || null,
lifecycleCategory: med.lifecycleCategory ?? "refill_when_empty",
packageType: normalizePackageType(med.inventory.packageType),
packageAmountValue: med.inventory.packageAmountValue ?? 0,
packageAmountUnit: med.inventory.packageAmountUnit ?? "ml",
packCount: med.inventory.packCount, packCount: med.inventory.packCount,
blistersPerPack: med.inventory.blistersPerPack, blistersPerPack: med.inventory.blistersPerPack,
pillsPerBlister: med.inventory.pillsPerBlister, pillsPerBlister: med.inventory.pillsPerBlister,
@@ -581,6 +609,8 @@ export async function exportRoutes(app: FastifyInstance) {
pillWeightMg: med.pillWeightMg || null, pillWeightMg: med.pillWeightMg || null,
doseUnit: med.doseUnit ?? "mg", doseUnit: med.doseUnit ?? "mg",
medicationStartDate: med.medicationStartDate || "", medicationStartDate: med.medicationStartDate || "",
medicationEndDate: med.medicationEndDate || null,
autoMarkObsoleteAfterEndDate: med.autoMarkObsoleteAfterEndDate ?? true,
intakesJson, intakesJson,
usageJson, usageJson,
everyJson, everyJson,
+2 -3
View File
@@ -10,11 +10,10 @@ const packageJson = JSON.parse(readFileSync(packageJsonPath, "utf-8"));
const backendVersion = packageJson.version || "unknown"; const backendVersion = packageJson.version || "unknown";
export async function healthRoutes(app: FastifyInstance) { export async function healthRoutes(app: FastifyInstance) {
// Exempt from rate limit - lightweight health check // Exempt from rate limit + suppress request logs (called every 30s by Docker healthcheck)
app.get("/health", { config: { rateLimit: false } }, async () => ({ app.get("/health", { config: { rateLimit: false }, logLevel: "warn" }, async () => ({
status: "ok", status: "ok",
version: backendVersion, version: backendVersion,
smtpConfigured: Boolean(process.env.SMTP_HOST), smtpConfigured: Boolean(process.env.SMTP_HOST),
shoutrrrConfigured: Boolean(process.env.SHOUTRRR_URL),
})); }));
} }
+331 -72
View File
@@ -14,15 +14,63 @@ import {
streamToBuffer, streamToBuffer,
writeOptimizedImageSet, writeOptimizedImageSet,
} from "../utils/image-upload.js"; } from "../utils/image-upload.js";
import { type Intake, parseIntakesJson, parseLocalDateTime, parseTakenByJson } from "../utils/scheduler-utils.js"; import {
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
normalizePackageType,
PACKAGE_TYPES,
} from "../utils/package-profiles.js";
import {
type Intake,
normalizeIntakeUsageForStock,
parseIntakesJson,
parseLocalDateTime,
parseTakenByJson,
} from "../utils/scheduler-utils.js";
const IMAGES_DIR = resolve(getDataDir(), "images"); const IMAGES_DIR = resolve(getDataDir(), "images");
function isIntakeUnit(value: unknown): value is "ml" | "tsp" | "tbsp" {
return value === "ml" || value === "tsp" || value === "tbsp";
}
function parseRawIntakeUnits(intakesJson: string | null | undefined): Array<"ml" | "tsp" | "tbsp" | null> {
if (!intakesJson) return [];
try {
const parsed = JSON.parse(intakesJson);
if (!Array.isArray(parsed)) return [];
return parsed.map((item: unknown) => {
if (!item || typeof item !== "object") return null;
const unit = (item as Record<string, unknown>).intakeUnit;
return isIntakeUnit(unit) ? unit : null;
});
} catch {
return [];
}
}
function parseIntakesWithUnits(
intakesJson: string | null | undefined,
legacyRow: { usageJson: string; everyJson: string; startJson: string },
medicationIntakeRemindersEnabled?: boolean
): Intake[] {
const intakes = parseIntakesJson(intakesJson, legacyRow, medicationIntakeRemindersEnabled);
const rawUnits = parseRawIntakeUnits(intakesJson);
if (rawUnits.length === 0) return intakes;
return intakes.map((intake, idx) => ({
...intake,
intakeUnit: rawUnits[idx] ?? intake.intakeUnit ?? null,
}));
}
// New intake schema with per-intake takenBy // New intake schema with per-intake takenBy
const intakeSchema = z.object({ const intakeSchema = z.object({
usage: z.number().nonnegative(), usage: z.number().nonnegative(),
every: z.number().int().min(1), every: z.number().int().min(1),
start: z.string().datetime({ local: true }), start: z.string().datetime({ local: true }),
intakeUnit: z.enum(["ml", "tsp", "tbsp"]).nullable().optional(),
takenBy: z.string().trim().max(100).nullable().optional(), // Person for this specific intake takenBy: z.string().trim().max(100).nullable().optional(), // Person for this specific intake
intakeRemindersEnabled: z.boolean().default(false), // Per-intake reminder setting intakeRemindersEnabled: z.boolean().default(false), // Per-intake reminder setting
}); });
@@ -34,26 +82,37 @@ const blisterSchema = z.object({
start: z.string().datetime({ local: true }), start: z.string().datetime({ local: true }),
}); });
const packageTypeSchema = z.enum(["blister", "bottle"]).default("blister"); const packageTypeSchema = z.enum(PACKAGE_TYPES).default("blister");
const medicationFormSchema = z.enum(["capsule", "tablet", "liquid", "topical"]).default("tablet");
const pillFormSchema = z.enum(["capsule", "tablet"]);
const lifecycleCategorySchema = z.enum(["refill_when_empty", "treatment_period"]).default("refill_when_empty");
const doseUnitSchema = z.enum(["mg", "g", "mcg", "ml", "IU", "units", "drops", "puffs"]).default("mg"); const doseUnitSchema = z.enum(["mg", "g", "mcg", "ml", "IU", "units", "drops", "puffs"]).default("mg");
const medicationStartDateSchema = z const medicationStartDateSchema = z
.union([z.string().regex(/^\d{4}-\d{2}-\d{2}$/), z.literal(""), z.null()]) .union([z.string().regex(/^\d{4}-\d{2}-\d{2}$/), z.literal(""), z.null()])
.optional(); .optional();
const medicationEndDateSchema = z.union([z.string().regex(/^\d{4}-\d{2}-\d{2}$/), z.literal(""), z.null()]).optional();
const medicationSchema = z const medicationSchema = z
.object({ .object({
name: z.string().trim().min(1).max(100), name: z.string().trim().max(100).default(""),
genericName: z.string().trim().max(100).nullable().optional(), genericName: z.string().trim().max(100).nullable().optional(),
takenBy: z.array(z.string().trim().max(100)).default([]), // Medication-level takenBy (fallback) takenBy: z.array(z.string().trim().max(100)).default([]), // Medication-level takenBy (fallback)
medicationForm: medicationFormSchema,
pillForm: pillFormSchema.nullable().optional(),
lifecycleCategory: lifecycleCategorySchema,
packageType: packageTypeSchema, packageType: packageTypeSchema,
packCount: z.number().int().min(0).default(1), packCount: z.number().int().min(0).default(1),
blistersPerPack: z.number().int().min(1).default(1), blistersPerPack: z.number().int().min(1).default(1),
pillsPerBlister: z.number().int().min(1).default(1), pillsPerBlister: z.number().int().min(1).default(1),
packageAmountValue: z.number().int().min(0).default(0),
packageAmountUnit: z.enum(["ml", "g"]).default("ml"),
totalPills: z.number().int().min(1).nullable().optional(), // For bottle type: total capacity totalPills: z.number().int().min(1).nullable().optional(), // For bottle type: total capacity
looseTablets: z.number().int().min(0).default(0), looseTablets: z.number().int().min(0).default(0),
pillWeightMg: z.number().nonnegative().nullable().optional(), pillWeightMg: z.number().nonnegative().nullable().optional(),
doseUnit: doseUnitSchema, doseUnit: doseUnitSchema,
medicationStartDate: medicationStartDateSchema, medicationStartDate: medicationStartDateSchema,
medicationEndDate: medicationEndDateSchema,
autoMarkObsoleteAfterEndDate: z.boolean().default(true),
expiryDate: z.string().nullable().optional(), expiryDate: z.string().nullable().optional(),
notes: z.string().max(2000).nullable().optional(), notes: z.string().max(2000).nullable().optional(),
prescriptionEnabled: z.boolean().default(false), prescriptionEnabled: z.boolean().default(false),
@@ -66,6 +125,10 @@ const medicationSchema = z
intakes: z.array(intakeSchema).min(1).max(12).optional(), intakes: z.array(intakeSchema).min(1).max(12).optional(),
blisters: z.array(blisterSchema).min(1).max(12).optional(), // Legacy format blisters: z.array(blisterSchema).min(1).max(12).optional(), // Legacy format
}) })
.refine((data) => (data.name && data.name.length > 0) || (data.genericName && data.genericName.length > 0), {
message: "Either 'name' or 'genericName' must be provided",
path: ["name"],
})
.refine((data) => data.intakes || data.blisters, { message: "Either 'intakes' or 'blisters' must be provided" }) .refine((data) => data.intakes || data.blisters, { message: "Either 'intakes' or 'blisters' must be provided" })
.refine( .refine(
(data) => { (data) => {
@@ -80,6 +143,77 @@ const medicationSchema = z
path: ["medicationStartDate"], path: ["medicationStartDate"],
} }
) )
.refine(
(data) => {
const startDate = data.medicationStartDate ?? "";
const endDate = data.medicationEndDate ?? "";
if (!startDate || !endDate) return true;
return startDate <= endDate;
},
{
message: "Medication end date must be on or after medication start date",
path: ["medicationEndDate"],
}
)
.refine(
(data) => {
if (data.medicationForm === "capsule" || data.medicationForm === "tablet") {
return data.pillForm == null || data.pillForm === "capsule" || data.pillForm === "tablet";
}
return true;
},
{
message: "pillForm must be capsule or tablet for capsule/tablet medications",
path: ["pillForm"],
}
)
.refine(
(data) => {
if (data.medicationForm === "topical") {
return isTubePackageType(data.packageType);
}
return true;
},
{
message: "Topical medications must use tube package type",
path: ["packageType"],
}
)
.refine(
(data) => {
if (data.medicationForm === "liquid") {
return isLiquidContainerPackageType(data.packageType);
}
return true;
},
{
message: "Liquid medications must use liquid_container package type",
path: ["packageType"],
}
)
.refine(
(data) => {
if (data.medicationForm === "capsule" || data.medicationForm === "tablet") {
return !isTubePackageType(data.packageType) && !isLiquidContainerPackageType(data.packageType);
}
return true;
},
{
message: "Capsule and tablet medications cannot use tube or liquid_container package type",
path: ["packageType"],
}
)
.refine(
(data) => {
const schedules = data.intakes ?? data.blisters ?? [];
if (data.pillForm !== "capsule") return true;
return schedules.every((entry) => Number.isInteger(entry.usage));
},
{
message: "Fractional intake is not allowed for capsule",
path: ["intakes"],
}
)
.refine( .refine(
(data) => { (data) => {
if (!data.prescriptionEnabled) return true; if (!data.prescriptionEnabled) return true;
@@ -127,13 +261,33 @@ export async function medicationRoutes(app: FastifyInstance) {
app.get<{ Querystring: { includeObsolete?: string } }>("/medications", async (request, reply) => { app.get<{ Querystring: { includeObsolete?: string } }>("/medications", async (request, reply) => {
const userId = await getUserId(request, reply); const userId = await getUserId(request, reply);
const includeObsolete = request.query.includeObsolete === "true"; const includeObsolete = request.query.includeObsolete === "true";
const initialRows = await db
.select()
.from(medications)
.where(eq(medications.userId, userId))
.orderBy(medications.id);
const todayDate = new Date().toISOString().slice(0, 10);
for (const row of initialRows) {
if (row.isObsolete) continue;
if (!(row.autoMarkObsoleteAfterEndDate ?? true)) continue;
const endDate = row.medicationEndDate?.slice(0, 10);
if (!endDate) continue;
if (endDate > todayDate) continue;
await db
.update(medications)
.set({ isObsolete: true, obsoleteAt: new Date(), updatedAt: new Date() })
.where(and(eq(medications.id, row.id), eq(medications.userId, userId)));
}
const whereClause = includeObsolete const whereClause = includeObsolete
? eq(medications.userId, userId) ? eq(medications.userId, userId)
: and(eq(medications.userId, userId), eq(medications.isObsolete, false)); : and(eq(medications.userId, userId), eq(medications.isObsolete, false));
const rows = await db.select().from(medications).where(whereClause).orderBy(medications.id); const rows = await db.select().from(medications).where(whereClause).orderBy(medications.id);
return rows.map((row) => { return rows.map((row) => {
// Parse intakes from new format, falling back to legacy // Parse intakes from new format, falling back to legacy
const intakes = parseIntakesJson( const intakes = parseIntakesWithUnits(
row.intakesJson, row.intakesJson,
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson }, { usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
row.intakeRemindersEnabled ?? false row.intakeRemindersEnabled ?? false
@@ -144,10 +298,15 @@ export async function medicationRoutes(app: FastifyInstance) {
name: row.name, name: row.name,
genericName: row.genericName, genericName: row.genericName,
takenBy: parseTakenByJson(row.takenByJson), takenBy: parseTakenByJson(row.takenByJson),
packageType: row.packageType ?? "blister", medicationForm: row.medicationForm ?? "tablet",
pillForm: row.pillForm ?? null,
lifecycleCategory: row.lifecycleCategory ?? "refill_when_empty",
packageType: normalizePackageType(row.packageType),
packCount: row.packCount ?? 1, packCount: row.packCount ?? 1,
blistersPerPack: row.blistersPerPack ?? 1, blistersPerPack: row.blistersPerPack ?? 1,
pillsPerBlister: row.pillsPerBlister ?? 1, pillsPerBlister: row.pillsPerBlister ?? 1,
packageAmountValue: row.packageAmountValue ?? 0,
packageAmountUnit: (row.packageAmountUnit ?? "ml") as "ml" | "g",
totalPills: row.totalPills ?? null, totalPills: row.totalPills ?? null,
looseTablets: row.looseTablets ?? 0, looseTablets: row.looseTablets ?? 0,
stockAdjustment: row.stockAdjustment ?? 0, stockAdjustment: row.stockAdjustment ?? 0,
@@ -155,6 +314,8 @@ export async function medicationRoutes(app: FastifyInstance) {
pillWeightMg: row.pillWeightMg, pillWeightMg: row.pillWeightMg,
doseUnit: row.doseUnit ?? "mg", doseUnit: row.doseUnit ?? "mg",
medicationStartDate: row.medicationStartDate || null, medicationStartDate: row.medicationStartDate || null,
medicationEndDate: row.medicationEndDate || null,
autoMarkObsoleteAfterEndDate: row.autoMarkObsoleteAfterEndDate ?? true,
intakes, // New unified format with per-intake takenBy intakes, // New unified format with per-intake takenBy
// Legacy blisters format (for backward compat with frontend during transition) // Legacy blisters format (for backward compat with frontend during transition)
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })), blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
@@ -184,15 +345,22 @@ export async function medicationRoutes(app: FastifyInstance) {
name, name,
genericName, genericName,
takenBy, takenBy,
medicationForm,
pillForm,
lifecycleCategory,
packageType, packageType,
packCount, packCount,
blistersPerPack, blistersPerPack,
pillsPerBlister, pillsPerBlister,
packageAmountValue,
packageAmountUnit,
totalPills, totalPills,
looseTablets, looseTablets,
pillWeightMg, pillWeightMg,
doseUnit, doseUnit,
medicationStartDate, medicationStartDate,
medicationEndDate,
autoMarkObsoleteAfterEndDate,
expiryDate, expiryDate,
notes, notes,
prescriptionEnabled, prescriptionEnabled,
@@ -205,6 +373,9 @@ export async function medicationRoutes(app: FastifyInstance) {
blisters: inputBlisters, blisters: inputBlisters,
} = parsed.data; } = parsed.data;
const normalizedPillForm =
medicationForm === "capsule" || medicationForm === "tablet" ? (pillForm ?? medicationForm) : null;
// Convert to unified intakes format // Convert to unified intakes format
let intakes: Intake[]; let intakes: Intake[];
if (inputIntakes) { if (inputIntakes) {
@@ -213,6 +384,7 @@ export async function medicationRoutes(app: FastifyInstance) {
usage: i.usage, usage: i.usage,
every: i.every, every: i.every,
start: i.start, start: i.start,
intakeUnit: i.intakeUnit ?? null,
takenBy: i.takenBy || null, takenBy: i.takenBy || null,
intakeRemindersEnabled: i.intakeRemindersEnabled ?? false, intakeRemindersEnabled: i.intakeRemindersEnabled ?? false,
})); }));
@@ -222,6 +394,7 @@ export async function medicationRoutes(app: FastifyInstance) {
usage: b.usage, usage: b.usage,
every: b.every, every: b.every,
start: b.start, start: b.start,
intakeUnit: null,
takenBy: null, // No per-intake takenBy from legacy takenBy: null, // No per-intake takenBy from legacy
intakeRemindersEnabled: intakeRemindersEnabled ?? false, intakeRemindersEnabled: intakeRemindersEnabled ?? false,
})); }));
@@ -243,15 +416,22 @@ export async function medicationRoutes(app: FastifyInstance) {
name, name,
genericName: genericName || null, genericName: genericName || null,
takenByJson, takenByJson,
packageType: packageType ?? "blister", medicationForm: medicationForm ?? "tablet",
pillForm: normalizedPillForm,
lifecycleCategory: lifecycleCategory ?? "refill_when_empty",
packageType: normalizePackageType(packageType),
packCount, packCount,
blistersPerPack, blistersPerPack,
pillsPerBlister, pillsPerBlister,
packageAmountValue,
packageAmountUnit,
totalPills: totalPills || null, totalPills: totalPills || null,
looseTablets, looseTablets,
pillWeightMg: pillWeightMg || null, pillWeightMg: pillWeightMg || null,
doseUnit: doseUnit ?? "mg", doseUnit: doseUnit ?? "mg",
medicationStartDate: medicationStartDate ?? "", medicationStartDate: medicationStartDate ?? "",
medicationEndDate: medicationEndDate || null,
autoMarkObsoleteAfterEndDate: autoMarkObsoleteAfterEndDate ?? true,
expiryDate: expiryDate || null, expiryDate: expiryDate || null,
notes: notes || null, notes: notes || null,
prescriptionEnabled: prescriptionEnabled ?? false, prescriptionEnabled: prescriptionEnabled ?? false,
@@ -272,10 +452,15 @@ export async function medicationRoutes(app: FastifyInstance) {
name: inserted.name, name: inserted.name,
genericName: inserted.genericName, genericName: inserted.genericName,
takenBy: parseTakenByJson(inserted.takenByJson), takenBy: parseTakenByJson(inserted.takenByJson),
packageType: inserted.packageType ?? "blister", medicationForm: inserted.medicationForm ?? "tablet",
pillForm: inserted.pillForm ?? null,
lifecycleCategory: inserted.lifecycleCategory ?? "refill_when_empty",
packageType: normalizePackageType(inserted.packageType),
packCount: inserted.packCount, packCount: inserted.packCount,
blistersPerPack: inserted.blistersPerPack, blistersPerPack: inserted.blistersPerPack,
pillsPerBlister: inserted.pillsPerBlister, pillsPerBlister: inserted.pillsPerBlister,
packageAmountValue: inserted.packageAmountValue ?? 0,
packageAmountUnit: (inserted.packageAmountUnit ?? "ml") as "ml" | "g",
totalPills: inserted.totalPills ?? null, totalPills: inserted.totalPills ?? null,
looseTablets: inserted.looseTablets, looseTablets: inserted.looseTablets,
stockAdjustment: inserted.stockAdjustment ?? 0, stockAdjustment: inserted.stockAdjustment ?? 0,
@@ -283,6 +468,8 @@ export async function medicationRoutes(app: FastifyInstance) {
pillWeightMg: inserted.pillWeightMg, pillWeightMg: inserted.pillWeightMg,
doseUnit: inserted.doseUnit ?? "mg", doseUnit: inserted.doseUnit ?? "mg",
medicationStartDate: inserted.medicationStartDate || null, medicationStartDate: inserted.medicationStartDate || null,
medicationEndDate: inserted.medicationEndDate || null,
autoMarkObsoleteAfterEndDate: inserted.autoMarkObsoleteAfterEndDate ?? true,
intakes, intakes,
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })), blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
imageUrl: inserted.imageUrl, imageUrl: inserted.imageUrl,
@@ -319,15 +506,22 @@ export async function medicationRoutes(app: FastifyInstance) {
name, name,
genericName, genericName,
takenBy, takenBy,
medicationForm,
pillForm,
lifecycleCategory,
packageType, packageType,
packCount, packCount,
blistersPerPack, blistersPerPack,
pillsPerBlister, pillsPerBlister,
packageAmountValue,
packageAmountUnit,
totalPills, totalPills,
looseTablets, looseTablets,
pillWeightMg, pillWeightMg,
doseUnit, doseUnit,
medicationStartDate, medicationStartDate,
medicationEndDate,
autoMarkObsoleteAfterEndDate,
expiryDate, expiryDate,
notes, notes,
prescriptionEnabled, prescriptionEnabled,
@@ -340,6 +534,9 @@ export async function medicationRoutes(app: FastifyInstance) {
blisters: inputBlisters, blisters: inputBlisters,
} = parsed.data; } = parsed.data;
const normalizedPillForm =
medicationForm === "capsule" || medicationForm === "tablet" ? (pillForm ?? medicationForm) : null;
// Convert to unified intakes format // Convert to unified intakes format
let intakes: Intake[]; let intakes: Intake[];
if (inputIntakes) { if (inputIntakes) {
@@ -348,6 +545,7 @@ export async function medicationRoutes(app: FastifyInstance) {
usage: i.usage, usage: i.usage,
every: i.every, every: i.every,
start: i.start, start: i.start,
intakeUnit: i.intakeUnit ?? null,
takenBy: i.takenBy || null, takenBy: i.takenBy || null,
intakeRemindersEnabled: i.intakeRemindersEnabled ?? false, intakeRemindersEnabled: i.intakeRemindersEnabled ?? false,
})); }));
@@ -357,6 +555,7 @@ export async function medicationRoutes(app: FastifyInstance) {
usage: b.usage, usage: b.usage,
every: b.every, every: b.every,
start: b.start, start: b.start,
intakeUnit: null,
takenBy: null, // No per-intake takenBy from legacy takenBy: null, // No per-intake takenBy from legacy
intakeRemindersEnabled: intakeRemindersEnabled ?? false, intakeRemindersEnabled: intakeRemindersEnabled ?? false,
})); }));
@@ -388,15 +587,22 @@ export async function medicationRoutes(app: FastifyInstance) {
name, name,
genericName: genericName || null, genericName: genericName || null,
takenByJson, takenByJson,
packageType: packageType ?? "blister", medicationForm: medicationForm ?? "tablet",
pillForm: normalizedPillForm,
lifecycleCategory: lifecycleCategory ?? "refill_when_empty",
packageType: normalizePackageType(packageType),
packCount, packCount,
blistersPerPack, blistersPerPack,
pillsPerBlister, pillsPerBlister,
totalPills: totalPills || null, totalPills: totalPills || null,
packageAmountValue,
packageAmountUnit,
looseTablets, looseTablets,
pillWeightMg: pillWeightMg || null, pillWeightMg: pillWeightMg || null,
doseUnit: doseUnit ?? "mg", doseUnit: doseUnit ?? "mg",
medicationStartDate: medicationStartDate ?? "", medicationStartDate: medicationStartDate ?? "",
medicationEndDate: medicationEndDate || null,
autoMarkObsoleteAfterEndDate: autoMarkObsoleteAfterEndDate ?? true,
expiryDate: expiryDate || null, expiryDate: expiryDate || null,
notes: notes || null, notes: notes || null,
prescriptionEnabled: prescriptionEnabled ?? false, prescriptionEnabled: prescriptionEnabled ?? false,
@@ -421,7 +627,7 @@ export async function medicationRoutes(app: FastifyInstance) {
// Migrate dose tracking IDs when intake schedule changes // Migrate dose tracking IDs when intake schedule changes
// --------------------------------------------------------------- // ---------------------------------------------------------------
// Parse old intakes from the existing medication row // Parse old intakes from the existing medication row
const oldIntakes = parseIntakesJson( const oldIntakes = parseIntakesWithUnits(
existing.intakesJson, existing.intakesJson,
{ usageJson: existing.usageJson, everyJson: existing.everyJson, startJson: existing.startJson }, { usageJson: existing.usageJson, everyJson: existing.everyJson, startJson: existing.startJson },
existing.intakeRemindersEnabled existing.intakeRemindersEnabled
@@ -541,10 +747,15 @@ export async function medicationRoutes(app: FastifyInstance) {
name: result[0].name, name: result[0].name,
genericName: result[0].genericName, genericName: result[0].genericName,
takenBy: parseTakenByJson(result[0].takenByJson), takenBy: parseTakenByJson(result[0].takenByJson),
packageType: result[0].packageType ?? "blister", medicationForm: result[0].medicationForm ?? "tablet",
pillForm: result[0].pillForm ?? null,
lifecycleCategory: result[0].lifecycleCategory ?? "refill_when_empty",
packageType: normalizePackageType(result[0].packageType),
packCount: result[0].packCount, packCount: result[0].packCount,
blistersPerPack: result[0].blistersPerPack, blistersPerPack: result[0].blistersPerPack,
pillsPerBlister: result[0].pillsPerBlister, pillsPerBlister: result[0].pillsPerBlister,
packageAmountValue: result[0].packageAmountValue ?? 0,
packageAmountUnit: (result[0].packageAmountUnit ?? "ml") as "ml" | "g",
totalPills: result[0].totalPills ?? null, totalPills: result[0].totalPills ?? null,
looseTablets: result[0].looseTablets, looseTablets: result[0].looseTablets,
stockAdjustment: result[0].stockAdjustment ?? 0, stockAdjustment: result[0].stockAdjustment ?? 0,
@@ -552,6 +763,8 @@ export async function medicationRoutes(app: FastifyInstance) {
pillWeightMg: result[0].pillWeightMg, pillWeightMg: result[0].pillWeightMg,
doseUnit: result[0].doseUnit ?? "mg", doseUnit: result[0].doseUnit ?? "mg",
medicationStartDate: result[0].medicationStartDate || null, medicationStartDate: result[0].medicationStartDate || null,
medicationEndDate: result[0].medicationEndDate || null,
autoMarkObsoleteAfterEndDate: result[0].autoMarkObsoleteAfterEndDate ?? true,
intakes, intakes,
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })), blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
imageUrl: result[0].imageUrl, imageUrl: result[0].imageUrl,
@@ -627,62 +840,101 @@ export async function medicationRoutes(app: FastifyInstance) {
}; };
}); });
// Stock correction endpoint - updates stockAdjustment and optionally looseTablets (for blister type) // Stock correction endpoint - updates stockAdjustment and optionally base amount fields for amount-based corrections
// Also sets lastStockCorrectionAt so consumed doses before this point don't count // Also sets lastStockCorrectionAt so consumed doses before this point don't count
app.patch<{ Params: { id: string }; Body: { stockAdjustment: number; looseTablets?: number } }>( app.patch<{
"/medications/:id/stock-adjustment", Params: { id: string };
async (req, reply) => { Body: {
const idNum = Number(req.params.id); stockAdjustment: number;
if (Number.isNaN(idNum)) return reply.badRequest("Invalid id"); looseTablets?: number;
totalPills?: number;
packageAmountValue?: number;
packCount?: number;
};
}>("/medications/:id/stock-adjustment", async (req, reply) => {
const idNum = Number(req.params.id);
if (Number.isNaN(idNum)) return reply.badRequest("Invalid id");
const userId = await getUserId(req, reply); const userId = await getUserId(req, reply);
// Verify ownership // Verify ownership
const [existing] = await db const [existing] = await db
.select() .select()
.from(medications) .from(medications)
.where(and(eq(medications.id, idNum), eq(medications.userId, userId))); .where(and(eq(medications.id, idNum), eq(medications.userId, userId)));
if (!existing) return reply.notFound(); if (!existing) return reply.notFound();
const { stockAdjustment, looseTablets } = req.body as { stockAdjustment: number; looseTablets?: number }; const { stockAdjustment, looseTablets, totalPills, packageAmountValue, packCount } = req.body as {
if (typeof stockAdjustment !== "number") return reply.badRequest("stockAdjustment must be a number"); stockAdjustment: number;
if ( looseTablets?: number;
looseTablets !== undefined && totalPills?: number;
(typeof looseTablets !== "number" || !Number.isInteger(looseTablets) || looseTablets < 0) packageAmountValue?: number;
) { packCount?: number;
return reply.badRequest("looseTablets must be a non-negative integer"); };
} if (typeof stockAdjustment !== "number") return reply.badRequest("stockAdjustment must be a number");
if (
const updateFields: { looseTablets !== undefined &&
stockAdjustment: number; (typeof looseTablets !== "number" || !Number.isInteger(looseTablets) || looseTablets < 0)
lastStockCorrectionAt: Date; ) {
updatedAt: Date; return reply.badRequest("looseTablets must be a non-negative integer");
looseTablets?: number;
} = {
stockAdjustment,
lastStockCorrectionAt: new Date(),
updatedAt: new Date(),
};
if (looseTablets !== undefined) {
updateFields.looseTablets = looseTablets;
}
const result = await db
.update(medications)
.set(updateFields)
.where(and(eq(medications.id, idNum), eq(medications.userId, userId)))
.returning();
if (!result.length) return reply.notFound();
return {
id: result[0].id,
stockAdjustment: result[0].stockAdjustment ?? 0,
lastStockCorrectionAt: result[0].lastStockCorrectionAt?.toISOString() ?? null,
updatedAt: result[0].updatedAt,
};
} }
); if (
totalPills !== undefined &&
(typeof totalPills !== "number" || !Number.isInteger(totalPills) || totalPills < 0)
) {
return reply.badRequest("totalPills must be a non-negative integer");
}
if (
packageAmountValue !== undefined &&
(typeof packageAmountValue !== "number" || !Number.isInteger(packageAmountValue) || packageAmountValue < 0)
) {
return reply.badRequest("packageAmountValue must be a non-negative integer");
}
if (packCount !== undefined && (typeof packCount !== "number" || !Number.isInteger(packCount) || packCount < 1)) {
return reply.badRequest("packCount must be an integer >= 1");
}
const updateFields: {
stockAdjustment: number;
lastStockCorrectionAt: Date;
updatedAt: Date;
looseTablets?: number;
totalPills?: number | null;
packageAmountValue?: number;
packCount?: number;
} = {
stockAdjustment,
lastStockCorrectionAt: new Date(),
updatedAt: new Date(),
};
const packageType = normalizePackageType(existing.packageType);
const allowsAmountBaseUpdate = isTubePackageType(packageType) || isLiquidContainerPackageType(packageType);
if (allowsAmountBaseUpdate) {
if (totalPills !== undefined) updateFields.totalPills = totalPills;
if (looseTablets !== undefined) updateFields.looseTablets = looseTablets;
if (packageAmountValue !== undefined) updateFields.packageAmountValue = packageAmountValue;
if (packCount !== undefined) updateFields.packCount = packCount;
}
if (looseTablets !== undefined) {
updateFields.looseTablets = looseTablets;
}
const result = await db
.update(medications)
.set(updateFields)
.where(and(eq(medications.id, idNum), eq(medications.userId, userId)))
.returning();
if (!result.length) return reply.notFound();
return {
id: result[0].id,
stockAdjustment: result[0].stockAdjustment ?? 0,
lastStockCorrectionAt: result[0].lastStockCorrectionAt?.toISOString() ?? null,
updatedAt: result[0].updatedAt,
};
});
app.delete<{ Params: { id: string } }>("/medications/:id", async (req, reply) => { app.delete<{ Params: { id: string } }>("/medications/:id", async (req, reply) => {
const idNum = Number(req.params.id); const idNum = Number(req.params.id);
@@ -836,24 +1088,29 @@ export async function medicationRoutes(app: FastifyInstance) {
const payload = rows.map((row) => { const payload = rows.map((row) => {
// Parse intakes from new format, falling back to legacy // Parse intakes from new format, falling back to legacy
const intakes = parseIntakesJson( const intakes = parseIntakesWithUnits(
row.intakesJson, row.intakesJson,
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson }, { usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
row.intakeRemindersEnabled ?? false row.intakeRemindersEnabled ?? false
); );
const blisters = intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })); const medForm = row.medicationForm ?? "tablet";
const blisters = intakes.map((i) => ({
usage: normalizeIntakeUsageForStock(i, medForm, row.packageType),
every: i.every,
start: i.start,
}));
const pillsPerBlister = row.pillsPerBlister ?? 1; const pillsPerBlister = row.pillsPerBlister ?? 1;
const packCount = row.packCount ?? 1; const packCount = row.packCount ?? 1;
const blistersPerPack = row.blistersPerPack ?? 1; const blistersPerPack = row.blistersPerPack ?? 1;
const looseTablets = row.looseTablets ?? 0; const looseTablets = row.looseTablets ?? 0;
const stockAdjustment = row.stockAdjustment ?? 0; const stockAdjustment = row.stockAdjustment ?? 0;
const packageType = row.packageType ?? "blister"; const packageType = normalizePackageType(row.packageType);
// For bottle type, looseTablets IS the current stock (no blister math) // For bottle type, looseTablets IS the current stock (no blister math)
const originalTotalPills = const isTopical = medForm === "topical" || isTubePackageType(packageType);
packageType === "bottle" const originalTotalPills = isAmountBasedPackageType(packageType)
? looseTablets + stockAdjustment ? looseTablets + stockAdjustment
: packCount * blistersPerPack * pillsPerBlister + looseTablets + stockAdjustment; : packCount * blistersPerPack * pillsPerBlister + looseTablets + stockAdjustment;
// Calculate consumption with the same automatic/manual behavior as frontend coverage. // Calculate consumption with the same automatic/manual behavior as frontend coverage.
const stockCorrectionCutoff = row.lastStockCorrectionAt ? new Date(row.lastStockCorrectionAt).getTime() : 0; const stockCorrectionCutoff = row.lastStockCorrectionAt ? new Date(row.lastStockCorrectionAt).getTime() : 0;
@@ -863,7 +1120,9 @@ export async function medicationRoutes(app: FastifyInstance) {
let consumedUntilNow = 0; let consumedUntilNow = 0;
const msPerDay = 86400000; const msPerDay = 86400000;
if (stockCalculationMode === "automatic") { if (isTopical) {
consumedUntilNow = 0;
} else if (stockCalculationMode === "automatic") {
blisters.forEach((blister, blisterIdx) => { blisters.forEach((blister, blisterIdx) => {
const blisterStart = parseLocalDateTime(blister.start).getTime(); const blisterStart = parseLocalDateTime(blister.start).getTime();
if (Number.isNaN(blisterStart)) return; if (Number.isNaN(blisterStart)) return;
@@ -959,7 +1218,7 @@ export async function medicationRoutes(app: FastifyInstance) {
}); });
} }
const currentStock = Math.max(0, originalTotalPills - consumedUntilNow); const currentStock = isTopical ? originalTotalPills : Math.max(0, originalTotalPills - consumedUntilNow);
// Calculate usage for the planning period // Calculate usage for the planning period
// Always use the user-selected start date for the usage calculation. // Always use the user-selected start date for the usage calculation.
@@ -969,7 +1228,7 @@ export async function medicationRoutes(app: FastifyInstance) {
// The stock already reflects consumed doses, so no double-counting occurs. // The stock already reflects consumed doses, so no double-counting occurs.
// When includeUntilStart is true, calculate from now to end (useful for trip planning) // When includeUntilStart is true, calculate from now to end (useful for trip planning)
const effectivePlannerStart = includeUntilStart ? now : start; const effectivePlannerStart = includeUntilStart ? now : start;
const usageTotal = calculateUsageInRange(blisters, effectivePlannerStart, end); const usageTotal = isTopical ? 0 : calculateUsageInRange(blisters, effectivePlannerStart, end);
const blistersNeeded = pillsPerBlister > 0 ? Math.ceil(usageTotal / pillsPerBlister) : 0; const blistersNeeded = pillsPerBlister > 0 ? Math.ceil(usageTotal / pillsPerBlister) : 0;
@@ -979,7 +1238,7 @@ export async function medicationRoutes(app: FastifyInstance) {
let fullBlisters: number; let fullBlisters: number;
let loosePills: number; let loosePills: number;
if (packageType === "bottle") { if (isAmountBasedPackageType(packageType)) {
// Bottle type: no blisters, everything is loose pills // Bottle type: no blisters, everything is loose pills
fullBlisters = 0; fullBlisters = 0;
loosePills = availableAfterPeriod; loosePills = availableAfterPeriod;
+234 -15
View File
@@ -15,6 +15,12 @@ import { getAnonymousUserId, requireAuth } from "../plugins/auth.js";
import { env } from "../plugins/env.js"; import { env } from "../plugins/env.js";
import { updateReminderSentTime, updateUserReminderSentTime } from "../services/reminder-scheduler.js"; import { updateReminderSentTime, updateUserReminderSentTime } from "../services/reminder-scheduler.js";
import type { AuthUser } from "../types/fastify.js"; import type { AuthUser } from "../types/fastify.js";
import {
getPlannerUnitKind,
isAmountBasedPackageType,
isTubePackageType,
normalizePackageType,
} from "../utils/package-profiles.js";
import { loadUserSettings, sendShoutrrrNotification } from "./settings.js"; import { loadUserSettings, sendShoutrrrNotification } from "./settings.js";
// Escape HTML to prevent XSS in email templates // Escape HTML to prevent XSS in email templates
@@ -29,6 +35,43 @@ function escapeHtml(text: string): string {
return text.replace(/[&<>"']/g, (char) => htmlEscapes[char] || char); return text.replace(/[&<>"']/g, (char) => htmlEscapes[char] || char);
} }
function maskEmail(email: string): string {
const [localPart, domain] = email.split("@");
if (!domain) return "invalid-email";
if (localPart.length <= 2) return `${localPart[0] ?? "*"}*@${domain}`;
return `${localPart.slice(0, 2)}***@${domain}`;
}
type MailDeliveryInfo = {
accepted?: unknown;
rejected?: unknown;
response?: unknown;
};
function normalizeRecipients(value: unknown): string[] {
if (!Array.isArray(value)) return [];
return value
.map((entry) => (typeof entry === "string" ? entry : String(entry ?? "")))
.map((entry) => entry.trim())
.filter(Boolean);
}
function getDeliveryError(info: MailDeliveryInfo): string | null {
const accepted = normalizeRecipients(info.accepted);
const rejected = normalizeRecipients(info.rejected);
if (accepted.length > 0) return null;
if (rejected.length > 0) {
return `SMTP rejected all recipients: ${rejected.join(", ")}`;
}
if (typeof info.response === "string" && info.response.trim()) {
return `SMTP did not confirm accepted recipients. Response: ${info.response}`;
}
return "SMTP did not confirm accepted recipients.";
}
type PlannerRow = { type PlannerRow = {
medicationId: number; medicationId: number;
medicationName: string; medicationName: string;
@@ -42,6 +85,17 @@ type PlannerRow = {
packageType?: string; packageType?: string;
}; };
function isContainerPackage(packageType?: string): boolean {
return isAmountBasedPackageType(packageType);
}
function getPlannerUnit(packageType: string | undefined, tr: ReturnType<typeof getTranslations>): string {
const unitKind = getPlannerUnitKind(packageType);
if (unitKind === "units") return tr.common.units;
if (unitKind === "ml") return tr.common.ml;
return tr.common.pills;
}
type SendEmailBody = { type SendEmailBody = {
email: string; email: string;
from: string; from: string;
@@ -96,6 +150,10 @@ export async function plannerRoutes(app: FastifyInstance) {
// Demand calculator notification (supports email and push) // Demand calculator notification (supports email and push)
app.post<{ Body: SendEmailBody }>("/planner/send-email", async (request, reply) => { app.post<{ Body: SendEmailBody }>("/planner/send-email", async (request, reply) => {
const { email, from, until, rows, language: bodyLanguage } = request.body; const { email, from, until, rows, language: bodyLanguage } = request.body;
request.log.info(
{ hasEmail: Boolean(email), rowCount: rows?.length ?? 0 },
"[Planner] Demand notification request received"
);
if (!rows || rows.length === 0) { if (!rows || rows.length === 0) {
return reply.status(400).send({ error: "Missing planner data" }); return reply.status(400).send({ error: "Missing planner data" });
@@ -110,6 +168,7 @@ export async function plannerRoutes(app: FastifyInstance) {
const activeMedIds = new Set(activeMeds.map((med) => med.id)); const activeMedIds = new Set(activeMeds.map((med) => med.id));
const activeRows = rows.filter((row) => activeMedIds.has(row.medicationId)); const activeRows = rows.filter((row) => activeMedIds.has(row.medicationId));
if (activeRows.length === 0) { if (activeRows.length === 0) {
request.log.warn("[Planner] Demand notification skipped: no active medications in request");
return reply.status(400).send({ error: "No active medications to notify" }); return reply.status(400).send({ error: "No active medications to notify" });
} }
@@ -119,6 +178,16 @@ export async function plannerRoutes(app: FastifyInstance) {
shoutrrrEnabled: userSettings.shoutrrrEnabled, shoutrrrEnabled: userSettings.shoutrrrEnabled,
shoutrrrUrl: userSettings.shoutrrrUrl || "", shoutrrrUrl: userSettings.shoutrrrUrl || "",
}; };
request.log.info(
{
userId,
emailEnabled: notificationSettings.emailEnabled,
pushEnabled: notificationSettings.shoutrrrEnabled,
hasPushUrl: Boolean(notificationSettings.shoutrrrUrl),
activeRowCount: activeRows.length,
},
"[Planner] Demand notification channel state"
);
// Get locale from user settings or use the language passed in the body // Get locale from user settings or use the language passed in the body
const language: Language = (userSettings.language as Language) || bodyLanguage || "en"; const language: Language = (userSettings.language as Language) || bodyLanguage || "en";
@@ -168,16 +237,18 @@ ${summaryText}
${activeRows ${activeRows
.map((r) => { .map((r) => {
const isBottle = r.packageType === "bottle"; const isBottle = isContainerPackage(r.packageType);
const usage = `${r.plannerUsage} ${tr.common.pills}`; const usageUnit = getPlannerUnit(r.packageType, tr);
const usage = `${r.plannerUsage} ${usageUnit}`;
const needed = isBottle ? "" : `${r.blistersNeeded} × ${r.blisterSize}`; const needed = isBottle ? "" : `${r.blistersNeeded} × ${r.blisterSize}`;
const medPrescription = prescriptionMap.get(r.medicationId); const medPrescription = prescriptionMap.get(r.medicationId);
const rxRefills = medPrescription?.prescriptionEnabled const rxRefills = medPrescription?.prescriptionEnabled
? String(medPrescription.prescriptionRemainingRefills ?? 0) ? String(medPrescription.prescriptionRemainingRefills ?? 0)
: dc.prescriptionNotApplicable; : dc.prescriptionNotApplicable;
const loosePills = Math.round((Number(r.loosePills) || 0) * 10) / 10; const loosePills = Math.round((Number(r.loosePills) || 0) * 10) / 10;
const availableUnit = getPlannerUnit(r.packageType, tr);
const available = isBottle const available = isBottle
? `${loosePills} ${tr.common.pills}` ? `${loosePills} ${availableUnit}`
: `${r.fullBlisters} ${tr.common.blisters}${loosePills > 0 ? ` + ${loosePills} ${tr.common.pills}` : ""}`; : `${r.fullBlisters} ${tr.common.blisters}${loosePills > 0 ? ` + ${loosePills} ${tr.common.pills}` : ""}`;
const status = r.enough ? dc.statusEnough : dc.statusEmpty; const status = r.enough ? dc.statusEnough : dc.statusEmpty;
return `${r.medicationName}: ${usage}, ${needed}, ${dc.tableHeaders.prescriptionRefills}: ${rxRefills}, ${available} - ${status}`; return `${r.medicationName}: ${usage}, ${needed}, ${dc.tableHeaders.prescriptionRefills}: ${rxRefills}, ${available} - ${status}`;
@@ -198,6 +269,19 @@ ${getFooterPlain(language)}`;
const smtpSecure = process.env.SMTP_SECURE === "true"; const smtpSecure = process.env.SMTP_SECURE === "true";
const smtpFrom = process.env.SMTP_FROM ?? smtpUser; const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
request.log.info(
{
hasSmtpHost: Boolean(smtpHost),
hasSmtpUser: Boolean(smtpUser),
hasSmtpPass: Boolean(smtpPass),
smtpPort,
smtpSecure,
hasSmtpFrom: Boolean(smtpFrom),
to: maskEmail(email),
},
"[Planner] Demand email path selected"
);
if (smtpHost && smtpUser) { if (smtpHost && smtpUser) {
// Build HTML table with horizontal scroll for mobile // Build HTML table with horizontal scroll for mobile
// Escape/coerce all user-provided values to prevent XSS // Escape/coerce all user-provided values to prevent XSS
@@ -209,7 +293,7 @@ ${getFooterPlain(language)}`;
const safeBlisterSize = Number(row.blisterSize) || 0; const safeBlisterSize = Number(row.blisterSize) || 0;
const safeFullBlisters = Number(row.fullBlisters) || 0; const safeFullBlisters = Number(row.fullBlisters) || 0;
const safeLoosePills = Math.round((Number(row.loosePills) || 0) * 10) / 10; const safeLoosePills = Math.round((Number(row.loosePills) || 0) * 10) / 10;
const isBottle = row.packageType === "bottle"; const isBottle = isContainerPackage(row.packageType);
// "Blisters needed" column: dash for bottles // "Blisters needed" column: dash for bottles
const neededCell = isBottle ? "" : `${safeBlistersNeeded} × ${safeBlisterSize}`; const neededCell = isBottle ? "" : `${safeBlistersNeeded} × ${safeBlisterSize}`;
@@ -223,7 +307,8 @@ ${getFooterPlain(language)}`;
// "Available" column: match frontend format // "Available" column: match frontend format
let availableCell: string; let availableCell: string;
if (isBottle) { if (isBottle) {
availableCell = `${safeLoosePills} ${tr.common.pills}`; const availableUnit = getPlannerUnit(row.packageType, tr);
availableCell = `${safeLoosePills} ${availableUnit}`;
} else { } else {
availableCell = `${safeFullBlisters} ${tr.common.blisters}`; availableCell = `${safeFullBlisters} ${tr.common.blisters}`;
if (safeLoosePills > 0) { if (safeLoosePills > 0) {
@@ -236,7 +321,7 @@ ${getFooterPlain(language)}`;
return ` return `
<tr style="${rowBg}"> <tr style="${rowBg}">
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; white-space: nowrap;">${safeName}</td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; white-space: nowrap;">${safeName}</td>
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;"><strong>${safePlannerUsage}</strong> ${tr.common.pills}</td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;"><strong>${safePlannerUsage}</strong> ${getPlannerUnit(row.packageType, tr)}</td>
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${neededCell}</td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${neededCell}</td>
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${rxCell}</td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${rxCell}</td>
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${availableCell}</td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${availableCell}</td>
@@ -303,7 +388,9 @@ ${getFooterPlain(language)}`;
}, },
}); });
await transporter.sendMail({ request.log.info({ to: maskEmail(email) }, "[Planner] Sending demand email");
const mailResult = await transporter.sendMail({
from: smtpFrom, from: smtpFrom,
to: email, to: email,
subject: t(dc.subject, { from: fromDate, until: untilDate }), subject: t(dc.subject, { from: fromDate, until: untilDate }),
@@ -311,12 +398,33 @@ ${getFooterPlain(language)}`;
html, html,
}); });
const deliveryError = getDeliveryError(mailResult);
if (deliveryError) {
throw new Error(deliveryError);
}
request.log.info({ to: maskEmail(email), messageId: mailResult.messageId }, "[Planner] Demand email sent");
results.email = true; results.email = true;
} catch (error) { } catch (error) {
request.log.error({ error, to: maskEmail(email) }, "[Planner] Demand email failed");
const errorMessage = error instanceof Error ? error.message : "Unknown error"; const errorMessage = error instanceof Error ? error.message : "Unknown error";
results.errors.push(`Email: ${errorMessage}`); results.errors.push(`Email: ${errorMessage}`);
} }
} else {
request.log.warn(
{
hasSmtpHost: Boolean(smtpHost),
hasSmtpUser: Boolean(smtpUser),
to: maskEmail(email),
},
"[Planner] Demand email skipped: SMTP not configured"
);
} }
} else {
request.log.info(
{ emailEnabled: notificationSettings.emailEnabled, hasRecipient: Boolean(email) },
"[Planner] Demand email channel not active"
);
} }
// Send push notification if enabled // Send push notification if enabled
@@ -324,7 +432,7 @@ ${getFooterPlain(language)}`;
const pushTitle = t(dc.subject, { from: fromDate, until: untilDate }); const pushTitle = t(dc.subject, { from: fromDate, until: untilDate });
const pushMessage = `${summaryText}\n\n${activeRows const pushMessage = `${summaryText}\n\n${activeRows
.map((r) => { .map((r) => {
const usage = `${r.plannerUsage} ${tr.common.pills}`; const usage = `${r.plannerUsage} ${getPlannerUnit(r.packageType, tr)}`;
const status = r.enough ? dc.statusEnough : dc.statusEmpty; const status = r.enough ? dc.statusEnough : dc.statusEmpty;
return `${r.enough ? "✓" : "✗"} ${r.medicationName}: ${usage} - ${status}`; return `${r.enough ? "✓" : "✗"} ${r.medicationName}: ${usage} - ${status}`;
}) })
@@ -363,6 +471,10 @@ ${getFooterPlain(language)}`;
// Reminder notification for low stock medications (supports email and push) // Reminder notification for low stock medications (supports email and push)
app.post<{ Body: ReminderEmailBody }>("/reminder/send-email", async (request, reply) => { app.post<{ Body: ReminderEmailBody }>("/reminder/send-email", async (request, reply) => {
const { email, lowStock } = request.body; const { email, lowStock } = request.body;
request.log.info(
{ hasEmail: Boolean(email), lowStockCount: lowStock?.length ?? 0 },
"[ReminderManual] Stock reminder request received"
);
if (!lowStock || lowStock.length === 0) { if (!lowStock || lowStock.length === 0) {
return reply.status(400).send({ error: "Missing low stock data" }); return reply.status(400).send({ error: "Missing low stock data" });
@@ -371,12 +483,22 @@ ${getFooterPlain(language)}`;
// Load user settings // Load user settings
const userId = await getUserId(request); const userId = await getUserId(request);
const activeMeds = await db const activeMeds = await db
.select({ name: medications.name }) .select({ name: medications.name, genericName: medications.genericName, packageType: medications.packageType })
.from(medications) .from(medications)
.where(and(eq(medications.userId, userId), eq(medications.isObsolete, false))); .where(and(eq(medications.userId, userId), eq(medications.isObsolete, false)));
const activeMedNames = new Set(activeMeds.map((med) => med.name)); const activeMedicationByName = new Map(
const filteredLowStock = lowStock.filter((item) => activeMedNames.has(item.name)); activeMeds
.map((med) => [med.name || med.genericName || "", normalizePackageType(med.packageType)] as const)
.filter(([name]) => name.length > 0)
);
const filteredLowStock = lowStock.filter((item) => {
const packageType = activeMedicationByName.get(item.name);
if (!packageType) return false;
if (isTubePackageType(packageType)) return false;
return true;
});
if (filteredLowStock.length === 0) { if (filteredLowStock.length === 0) {
request.log.warn("[ReminderManual] Stock reminder skipped: no active medications after filtering");
return reply.status(400).send({ error: "No active medications to notify" }); return reply.status(400).send({ error: "No active medications to notify" });
} }
@@ -386,6 +508,16 @@ ${getFooterPlain(language)}`;
shoutrrrEnabled: userSettings.shoutrrrEnabled, shoutrrrEnabled: userSettings.shoutrrrEnabled,
shoutrrrUrl: userSettings.shoutrrrUrl || "", shoutrrrUrl: userSettings.shoutrrrUrl || "",
}; };
request.log.info(
{
userId,
emailEnabled: notificationSettings.emailEnabled,
pushEnabled: notificationSettings.shoutrrrEnabled,
hasPushUrl: Boolean(notificationSettings.shoutrrrUrl),
filteredLowStockCount: filteredLowStock.length,
},
"[ReminderManual] Stock reminder channel state"
);
// Get translations based on user language // Get translations based on user language
const language = (userSettings.language as Language) || "en"; const language = (userSettings.language as Language) || "en";
@@ -457,6 +589,19 @@ ${getFooterPlain(language)}`;
const smtpSecure = process.env.SMTP_SECURE === "true"; const smtpSecure = process.env.SMTP_SECURE === "true";
const smtpFrom = process.env.SMTP_FROM ?? smtpUser; const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
request.log.info(
{
hasSmtpHost: Boolean(smtpHost),
hasSmtpUser: Boolean(smtpUser),
hasSmtpPass: Boolean(smtpPass),
smtpPort,
smtpSecure,
hasSmtpFrom: Boolean(smtpFrom),
to: maskEmail(email),
},
"[ReminderManual] Stock email path selected"
);
if (smtpHost && smtpUser) { if (smtpHost && smtpUser) {
// Build subject line from shared title parts // Build subject line from shared title parts
const subjectText = titleParts.join(", "); const subjectText = titleParts.join(", ");
@@ -570,7 +715,9 @@ ${getFooterPlain(language)}`;
}, },
}); });
await transporter.sendMail({ request.log.info({ to: maskEmail(email) }, "[ReminderManual] Sending stock reminder email");
const mailResult = await transporter.sendMail({
from: smtpFrom, from: smtpFrom,
to: email, to: email,
subject: `MedAssist-ng: ${subjectText}`, subject: `MedAssist-ng: ${subjectText}`,
@@ -578,12 +725,36 @@ ${getFooterPlain(language)}`;
html, html,
}); });
const deliveryError = getDeliveryError(mailResult);
if (deliveryError) {
throw new Error(deliveryError);
}
request.log.info(
{ to: maskEmail(email), messageId: mailResult.messageId },
"[ReminderManual] Stock reminder email sent"
);
results.email = true; results.email = true;
} catch (error) { } catch (error) {
request.log.error({ error, to: maskEmail(email) }, "[ReminderManual] Stock reminder email failed");
const errorMessage = error instanceof Error ? error.message : "Unknown error"; const errorMessage = error instanceof Error ? error.message : "Unknown error";
results.errors.push(`Email: ${errorMessage}`); results.errors.push(`Email: ${errorMessage}`);
} }
} else {
request.log.warn(
{
hasSmtpHost: Boolean(smtpHost),
hasSmtpUser: Boolean(smtpUser),
to: maskEmail(email),
},
"[ReminderManual] Stock reminder email skipped: SMTP not configured"
);
} }
} else {
request.log.info(
{ emailEnabled: notificationSettings.emailEnabled, hasRecipient: Boolean(email) },
"[ReminderManual] Stock email channel not active"
);
} }
// Send push notification if enabled // Send push notification if enabled
@@ -634,6 +805,10 @@ ${getFooterPlain(language)}`;
// Manual prescription reminder (supports email and push) // Manual prescription reminder (supports email and push)
app.post<{ Body: PrescriptionReminderBody }>("/reminder/send-prescription", async (request, reply) => { app.post<{ Body: PrescriptionReminderBody }>("/reminder/send-prescription", async (request, reply) => {
const { email, prescriptionLow } = request.body; const { email, prescriptionLow } = request.body;
request.log.info(
{ hasEmail: Boolean(email), prescriptionCount: prescriptionLow?.length ?? 0 },
"[ReminderManual] Prescription reminder request received"
);
if (!prescriptionLow || prescriptionLow.length === 0) { if (!prescriptionLow || prescriptionLow.length === 0) {
return reply.status(400).send({ error: "Missing prescription reminder data" }); return reply.status(400).send({ error: "Missing prescription reminder data" });
@@ -641,12 +816,13 @@ ${getFooterPlain(language)}`;
const userId = await getUserId(request); const userId = await getUserId(request);
const activeMeds = await db const activeMeds = await db
.select({ name: medications.name }) .select({ name: medications.name, genericName: medications.genericName })
.from(medications) .from(medications)
.where(and(eq(medications.userId, userId), eq(medications.isObsolete, false))); .where(and(eq(medications.userId, userId), eq(medications.isObsolete, false)));
const activeMedNames = new Set(activeMeds.map((med) => med.name)); const activeMedNames = new Set(activeMeds.map((med) => med.name || med.genericName || ""));
const filteredPrescriptionLow = prescriptionLow.filter((item) => activeMedNames.has(item.name)); const filteredPrescriptionLow = prescriptionLow.filter((item) => activeMedNames.has(item.name));
if (filteredPrescriptionLow.length === 0) { if (filteredPrescriptionLow.length === 0) {
request.log.warn("[ReminderManual] Prescription reminder skipped: no active medications after filtering");
return reply.status(400).send({ error: "No active medications to notify" }); return reply.status(400).send({ error: "No active medications to notify" });
} }
@@ -684,6 +860,19 @@ ${getFooterPlain(language)}`;
const smtpSecure = process.env.SMTP_SECURE === "true"; const smtpSecure = process.env.SMTP_SECURE === "true";
const smtpFrom = process.env.SMTP_FROM ?? smtpUser; const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
request.log.info(
{
hasSmtpHost: Boolean(smtpHost),
hasSmtpUser: Boolean(smtpUser),
hasSmtpPass: Boolean(smtpPass),
smtpPort,
smtpSecure,
hasSmtpFrom: Boolean(smtpFrom),
to: maskEmail(email),
},
"[ReminderManual] Prescription email path selected"
);
if (smtpHost && smtpUser) { if (smtpHost && smtpUser) {
try { try {
const transporter = nodemailer.createTransport({ const transporter = nodemailer.createTransport({
@@ -767,7 +956,9 @@ ${getFooterPlain(language)}`;
</div> </div>
`; `;
await transporter.sendMail({ request.log.info({ to: maskEmail(email) }, "[ReminderManual] Sending prescription reminder email");
const mailResult = await transporter.sendMail({
from: smtpFrom, from: smtpFrom,
to: email, to: email,
subject, subject,
@@ -775,12 +966,40 @@ ${getFooterPlain(language)}`;
html, html,
}); });
const deliveryError = getDeliveryError(mailResult);
if (deliveryError) {
throw new Error(deliveryError);
}
request.log.info(
{ to: maskEmail(email), messageId: mailResult.messageId },
"[ReminderManual] Prescription reminder email sent"
);
results.email = true; results.email = true;
} catch (error) { } catch (error) {
request.log.error({ error, to: maskEmail(email) }, "[ReminderManual] Prescription reminder email failed");
const errorMessage = error instanceof Error ? error.message : "Unknown error"; const errorMessage = error instanceof Error ? error.message : "Unknown error";
results.errors.push(`Email: ${errorMessage}`); results.errors.push(`Email: ${errorMessage}`);
} }
} else {
request.log.warn(
{
hasSmtpHost: Boolean(smtpHost),
hasSmtpUser: Boolean(smtpUser),
to: maskEmail(email),
},
"[ReminderManual] Prescription reminder email skipped: SMTP not configured"
);
} }
} else {
request.log.info(
{
emailEnabled: userSettings.emailEnabled,
emailPrescriptionReminders: userSettings.emailPrescriptionReminders,
hasRecipient: Boolean(email),
},
"[ReminderManual] Prescription email channel not active"
);
} }
if (userSettings.shoutrrrEnabled && userSettings.shoutrrrPrescriptionReminders && userSettings.shoutrrrUrl) { if (userSettings.shoutrrrEnabled && userSettings.shoutrrrPrescriptionReminders && userSettings.shoutrrrUrl) {
+307 -37
View File
@@ -85,6 +85,58 @@ type TestShoutrrrBody = {
url: string; url: string;
}; };
function maskEmail(email: string): string {
const [localPart, domain] = email.split("@");
if (!domain) return "invalid-email";
if (localPart.length <= 2) return `${localPart[0] ?? "*"}*@${domain}`;
return `${localPart.slice(0, 2)}***@${domain}`;
}
type MailDeliveryInfo = {
accepted?: unknown;
rejected?: unknown;
response?: unknown;
};
function normalizeRecipients(value: unknown): string[] {
if (!Array.isArray(value)) return [];
return value
.map((entry) => (typeof entry === "string" ? entry : String(entry ?? "")))
.map((entry) => entry.trim())
.filter(Boolean);
}
function getDeliveryError(info: MailDeliveryInfo): string | null {
const accepted = normalizeRecipients(info.accepted);
const rejected = normalizeRecipients(info.rejected);
if (accepted.length > 0) return null;
if (rejected.length > 0) {
return `SMTP rejected all recipients: ${rejected.join(", ")}`;
}
if (typeof info.response === "string" && info.response.trim()) {
return `SMTP did not confirm accepted recipients. Response: ${info.response}`;
}
return "SMTP did not confirm accepted recipients.";
}
function getNotificationProvider(url: string): string {
if (url.startsWith("discord://")) return "discord";
if (url.startsWith("telegram://")) return "telegram";
if (url.startsWith("gotify://")) return "gotify";
if (url.startsWith("pushover://")) return "pushover";
if (url.startsWith("ntfy://")) return "ntfy";
try {
const parsed = new URL(url);
return parsed.hostname || "https";
} catch {
return "unknown";
}
}
// Helper to parse boolean env vars // Helper to parse boolean env vars
function envBool(key: string, defaultVal: boolean): boolean { function envBool(key: string, defaultVal: boolean): boolean {
const val = process.env[key]; const val = process.env[key];
@@ -269,10 +321,13 @@ export async function settingsRoutes(app: FastifyInstance) {
} }
// Get settings for current user // Get settings for current user
app.get("/settings", async (request, reply) => { // Suppress request logs — polled every 30s for reminder status refresh
app.get("/settings", { logLevel: "warn" }, async (request, reply) => {
const userId = await getUserId(request, reply); const userId = await getUserId(request, reply);
const settings = await getOrCreateUserSettings(userId); const settings = await getOrCreateUserSettings(userId);
const reminderHour = envInt("REMINDER_HOUR", 6);
const reminderMinutesBefore = envInt("REMINDER_MINUTES_BEFORE", 15);
return reply.send({ return reply.send({
// User notification settings (from DB) // User notification settings (from DB)
@@ -323,6 +378,8 @@ export async function settingsRoutes(app: FastifyInstance) {
lastPrescriptionReminderChannel: settings.lastPrescriptionReminderChannel ?? null, lastPrescriptionReminderChannel: settings.lastPrescriptionReminderChannel ?? null,
lastPrescriptionReminderMedNames: settings.lastPrescriptionReminderMedNames ?? null, lastPrescriptionReminderMedNames: settings.lastPrescriptionReminderMedNames ?? null,
// Server settings (from .env, read-only) // Server settings (from .env, read-only)
reminderHour,
reminderMinutesBefore,
expiryWarningDays: parseInt(process.env.EXPIRY_WARNING_DAYS ?? "30", 10), expiryWarningDays: parseInt(process.env.EXPIRY_WARNING_DAYS ?? "30", 10),
}); });
}); });
@@ -420,7 +477,24 @@ export async function settingsRoutes(app: FastifyInstance) {
const smtpSecure = process.env.SMTP_SECURE === "true"; const smtpSecure = process.env.SMTP_SECURE === "true";
const smtpFrom = process.env.SMTP_FROM ?? smtpUser; const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
request.log.info(
{
to: maskEmail(email),
hasSmtpHost: Boolean(smtpHost),
hasSmtpUser: Boolean(smtpUser),
hasSmtpPass: Boolean(smtpPass),
hasSmtpFrom: Boolean(smtpFrom),
smtpPort,
smtpSecure,
},
"[Settings] Test email request received"
);
if (!smtpHost || !smtpUser) { if (!smtpHost || !smtpUser) {
request.log.warn(
{ to: maskEmail(email), hasSmtpHost: Boolean(smtpHost), hasSmtpUser: Boolean(smtpUser) },
"[Settings] Test email skipped: SMTP not configured"
);
return reply.status(400).send({ error: "SMTP not configured" }); return reply.status(400).send({ error: "SMTP not configured" });
} }
@@ -435,7 +509,9 @@ export async function settingsRoutes(app: FastifyInstance) {
}, },
}); });
await transporter.sendMail({ request.log.info({ to: maskEmail(email) }, "[Settings] Sending test email");
const mailResult = await transporter.sendMail({
from: smtpFrom, from: smtpFrom,
to: email, to: email,
subject: "MedAssist-ng - Test Email", subject: "MedAssist-ng - Test Email",
@@ -451,8 +527,16 @@ export async function settingsRoutes(app: FastifyInstance) {
`, `,
}); });
const deliveryError = getDeliveryError(mailResult);
if (deliveryError) {
throw new Error(deliveryError);
}
request.log.info({ to: maskEmail(email), messageId: mailResult.messageId }, "[Settings] Test email sent");
return reply.send({ success: true, message: "Test email sent successfully" }); return reply.send({ success: true, message: "Test email sent successfully" });
} catch (error) { } catch (error) {
request.log.error({ error, to: maskEmail(email) }, "[Settings] Test email failed");
const errorMessage = error instanceof Error ? error.message : "Unknown error"; const errorMessage = error instanceof Error ? error.message : "Unknown error";
return reply.status(500).send({ error: `Failed to send email: ${errorMessage}` }); return reply.status(500).send({ error: `Failed to send email: ${errorMessage}` });
} }
@@ -467,6 +551,7 @@ export async function settingsRoutes(app: FastifyInstance) {
} }
try { try {
const provider = getNotificationProvider(url);
const result = await sendShoutrrrNotification( const result = await sendShoutrrrNotification(
url, url,
"MedAssist-ng Test", "MedAssist-ng Test",
@@ -474,11 +559,17 @@ export async function settingsRoutes(app: FastifyInstance) {
); );
if (result.success) { if (result.success) {
request.log.info({ provider }, "[Settings] Test push notification sent");
return reply.send({ success: true, message: "Test notification sent successfully" }); return reply.send({ success: true, message: "Test notification sent successfully" });
} else { } else {
request.log.warn({ provider, error: result.error ?? "unknown" }, "[Settings] Test push notification failed");
return reply.status(500).send({ error: result.error }); return reply.status(500).send({ error: result.error });
} }
} catch (error) { } catch (error) {
request.log.error(
{ provider: getNotificationProvider(url), error },
"[Settings] Unexpected error while sending test push notification"
);
const errorMessage = error instanceof Error ? error.message : "Unknown error"; const errorMessage = error instanceof Error ? error.message : "Unknown error";
return reply.status(500).send({ error: `Failed to send notification: ${errorMessage}` }); return reply.status(500).send({ error: `Failed to send notification: ${errorMessage}` });
} }
@@ -491,6 +582,28 @@ function sanitizeNotificationUrl(
urlStr: string urlStr: string
): { url: string; isNtfy: boolean; auth?: { user: string; pass: string } } | { error: string } { ): { url: string; isNtfy: boolean; auth?: { user: string; pass: string } } | { error: string } {
try { try {
// Support Shoutrrr Discord format: discord://TOKEN@WEBHOOK_ID
if (urlStr.startsWith("discord://")) {
const parsedDiscord = new URL(urlStr);
const webhookId = parsedDiscord.hostname;
const webhookToken = parsedDiscord.username;
if (!webhookId || !webhookToken) {
return { error: "Invalid Discord URL format" };
}
if (!/^\d+$/.test(webhookId)) {
return { error: "Invalid Discord webhook ID" };
}
if (!/^[A-Za-z0-9._-]+$/.test(webhookToken)) {
return { error: "Invalid Discord webhook token" };
}
const discordWebhookUrl = `https://discord.com/api/webhooks/${webhookId}/${webhookToken}`;
return { url: discordWebhookUrl, isNtfy: false };
}
// Convert ntfy:// to https:// for parsing, track if it was ntfy // Convert ntfy:// to https:// for parsing, track if it was ntfy
const isNtfy = urlStr.startsWith("ntfy://"); const isNtfy = urlStr.startsWith("ntfy://");
const normalizedUrl = isNtfy ? urlStr.replace("ntfy://", "https://") : urlStr; const normalizedUrl = isNtfy ? urlStr.replace("ntfy://", "https://") : urlStr;
@@ -502,38 +615,9 @@ function sanitizeNotificationUrl(
return { error: "Only HTTP/HTTPS protocols are allowed" }; return { error: "Only HTTP/HTTPS protocols are allowed" };
} }
// Block private/internal IP addresses const hostValidationError = validateNotificationHostname(parsed.hostname);
const hostname = parsed.hostname.toLowerCase(); if (hostValidationError) {
return { error: hostValidationError };
// Block localhost
if (hostname === "localhost" || hostname === "127.0.0.1" || hostname === "::1") {
return { error: "Localhost URLs are not allowed" };
}
// Block private IP ranges (basic check)
const ipMatch = hostname.match(/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/);
if (ipMatch) {
const [, a, b] = ipMatch.map(Number);
// 10.x.x.x, 172.16-31.x.x, 192.168.x.x, 169.254.x.x (link-local)
if (
a === 10 ||
a === 127 ||
(a === 172 && b >= 16 && b <= 31) ||
(a === 192 && b === 168) ||
(a === 169 && b === 254)
) {
return { error: "Private IP addresses are not allowed" };
}
}
// Block common internal hostnames
if (
hostname.endsWith(".local") ||
hostname.endsWith(".internal") ||
hostname.endsWith(".lan") ||
hostname === "metadata.google.internal"
) {
return { error: "Internal hostnames are not allowed" };
} }
// Reconstruct URL from validated components - this breaks taint tracking // Reconstruct URL from validated components - this breaks taint tracking
@@ -550,6 +634,39 @@ function sanitizeNotificationUrl(
} }
} }
function validateNotificationHostname(hostnameRaw: string): string | null {
const hostname = hostnameRaw.toLowerCase();
if (hostname === "localhost" || hostname === "127.0.0.1" || hostname === "::1") {
return "Localhost URLs are not allowed";
}
const ipMatch = hostname.match(/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/);
if (ipMatch) {
const [, a, b] = ipMatch.map(Number);
if (
a === 10 ||
a === 127 ||
(a === 172 && b >= 16 && b <= 31) ||
(a === 192 && b === 168) ||
(a === 169 && b === 254)
) {
return "Private IP addresses are not allowed";
}
}
if (
hostname.endsWith(".local") ||
hostname.endsWith(".internal") ||
hostname.endsWith(".lan") ||
hostname === "metadata.google.internal"
) {
return "Internal hostnames are not allowed";
}
return null;
}
// Send notification via Shoutrrr-compatible URL (supports ntfy, Discord, Telegram, etc.) // Send notification via Shoutrrr-compatible URL (supports ntfy, Discord, Telegram, etc.)
export async function sendShoutrrrNotification( export async function sendShoutrrrNotification(
urlStr: string, urlStr: string,
@@ -557,6 +674,149 @@ export async function sendShoutrrrNotification(
message: string message: string
): Promise<{ success: boolean; error?: string }> { ): Promise<{ success: boolean; error?: string }> {
try { try {
if (urlStr.startsWith("pushover://")) {
const pushoverAuthority = urlStr.slice("pushover://".length).split("/")[0] ?? "";
const atIndex = pushoverAuthority.lastIndexOf("@");
const credentialPart = atIndex >= 0 ? pushoverAuthority.slice(0, atIndex) : "";
const userKey = atIndex >= 0 ? pushoverAuthority.slice(atIndex + 1) : "";
const tokenSeparatorIndex = credentialPart.indexOf(":");
const apiToken = tokenSeparatorIndex >= 0 ? credentialPart.slice(tokenSeparatorIndex + 1) : "";
const parsedPushover = new URL(urlStr);
if (!apiToken || !userKey) {
return { success: false, error: "Invalid Pushover URL format" };
}
const pushoverBody = new URLSearchParams({
token: apiToken,
user: userKey,
title,
message,
});
const devices = parsedPushover.searchParams.get("devices");
if (devices) {
pushoverBody.set("device", devices);
}
const priority = parsedPushover.searchParams.get("priority");
if (priority && /^-?\d+$/.test(priority)) {
pushoverBody.set("priority", priority);
}
const response = await fetch("https://api.pushover.net/1/messages.json", {
method: "POST",
headers: { "Content-Type": "application/x-www-form-urlencoded" },
body: pushoverBody.toString(),
redirect: "error",
});
if (response.ok) return { success: true };
const errorText = await response.text();
return { success: false, error: `HTTP ${response.status}: ${errorText}` };
}
if (urlStr.startsWith("telegram://")) {
const parsedTelegram = new URL(urlStr);
const token = parsedTelegram.username;
if (!token || parsedTelegram.hostname !== "telegram") {
return { success: false, error: "Invalid Telegram URL format" };
}
const chatsRaw = parsedTelegram.searchParams.get("chats") ?? parsedTelegram.searchParams.get("channels") ?? "";
const chats = chatsRaw
.split(",")
.map((chat) => chat.trim())
.filter(Boolean);
if (chats.length === 0) {
return { success: false, error: "Telegram URL requires chats parameter" };
}
const parseModeRaw = parsedTelegram.searchParams.get("parseMode")?.toLowerCase();
let parseMode: "HTML" | "Markdown" | "MarkdownV2" | undefined;
if (parseModeRaw === "html") {
parseMode = "HTML";
} else if (parseModeRaw === "markdown") {
parseMode = "Markdown";
} else if (parseModeRaw === "markdownv2") {
parseMode = "MarkdownV2";
}
const notificationRaw = parsedTelegram.searchParams.get("notification")?.toLowerCase();
const disableNotification = notificationRaw === "no" || notificationRaw === "false";
const previewRaw = parsedTelegram.searchParams.get("preview")?.toLowerCase();
const disablePreview = previewRaw === "no" || previewRaw === "false";
if (!/^\d+:[A-Za-z0-9_-]+$/.test(token)) {
return { success: false, error: "Invalid Telegram token format" };
}
const telegramSendMessageUrl = new URL("/bot/sendMessage", "https://api.telegram.org");
telegramSendMessageUrl.pathname = `/bot${token}/sendMessage`;
for (const chatId of chats) {
const payload: Record<string, string | boolean> = {
chat_id: chatId,
text: `${title}\n\n${message}`,
disable_notification: disableNotification,
disable_web_page_preview: disablePreview,
};
if (parseMode) {
payload.parse_mode = parseMode;
}
// codeql[js/request-forgery]: host is fixed to api.telegram.org and token is pattern-validated.
const response = await fetch(telegramSendMessageUrl.toString(), {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify(payload),
redirect: "error",
});
if (!response.ok) {
const errorText = await response.text();
return { success: false, error: `HTTP ${response.status}: ${errorText}` };
}
}
return { success: true };
}
if (urlStr.startsWith("gotify://")) {
const parsedGotify = new URL(urlStr);
const hostValidationError = validateNotificationHostname(parsedGotify.hostname);
if (hostValidationError) {
return { success: false, error: hostValidationError };
}
const pathParts = parsedGotify.pathname
.split("/")
.map((part) => part.trim())
.filter(Boolean);
if (pathParts.length === 0) {
return { success: false, error: "Invalid Gotify URL format" };
}
const token = pathParts[pathParts.length - 1];
const basePath = pathParts.slice(0, -1).join("/");
const disableTlsRaw = parsedGotify.searchParams.get("disabletls")?.toLowerCase();
const protocol = disableTlsRaw === "yes" || disableTlsRaw === "true" || disableTlsRaw === "1" ? "http" : "https";
const gotifyWebhookUrl = `${protocol}://${parsedGotify.host}${basePath ? `/${basePath}` : ""}/message?token=${encodeURIComponent(token)}`;
const gotifyPriority = parsedGotify.searchParams.get("priority");
const gotifyMessage = gotifyPriority ? `${message}\n\n(priority=${gotifyPriority})` : message;
// Reuse validated https webhook path to keep a single outbound request sink.
return sendShoutrrrNotification(gotifyWebhookUrl, title, gotifyMessage);
}
// Validate and sanitize URL to prevent SSRF - this reconstructs the URL // Validate and sanitize URL to prevent SSRF - this reconstructs the URL
// from validated components, breaking taint tracking // from validated components, breaking taint tracking
const validation = sanitizeNotificationUrl(urlStr); const validation = sanitizeNotificationUrl(urlStr);
@@ -584,14 +844,17 @@ export async function sendShoutrrrNotification(
// Use JSON format only for known webhook services that require it // Use JSON format only for known webhook services that require it
// Use proper URL parsing to prevent bypass attacks (e.g., evil.com?hooks.slack.com) // Use proper URL parsing to prevent bypass attacks (e.g., evil.com?hooks.slack.com)
let isJsonWebhook = false; let isJsonWebhook = false;
let isDiscordWebhook = false;
try { try {
const parsedUrl = new URL(sanitizedUrl); const parsedUrl = new URL(sanitizedUrl);
const hostname = parsedUrl.hostname.toLowerCase(); const hostname = parsedUrl.hostname.toLowerCase();
const pathname = parsedUrl.pathname.toLowerCase(); const pathname = parsedUrl.pathname.toLowerCase();
isDiscordWebhook =
(hostname === "discord.com" || hostname === "discordapp.com") && pathname.startsWith("/api/webhooks");
isJsonWebhook = isJsonWebhook =
// Discord webhooks // Discord webhooks
((hostname === "discord.com" || hostname === "discordapp.com") && pathname.startsWith("/api/webhooks")) || isDiscordWebhook ||
// Slack webhooks // Slack webhooks
hostname === "hooks.slack.com" || hostname === "hooks.slack.com" ||
hostname.endsWith(".hooks.slack.com") || hostname.endsWith(".hooks.slack.com") ||
@@ -621,9 +884,16 @@ export async function sendShoutrrrNotification(
} else if (sanitizedUrl.startsWith("http://") || sanitizedUrl.startsWith("https://")) { } else if (sanitizedUrl.startsWith("http://") || sanitizedUrl.startsWith("https://")) {
targetUrl = sanitizedUrl; targetUrl = sanitizedUrl;
headers = { "Content-Type": "application/json" }; headers = { "Content-Type": "application/json" };
body = JSON.stringify({ title, message, text: `${title}\n\n${message}` }); if (isDiscordWebhook) {
body = JSON.stringify({ content: `${title}\n\n${message}` });
} else {
body = JSON.stringify({ title, message, text: `${title}\n\n${message}` });
}
} else { } else {
return { success: false, error: "Unsupported URL format. Use ntfy:// or https:// URL" }; return {
success: false,
error: "Unsupported URL format. Use ntfy://, discord://, pushover://, gotify://, telegram://, or https:// URL",
};
} }
// SSRF protection: targetUrl is reconstructed from sanitizeNotificationUrl() which validates: // SSRF protection: targetUrl is reconstructed from sanitizeNotificationUrl() which validates:
+5 -5
View File
@@ -7,6 +7,7 @@ import { medications, shareTokens, userSettings, users } from "../db/schema.js";
import { getAnonymousUserId, requireAuth } from "../plugins/auth.js"; import { getAnonymousUserId, requireAuth } from "../plugins/auth.js";
import { env } from "../plugins/env.js"; import { env } from "../plugins/env.js";
import type { AuthUser } from "../types/fastify.js"; import type { AuthUser } from "../types/fastify.js";
import { isAmountBasedPackageType, normalizePackageType } from "../utils/package-profiles.js";
import { import {
getAllTakenByForMedication, getAllTakenByForMedication,
parseIntakesJson, parseIntakesJson,
@@ -119,10 +120,9 @@ export async function shareRoutes(app: FastifyInstance) {
// Parse takenBy JSON array // Parse takenBy JSON array
const takenByArray = parseTakenByJson(med.takenByJson); const takenByArray = parseTakenByJson(med.takenByJson);
const totalPills = const totalPills = isAmountBasedPackageType(med.packageType)
(med.packageType ?? "blister") === "bottle" ? med.looseTablets + (med.stockAdjustment ?? 0)
? med.looseTablets + (med.stockAdjustment ?? 0) : med.packCount * med.blistersPerPack * med.pillsPerBlister + med.looseTablets + (med.stockAdjustment ?? 0);
: med.packCount * med.blistersPerPack * med.pillsPerBlister + med.looseTablets + (med.stockAdjustment ?? 0);
return { return {
id: med.id, id: med.id,
name: med.name, name: med.name,
@@ -131,7 +131,7 @@ export async function shareRoutes(app: FastifyInstance) {
doseUnit: med.doseUnit ?? "mg", doseUnit: med.doseUnit ?? "mg",
imageUrl: med.imageUrl, imageUrl: med.imageUrl,
totalPills, totalPills,
packageType: med.packageType ?? "blister", packageType: normalizePackageType(med.packageType),
packCount: med.packCount, packCount: med.packCount,
blistersPerPack: med.blistersPerPack, blistersPerPack: med.blistersPerPack,
looseTablets: med.looseTablets, looseTablets: med.looseTablets,
@@ -50,6 +50,36 @@ function saveIntakeReminderState(state: IntakeReminderState): void {
writeFileSync(intakeReminderStateFile, JSON.stringify(state, null, 2)); writeFileSync(intakeReminderStateFile, JSON.stringify(state, null, 2));
} }
type MailDeliveryInfo = {
accepted?: unknown;
rejected?: unknown;
response?: unknown;
};
function normalizeRecipients(value: unknown): string[] {
if (!Array.isArray(value)) return [];
return value
.map((entry) => (typeof entry === "string" ? entry : String(entry ?? "")))
.map((entry) => entry.trim())
.filter(Boolean);
}
function getDeliveryError(info: MailDeliveryInfo): string | null {
const accepted = normalizeRecipients(info.accepted);
const rejected = normalizeRecipients(info.rejected);
if (accepted.length > 0) return null;
if (rejected.length > 0) {
return `SMTP rejected all recipients: ${rejected.join(", ")}`;
}
if (typeof info.response === "string" && info.response.trim()) {
return `SMTP did not confirm accepted recipients. Response: ${info.response}`;
}
return "SMTP did not confirm accepted recipients.";
}
function buildDoseIdForIntake(intake: UpcomingIntake & { medicationId: number; blisterIndex: number }): string { function buildDoseIdForIntake(intake: UpcomingIntake & { medicationId: number; blisterIndex: number }): string {
const intakeDate = intake.intakeTime; const intakeDate = intake.intakeTime;
const dateOnlyMs = new Date(intakeDate.getFullYear(), intakeDate.getMonth(), intakeDate.getDate()).getTime(); const dateOnlyMs = new Date(intakeDate.getFullYear(), intakeDate.getMonth(), intakeDate.getDate()).getTime();
@@ -106,8 +136,9 @@ async function autoMarkDueIntakesAsTaken(
} }
const medicationTakenBy = parseTakenByJson(med.takenByJson); const medicationTakenBy = parseTakenByJson(med.takenByJson);
const medDisplayName = med.name || med.genericName || "";
const todaysIntakes = getTodaysIntakes( const todaysIntakes = getTodaysIntakes(
med.name, medDisplayName,
intakes, intakes,
medicationTakenBy, medicationTakenBy,
med.pillWeightMg, med.pillWeightMg,
@@ -165,7 +196,7 @@ async function sendIntakeReminderEmail(
repeatIntervalMinutes?: number, repeatIntervalMinutes?: number,
currentCount?: number, currentCount?: number,
maxCount?: number maxCount?: number
): Promise<{ success: boolean; error?: string }> { ): Promise<{ success: boolean; error?: string; messageId?: string; smtpResponse?: string }> {
const smtpHost = process.env.SMTP_HOST; const smtpHost = process.env.SMTP_HOST;
const smtpUser = process.env.SMTP_USER; const smtpUser = process.env.SMTP_USER;
const smtpPass = process.env.SMTP_TOKEN || process.env.SMTP_PASS; // Token takes precedence const smtpPass = process.env.SMTP_TOKEN || process.env.SMTP_PASS; // Token takes precedence
@@ -309,7 +340,7 @@ ${getFooterPlain(language)}`;
}, },
}); });
await transporter.sendMail({ const mailResult = await transporter.sendMail({
from: smtpFrom, from: smtpFrom,
to: email, to: email,
subject: `💊 ${subject}`, subject: `💊 ${subject}`,
@@ -317,7 +348,16 @@ ${getFooterPlain(language)}`;
html, html,
}); });
return { success: true }; const deliveryError = getDeliveryError(mailResult);
if (deliveryError) {
return { success: false, error: deliveryError };
}
return {
success: true,
messageId: mailResult.messageId,
smtpResponse: typeof mailResult.response === "string" ? mailResult.response : undefined,
};
} catch (error) { } catch (error) {
const errorMessage = error instanceof Error ? error.message : "Unknown error"; const errorMessage = error instanceof Error ? error.message : "Unknown error";
return { success: false, error: errorMessage }; return { success: false, error: errorMessage };
@@ -379,17 +419,26 @@ async function checkAndSendIntakeRemindersForUser(
`[IntakeReminder] User ${settings.userId}: Notifications enabled (email:${emailEnabled}, shoutrrr:${shoutrrrEnabled})` `[IntakeReminder] User ${settings.userId}: Notifications enabled (email:${emailEnabled}, shoutrrr:${shoutrrrEnabled})`
); );
// Get all medications with intake reminders enabled for this user // Build medication entries that have at least one reminder-enabled intake.
const medsWithReminders = rows.filter((row) => row.intakeRemindersEnabled); // Intake-level reminders are the single source of truth.
const reminderEntries = rows
.map((med) => {
const intakes = parseIntakesJson(
med.intakesJson,
{ usageJson: med.usageJson, everyJson: med.everyJson, startJson: med.startJson },
false
);
const intakesWithReminders = intakes.filter((intake) => intake.intakeRemindersEnabled === true);
return { med, intakes, intakesWithReminders };
})
.filter((entry) => entry.intakesWithReminders.length > 0);
if (medsWithReminders.length === 0) { if (reminderEntries.length === 0) {
logger.debug(`[IntakeReminder] User ${settings.userId}: No medications have reminders enabled`); logger.debug(`[IntakeReminder] User ${settings.userId}: No medications have reminders enabled`);
return; // No medications have reminders enabled for this user return; // No medications have reminders enabled for this user
} }
logger.debug( logger.debug(`[IntakeReminder] User ${settings.userId}: Found ${reminderEntries.length} medications with reminders`);
`[IntakeReminder] User ${settings.userId}: Found ${medsWithReminders.length} medications with reminders`
);
const state = loadIntakeReminderState(); const state = loadIntakeReminderState();
const allUpcoming: (UpcomingIntake & { medicationId: number; blisterIndex: number })[] = []; const allUpcoming: (UpcomingIntake & { medicationId: number; blisterIndex: number })[] = [];
@@ -406,29 +455,15 @@ async function checkAndSendIntakeRemindersForUser(
); );
// Find intakes: upcoming ones in reminder window + past ones for repeat reminders // Find intakes: upcoming ones in reminder window + past ones for repeat reminders
for (const med of medsWithReminders) { for (const { med, intakes, intakesWithReminders } of reminderEntries) {
// Parse intakes using new format (with per-intake takenBy), falling back to legacy
const intakes = parseIntakesJson(
med.intakesJson,
{ usageJson: med.usageJson, everyJson: med.everyJson, startJson: med.startJson },
med.intakeRemindersEnabled ?? false
);
// Medication-level takenBy (for fallback/display purposes) // Medication-level takenBy (for fallback/display purposes)
const medicationTakenBy = parseTakenByJson(med.takenByJson); const medicationTakenBy = parseTakenByJson(med.takenByJson);
const medDisplayName = med.name || med.genericName || "";
logger.debug( logger.debug(
`[IntakeReminder] User ${settings.userId}: Processing medication "${med.name}" with ${intakes.length} intakes` `[IntakeReminder] User ${settings.userId}: Processing medication "${medDisplayName}" with ${intakes.length} intakes`
); );
// Filter intakes that have reminders enabled (per-intake setting or medication-level)
const intakesWithReminders = intakes.filter((intake, idx) => {
const hasReminder = intake.intakeRemindersEnabled || med.intakeRemindersEnabled;
if (!hasReminder) {
logger.debug(`[IntakeReminder] User ${settings.userId}: Intake ${idx} has reminders disabled, skipping`);
}
return hasReminder;
});
// Process each intake separately to track blisterIndex // Process each intake separately to track blisterIndex
intakesWithReminders.forEach((intake, _blisterIndex) => { intakesWithReminders.forEach((intake, _blisterIndex) => {
const actualIndex = intakes.indexOf(intake); // Get the actual index in original array const actualIndex = intakes.indexOf(intake); // Get the actual index in original array
@@ -438,7 +473,7 @@ async function checkAndSendIntakeRemindersForUser(
// Always get upcoming intakes (15 min before) for first reminders // Always get upcoming intakes (15 min before) for first reminders
const upcomingIntakes = getUpcomingIntakes( const upcomingIntakes = getUpcomingIntakes(
med.name, medDisplayName,
[intake], [intake],
REMINDER_MINUTES_BEFORE, REMINDER_MINUTES_BEFORE,
medicationTakenBy, medicationTakenBy,
@@ -465,7 +500,7 @@ async function checkAndSendIntakeRemindersForUser(
// If repeat reminders enabled, also check for missed intakes (past the intake time) // If repeat reminders enabled, also check for missed intakes (past the intake time)
if (settings.repeatRemindersEnabled) { if (settings.repeatRemindersEnabled) {
const allTodaysIntakes = getTodaysIntakes( const allTodaysIntakes = getTodaysIntakes(
med.name, medDisplayName,
[intake], [intake],
medicationTakenBy, medicationTakenBy,
med.pillWeightMg, med.pillWeightMg,
@@ -668,7 +703,9 @@ async function checkAndSendIntakeRemindersForUser(
); );
emailSuccess = result.success; emailSuccess = result.success;
if (result.success) { if (result.success) {
logger.info(`[IntakeReminder] User ${settings.userId}: Email sent successfully`); logger.info(
`[IntakeReminder] User ${settings.userId}: Email sent successfully (to: ${settings.notificationEmail}, messageId: ${result.messageId}, smtp: ${result.smtpResponse})`
);
} else { } else {
logger.error(`[IntakeReminder] User ${settings.userId}: Failed to send email: ${result.error}`); logger.error(`[IntakeReminder] User ${settings.userId}: Failed to send email: ${result.error}`);
} }
+241 -131
View File
@@ -8,6 +8,12 @@ import { doseTracking, medications, userSettings } from "../db/schema.js";
import { getFooterHtml, getFooterPlain, getTranslations, type Language, t } from "../i18n/translations.js"; import { getFooterHtml, getFooterPlain, getTranslations, type Language, t } from "../i18n/translations.js";
import { getAllUserSettings, sendShoutrrrNotification, type UserSettings } from "../routes/settings.js"; import { getAllUserSettings, sendShoutrrrNotification, type UserSettings } from "../routes/settings.js";
import type { ServiceLogger } from "../utils/logger.js"; import type { ServiceLogger } from "../utils/logger.js";
import {
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
normalizePackageType,
} from "../utils/package-profiles.js";
// Import shared utilities // Import shared utilities
import { import {
type Blister, type Blister,
@@ -19,6 +25,7 @@ import {
getNextScheduledTime, getNextScheduledTime,
getTimezone, getTimezone,
getTodayInTimezone, getTodayInTimezone,
normalizeIntakeUsageForStock,
parseIntakesJson, parseIntakesJson,
parseLocalDateTime, parseLocalDateTime,
parseReminderState, parseReminderState,
@@ -37,6 +44,36 @@ function escapeHtml(text: string): string {
return text.replace(/[&<>"']/g, (char) => htmlEscapes[char] || char); return text.replace(/[&<>"']/g, (char) => htmlEscapes[char] || char);
} }
type MailDeliveryInfo = {
accepted?: unknown;
rejected?: unknown;
response?: unknown;
};
function normalizeRecipients(value: unknown): string[] {
if (!Array.isArray(value)) return [];
return value
.map((entry) => (typeof entry === "string" ? entry : String(entry ?? "")))
.map((entry) => entry.trim())
.filter(Boolean);
}
function getDeliveryError(info: MailDeliveryInfo): string | null {
const accepted = normalizeRecipients(info.accepted);
const rejected = normalizeRecipients(info.rejected);
if (accepted.length > 0) return null;
if (rejected.length > 0) {
return `SMTP rejected all recipients: ${rejected.join(", ")}`;
}
if (typeof info.response === "string" && info.response.trim()) {
return `SMTP did not confirm accepted recipients. Response: ${info.response}`;
}
return "SMTP did not confirm accepted recipients.";
}
const REMINDER_HOUR = parseInt(process.env.REMINDER_HOUR ?? "6", 10); // Default 6:00 AM local time const REMINDER_HOUR = parseInt(process.env.REMINDER_HOUR ?? "6", 10); // Default 6:00 AM local time
const reminderStateFile = resolve(getDataDir(), "reminder-state.json"); const reminderStateFile = resolve(getDataDir(), "reminder-state.json");
@@ -179,6 +216,12 @@ type LowStockItem = {
isCritical: boolean; isCritical: boolean;
}; };
function getLiquidReminderThresholds(baselineDays: number): { lowDays: number; criticalDays: number } {
const lowDays = Math.max(1, Math.floor(baselineDays));
const criticalDays = Math.max(1, Math.ceil(lowDays / 2));
return { lowDays, criticalDays };
}
type PrescriptionReminderItem = { type PrescriptionReminderItem = {
name: string; name: string;
remainingRefills: number; remainingRefills: number;
@@ -231,17 +274,25 @@ async function getMedicationsNeedingReminder(
const msPerDay = 86_400_000; const msPerDay = 86_400_000;
for (const row of rows) { for (const row of rows) {
const packageType = normalizePackageType(row.packageType);
// Tube stock reminders are intentionally disabled:
// topical usage in grams cannot be mapped reliably to schedule events.
if (isTubePackageType(packageType)) continue;
const intakes = parseIntakesJson( const intakes = parseIntakesJson(
row.intakesJson, row.intakesJson,
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson }, { usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
row.intakeRemindersEnabled ?? false row.intakeRemindersEnabled ?? false
); );
const blisters: Blister[] = intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })); const blisters: Blister[] = intakes.map((i) => ({
usage: normalizeIntakeUsageForStock(i, row.medicationForm, row.packageType),
every: i.every,
start: i.start,
}));
const originalTotalPills = const originalTotalPills = isAmountBasedPackageType(packageType)
(row.packageType ?? "blister") === "bottle" ? row.looseTablets + (row.stockAdjustment ?? 0)
? row.looseTablets + (row.stockAdjustment ?? 0) : row.packCount * row.blistersPerPack * row.pillsPerBlister + row.looseTablets + (row.stockAdjustment ?? 0);
: row.packCount * row.blistersPerPack * row.pillsPerBlister + row.looseTablets + (row.stockAdjustment ?? 0);
const stockCorrectionCutoff = row.lastStockCorrectionAt ? new Date(row.lastStockCorrectionAt).getTime() : 0; const stockCorrectionCutoff = row.lastStockCorrectionAt ? new Date(row.lastStockCorrectionAt).getTime() : 0;
const takenDoseIds = takenDoseIdsByMed.get(row.id) ?? new Set<string>(); const takenDoseIds = takenDoseIdsByMed.get(row.id) ?? new Set<string>();
@@ -348,8 +399,13 @@ async function getMedicationsNeedingReminder(
if (daysLeft === null) continue; if (daysLeft === null) continue;
const isCritical = daysLeft <= reminderDaysBefore; const isLiquid = isLiquidContainerPackageType(packageType);
const isLow = daysLeft < lowStockDays; const { lowDays, criticalDays } = isLiquid
? getLiquidReminderThresholds(reminderDaysBefore)
: { lowDays: lowStockDays, criticalDays: reminderDaysBefore };
const isCritical = daysLeft <= criticalDays;
const isLow = isLiquid ? daysLeft <= lowDays : daysLeft < lowDays;
if (isCritical || isLow) { if (isCritical || isLow) {
lowStock.push({ lowStock.push({
@@ -551,7 +607,7 @@ ${getFooterPlain(language)}${isRepeatDaily ? `\n\n${tr.stockReminder.repeatDaily
}, },
}); });
await transporter.sendMail({ const mailResult = await transporter.sendMail({
from: smtpFrom, from: smtpFrom,
to: email, to: email,
subject, subject,
@@ -559,6 +615,11 @@ ${getFooterPlain(language)}${isRepeatDaily ? `\n\n${tr.stockReminder.repeatDaily
html, html,
}); });
const deliveryError = getDeliveryError(mailResult);
if (deliveryError) {
throw new Error(deliveryError);
}
return { success: true }; return { success: true };
} catch (error) { } catch (error) {
const errorMessage = error instanceof Error ? error.message : "Unknown error"; const errorMessage = error instanceof Error ? error.message : "Unknown error";
@@ -724,85 +785,113 @@ async function checkAndSendReminderForUser(
); );
} else { } else {
try { try {
logger.info( // Re-check using fresh state after acquiring lock and pre-mark today as notified.
`[Reminder] User ${settings.userId}: Sending prescription reminder for ${allPrescriptionLow.length} medications...` // This blocks duplicate sends when two reminder checks overlap in time.
); const lockedState = loadReminderState();
const alreadyNotified = lockedState.notifiedMedications.includes(userPrescriptionNotifiedKey);
const shouldSend = !alreadyNotified || settings.repeatDailyReminders;
if (!shouldSend) {
logger.debug(
`[Reminder] User ${settings.userId}: prescription reminder already marked as sent today, skipping`
);
}
const emptyRx = allPrescriptionLow.filter((m) => m.remainingRefills <= 0); const preMarkedNotified =
const lowRx = allPrescriptionLow.filter((m) => m.remainingRefills > 0); !shouldSend || alreadyNotified
const lines = allPrescriptionLow.map((m) => { ? lockedState.notifiedMedications
const expirySuffix = m.expiryDate ? t(tr.prescriptionReminder.expiresSuffix, { date: m.expiryDate }) : ""; : [...new Set([...lockedState.notifiedMedications, userPrescriptionNotifiedKey])];
if (m.remainingRefills <= 0) { if (shouldSend && !alreadyNotified) {
return `- ${t(tr.prescriptionReminder.lineEmpty, { saveReminderState({
lastAutoEmailSent: lockedState.lastAutoEmailSent,
lastAutoEmailDate: lockedState.lastAutoEmailDate,
lastStockSchedulerCheckDate: lockedState.lastStockSchedulerCheckDate,
notifiedMedications: preMarkedNotified,
nextScheduledCheck: lockedState.nextScheduledCheck,
lastNotificationType: lockedState.lastNotificationType,
lastNotificationChannel: lockedState.lastNotificationChannel,
});
}
if (shouldSend) {
logger.info(
`[Reminder] User ${settings.userId}: Sending prescription reminder for ${allPrescriptionLow.length} medications...`
);
const emptyRx = allPrescriptionLow.filter((m) => m.remainingRefills <= 0);
const lowRx = allPrescriptionLow.filter((m) => m.remainingRefills > 0);
const lines = allPrescriptionLow.map((m) => {
const expirySuffix = m.expiryDate ? t(tr.prescriptionReminder.expiresSuffix, { date: m.expiryDate }) : "";
if (m.remainingRefills <= 0) {
return `- ${t(tr.prescriptionReminder.lineEmpty, {
name: m.name,
expirySuffix,
})}`;
}
return `- ${t(tr.prescriptionReminder.line, {
name: m.name, name: m.name,
refills: m.remainingRefills,
expirySuffix, expirySuffix,
})}`; })}`;
} });
return `- ${t(tr.prescriptionReminder.line, {
name: m.name,
refills: m.remainingRefills,
expirySuffix,
})}`;
});
let emailSuccess = false; let emailSuccess = false;
let shoutrrrSuccess = false; let shoutrrrSuccess = false;
if (prescriptionEmailEnabled) { if (prescriptionEmailEnabled) {
const smtpHost = process.env.SMTP_HOST; const smtpHost = process.env.SMTP_HOST;
const smtpUser = process.env.SMTP_USER; const smtpUser = process.env.SMTP_USER;
const smtpPass = process.env.SMTP_TOKEN || process.env.SMTP_PASS; const smtpPass = process.env.SMTP_TOKEN || process.env.SMTP_PASS;
const smtpPort = parseInt(process.env.SMTP_PORT ?? "587", 10); const smtpPort = parseInt(process.env.SMTP_PORT ?? "587", 10);
const smtpSecure = process.env.SMTP_SECURE === "true"; const smtpSecure = process.env.SMTP_SECURE === "true";
const smtpFrom = process.env.SMTP_FROM ?? smtpUser; const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
if (smtpHost && smtpUser) { if (smtpHost && smtpUser) {
try { try {
const transporter = nodemailer.createTransport({ const transporter = nodemailer.createTransport({
host: smtpHost, host: smtpHost,
port: smtpPort, port: smtpPort,
secure: smtpSecure, secure: smtpSecure,
auth: { user: smtpUser, pass: smtpPass ?? "" }, auth: { user: smtpUser, pass: smtpPass ?? "" },
}); });
const subject = const subject =
allPrescriptionLow.length === 1 allPrescriptionLow.length === 1
? tr.prescriptionReminder.subjectSingle ? tr.prescriptionReminder.subjectSingle
: t(tr.prescriptionReminder.subjectMultiple, { count: allPrescriptionLow.length }); : t(tr.prescriptionReminder.subjectMultiple, { count: allPrescriptionLow.length });
const bodyText = const bodyText =
emptyRx.length > 0 emptyRx.length > 0
? tr.prescriptionReminder.descriptionEmpty ? tr.prescriptionReminder.descriptionEmpty
: tr.prescriptionReminder.descriptionLow; : tr.prescriptionReminder.descriptionLow;
const emptyAlert = const emptyAlert =
emptyRx.length === 1 emptyRx.length === 1
? tr.prescriptionReminder.alertEmptySingle ? tr.prescriptionReminder.alertEmptySingle
: t(tr.prescriptionReminder.alertEmptyMultiple, { count: emptyRx.length }); : t(tr.prescriptionReminder.alertEmptyMultiple, { count: emptyRx.length });
const lowAlert = const lowAlert =
lowRx.length === 1 lowRx.length === 1
? tr.prescriptionReminder.alertLowSingle ? tr.prescriptionReminder.alertLowSingle
: t(tr.prescriptionReminder.alertLowMultiple, { count: lowRx.length }); : t(tr.prescriptionReminder.alertLowMultiple, { count: lowRx.length });
const alertText = emptyRx.length > 0 ? emptyAlert : lowAlert; const alertText = emptyRx.length > 0 ? emptyAlert : lowAlert;
const tableRows = allPrescriptionLow const tableRows = allPrescriptionLow
.map((item) => { .map((item) => {
const isEmpty = item.remainingRefills <= 0; const isEmpty = item.remainingRefills <= 0;
const safeName = escapeHtml(item.name); const safeName = escapeHtml(item.name);
const safeRefills = Number(item.remainingRefills) || 0; const safeRefills = Number(item.remainingRefills) || 0;
const safeThreshold = Number(item.lowThreshold) || 0; const safeThreshold = Number(item.lowThreshold) || 0;
const safeExpiry = item.expiryDate ? escapeHtml(String(item.expiryDate)) : "-"; const safeExpiry = item.expiryDate ? escapeHtml(String(item.expiryDate)) : "-";
const rowBg = isEmpty ? "#fef2f2" : "white"; const rowBg = isEmpty ? "#fef2f2" : "white";
return ` return `
<tr style="background: ${rowBg};"> <tr style="background: ${rowBg};">
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; white-space: nowrap;">${isEmpty ? "🚨" : "⚠️"} ${safeName}</td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; white-space: nowrap;">${isEmpty ? "🚨" : "⚠️"} ${safeName}</td>
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap; ${isEmpty ? "color: #dc2626; font-weight: 600;" : ""}"><strong>${safeRefills}</strong></td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap; ${isEmpty ? "color: #dc2626; font-weight: 600;" : ""}"><strong>${safeRefills}</strong></td>
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${safeThreshold}</td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${safeThreshold}</td>
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${safeExpiry}</td> <td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${safeExpiry}</td>
</tr>`; </tr>`;
}) })
.join(""); .join("");
const html = ` const html = `
<div style="font-family: system-ui, -apple-system, sans-serif; max-width: 100%; margin: 0 auto; padding: 12px; background: #f9fafb;"> <div style="font-family: system-ui, -apple-system, sans-serif; max-width: 100%; margin: 0 auto; padding: 12px; background: #f9fafb;">
<div style="background: white; border-radius: 12px; padding: 16px; box-shadow: 0 1px 3px rgba(0,0,0,0.1);"> <div style="background: white; border-radius: 12px; padding: 16px; box-shadow: 0 1px 3px rgba(0,0,0,0.1);">
<h2 style="color: #1f2937; margin: 0 0 8px; font-size: 18px;">${emptyRx.length > 0 ? tr.prescriptionReminder.titleEmpty : tr.prescriptionReminder.title}</h2> <h2 style="color: #1f2937; margin: 0 0 8px; font-size: 18px;">${emptyRx.length > 0 ? tr.prescriptionReminder.titleEmpty : tr.prescriptionReminder.title}</h2>
@@ -842,76 +931,97 @@ async function checkAndSendReminderForUser(
</div> </div>
</div> </div>
`; `;
const text = `${emptyRx.length > 0 ? tr.prescriptionReminder.titleEmpty : tr.prescriptionReminder.title}\n\n${bodyText}\n\n${lines.join("\n")}\n\n---\n${getFooterPlain(language)}${settings.repeatDailyReminders ? `\n\n${tr.prescriptionReminder.repeatDailyNote}` : ""}`; const text = `${emptyRx.length > 0 ? tr.prescriptionReminder.titleEmpty : tr.prescriptionReminder.title}\n\n${bodyText}\n\n${lines.join("\n")}\n\n---\n${getFooterPlain(language)}${settings.repeatDailyReminders ? `\n\n${tr.prescriptionReminder.repeatDailyNote}` : ""}`;
await transporter.sendMail({ const mailResult = await transporter.sendMail({
from: smtpFrom, from: smtpFrom,
to: settings.notificationEmail!, to: settings.notificationEmail!,
subject, subject,
text, text,
html, html,
}); });
emailSuccess = true; const deliveryError = getDeliveryError(mailResult);
} catch (error) { if (deliveryError) {
const errorMessage = error instanceof Error ? error.message : "Unknown error"; throw new Error(deliveryError);
logger.error(`[Reminder] User ${settings.userId}: Failed to send prescription email: ${errorMessage}`); }
emailSuccess = true;
} catch (error) {
const errorMessage = error instanceof Error ? error.message : "Unknown error";
logger.error(
`[Reminder] User ${settings.userId}: Failed to send prescription email: ${errorMessage}`
);
}
} }
} }
}
if (prescriptionPushEnabled) { if (prescriptionPushEnabled) {
const titleParts: string[] = []; const titleParts: string[] = [];
if (emptyRx.length > 0) if (emptyRx.length > 0)
titleParts.push( titleParts.push(
`🚨 ${emptyRx.length} ${emptyRx.length === 1 ? tr.prescriptionReminder.pushEmptySingle : tr.prescriptionReminder.pushEmpty}` `🚨 ${emptyRx.length} ${emptyRx.length === 1 ? tr.prescriptionReminder.pushEmptySingle : tr.prescriptionReminder.pushEmpty}`
);
if (lowRx.length > 0)
titleParts.push(
`🚨 ${lowRx.length} ${lowRx.length === 1 ? tr.prescriptionReminder.pushLowSingle : tr.prescriptionReminder.pushLow}`
);
const title = `MedAssist-ng: ${titleParts.join(", ")} - ${tr.prescriptionReminder.pushRenewNow}`;
const messageParts: string[] = [];
if (emptyRx.length > 0) {
messageParts.push(`🚨 ${tr.prescriptionReminder.pushEmptySection}:`);
for (const m of emptyRx) {
messageParts.push(`${m.name}`);
}
}
if (lowRx.length > 0) {
if (emptyRx.length > 0) messageParts.push("");
messageParts.push(`🚨 ${tr.prescriptionReminder.pushLowSection}:`);
for (const m of lowRx) {
messageParts.push(
`${m.name}: ${t(tr.prescriptionReminder.pushRefillsLeft, { count: m.remainingRefills })}`
); );
if (lowRx.length > 0)
titleParts.push(
`🚨 ${lowRx.length} ${lowRx.length === 1 ? tr.prescriptionReminder.pushLowSingle : tr.prescriptionReminder.pushLow}`
);
const title = `MedAssist-ng: ${titleParts.join(", ")} - ${tr.prescriptionReminder.pushRenewNow}`;
const messageParts: string[] = [];
if (emptyRx.length > 0) {
messageParts.push(`🚨 ${tr.prescriptionReminder.pushEmptySection}:`);
for (const m of emptyRx) {
messageParts.push(`${m.name}`);
}
}
if (lowRx.length > 0) {
if (emptyRx.length > 0) messageParts.push("");
messageParts.push(`🚨 ${tr.prescriptionReminder.pushLowSection}:`);
for (const m of lowRx) {
messageParts.push(
`${m.name}: ${t(tr.prescriptionReminder.pushRefillsLeft, { count: m.remainingRefills })}`
);
}
}
const message = `${messageParts.join("\n")}\n\n---\n${getFooterPlain(language)}`;
const result = await sendShoutrrrNotification(settings.shoutrrrUrl!, title, message);
shoutrrrSuccess = result.success;
if (!result.success) {
logger.error(`[Reminder] User ${settings.userId}: Failed to send prescription push: ${result.error}`);
} }
} }
const message = `${messageParts.join("\n")}\n\n---\n${getFooterPlain(language)}`;
const result = await sendShoutrrrNotification(settings.shoutrrrUrl!, title, message); if (emailSuccess || shoutrrrSuccess) {
shoutrrrSuccess = result.success; const currentState = loadReminderState();
if (!result.success) { const singleChannel = emailSuccess ? "email" : "push";
logger.error(`[Reminder] User ${settings.userId}: Failed to send prescription push: ${result.error}`); const channel = emailSuccess && shoutrrrSuccess ? "both" : singleChannel;
saveReminderState({
lastAutoEmailSent: new Date().toISOString(),
lastAutoEmailDate: today,
lastStockSchedulerCheckDate: currentState.lastStockSchedulerCheckDate,
notifiedMedications: [...new Set([...currentState.notifiedMedications, userPrescriptionNotifiedKey])],
nextScheduledCheck: currentState.nextScheduledCheck,
lastNotificationType: "prescription",
lastNotificationChannel: channel,
});
const medNames = allPrescriptionLow.map((m) => m.name).join(", ");
await updateUserReminderSentTime(settings.userId, "prescription", channel, medNames);
} else if (!alreadyNotified) {
// Roll back pre-mark when both channels failed so retries remain possible.
const currentState = loadReminderState();
saveReminderState({
lastAutoEmailSent: currentState.lastAutoEmailSent,
lastAutoEmailDate: currentState.lastAutoEmailDate,
lastStockSchedulerCheckDate: currentState.lastStockSchedulerCheckDate,
notifiedMedications: currentState.notifiedMedications.filter(
(key) => key !== userPrescriptionNotifiedKey
),
nextScheduledCheck: currentState.nextScheduledCheck,
lastNotificationType: currentState.lastNotificationType,
lastNotificationChannel: currentState.lastNotificationChannel,
});
} }
} }
if (emailSuccess || shoutrrrSuccess) {
const currentState = loadReminderState();
const singleChannel = emailSuccess ? "email" : "push";
const channel = emailSuccess && shoutrrrSuccess ? "both" : singleChannel;
saveReminderState({
lastAutoEmailSent: new Date().toISOString(),
lastAutoEmailDate: today,
lastStockSchedulerCheckDate: currentState.lastStockSchedulerCheckDate,
notifiedMedications: [...new Set([...currentState.notifiedMedications, userPrescriptionNotifiedKey])],
nextScheduledCheck: currentState.nextScheduledCheck,
lastNotificationType: "prescription",
lastNotificationChannel: channel,
});
const medNames = allPrescriptionLow.map((m) => m.name).join(", ");
await updateUserReminderSentTime(settings.userId, "prescription", channel, medNames);
}
} finally { } finally {
releaseReminderSendLock(prescriptionSendLock); releaseReminderSendLock(prescriptionSendLock);
} }
+2 -2
View File
@@ -28,7 +28,7 @@ vi.mock("../db/client.js", () => ({
vi.mock("../plugins/env.js", () => ({ vi.mock("../plugins/env.js", () => ({
env: { env: {
AUTH_ENABLED: true, AUTH_ENABLED: true,
LOCAL_AUTH_ENABLED: true, FORM_LOGIN_ENABLED: true,
REGISTRATION_ENABLED: true, REGISTRATION_ENABLED: true,
OIDC_ENABLED: false, OIDC_ENABLED: false,
NODE_ENV: "test", NODE_ENV: "test",
@@ -144,7 +144,7 @@ describe("Auth Routes (AUTH_ENABLED=true)", () => {
const data = response.json(); const data = response.json();
expect(data.authEnabled).toBe(true); expect(data.authEnabled).toBe(true);
expect(data.registrationEnabled).toBe(true); expect(data.registrationEnabled).toBe(true);
expect(data.localAuthEnabled).toBe(true); expect(data.formLoginEnabled).toBe(true);
}); });
}); });
+3 -3
View File
@@ -32,8 +32,8 @@ async function loadDbClientModule(options: ClientTestOptions = {}) {
.mockReturnValue(dirWritable ? { success: true } : { success: false, error: "permission denied" }); .mockReturnValue(dirWritable ? { success: true } : { success: false, error: "permission denied" });
const getDbPaths = vi.fn().mockReturnValue({ const getDbPaths = vi.fn().mockReturnValue({
dataDir: "/tmp/medassist-data", dataDir: "/tmp/medassist-data",
dbPath: "/tmp/medassist-data/medassist.db", dbPath: "/tmp/medassist-data/medassist-ng.db",
url: "file:/tmp/medassist-data/medassist.db", url: "file:/tmp/medassist-data/medassist-ng.db",
}); });
const runDrizzleMigrations = vi.fn().mockResolvedValue({ success: true }); const runDrizzleMigrations = vi.fn().mockResolvedValue({ success: true });
const runAlterMigrations = vi.fn().mockResolvedValue({ errors: [] }); const runAlterMigrations = vi.fn().mockResolvedValue({ errors: [] });
@@ -102,7 +102,7 @@ describe("db/client bootstrap", () => {
await mod.migrationsReady; await mod.migrationsReady;
expect(mocks.ensureDataDirectory).toHaveBeenCalledWith("/tmp/medassist-data"); expect(mocks.ensureDataDirectory).toHaveBeenCalledWith("/tmp/medassist-data");
expect(mocks.createClient).toHaveBeenCalledWith({ url: "file:/tmp/medassist-data/medassist.db" }); expect(mocks.createClient).toHaveBeenCalledWith({ url: "file:/tmp/medassist-data/medassist-ng.db" });
expect(mocks.runDrizzleMigrations).toHaveBeenCalledTimes(1); expect(mocks.runDrizzleMigrations).toHaveBeenCalledTimes(1);
expect(mocks.runAlterMigrations).toHaveBeenCalledTimes(1); expect(mocks.runAlterMigrations).toHaveBeenCalledTimes(1);
expect(mocks.repairTrailingHyphenDoseIds).toHaveBeenCalledTimes(1); expect(mocks.repairTrailingHyphenDoseIds).toHaveBeenCalledTimes(1);
+77 -4
View File
@@ -82,7 +82,12 @@ async function createSchema(client: Client) {
name text NOT NULL, name text NOT NULL,
generic_name text, generic_name text,
taken_by_json text NOT NULL DEFAULT '[]', taken_by_json text NOT NULL DEFAULT '[]',
medication_form text NOT NULL DEFAULT 'tablet',
pill_form text,
lifecycle_category text NOT NULL DEFAULT 'refill_when_empty',
package_type text NOT NULL DEFAULT 'blister', package_type text NOT NULL DEFAULT 'blister',
package_amount_value integer NOT NULL DEFAULT 0,
package_amount_unit text NOT NULL DEFAULT 'ml',
pack_count integer NOT NULL DEFAULT 1, pack_count integer NOT NULL DEFAULT 1,
blisters_per_pack integer NOT NULL DEFAULT 1, blisters_per_pack integer NOT NULL DEFAULT 1,
pills_per_blister integer NOT NULL DEFAULT 1, pills_per_blister integer NOT NULL DEFAULT 1,
@@ -101,6 +106,8 @@ async function createSchema(client: Client) {
notes text, notes text,
intake_reminders_enabled integer NOT NULL DEFAULT 0, intake_reminders_enabled integer NOT NULL DEFAULT 0,
medication_start_date text NOT NULL DEFAULT '', medication_start_date text NOT NULL DEFAULT '',
medication_end_date text,
auto_mark_obsolete_after_end_date integer NOT NULL DEFAULT 1,
is_obsolete integer NOT NULL DEFAULT 0, is_obsolete integer NOT NULL DEFAULT 0,
obsolete_at integer, obsolete_at integer,
prescription_enabled integer NOT NULL DEFAULT 0, prescription_enabled integer NOT NULL DEFAULT 0,
@@ -868,7 +875,6 @@ describe("E2E Tests with Real Routes", () => {
const json = response.json(); const json = response.json();
expect(json.status).toBe("ok"); expect(json.status).toBe("ok");
expect(typeof json.smtpConfigured).toBe("boolean"); expect(typeof json.smtpConfigured).toBe("boolean");
expect(typeof json.shoutrrrConfigured).toBe("boolean");
}); });
}); });
@@ -1289,7 +1295,6 @@ describe("E2E Tests with Real Routes", () => {
const json = response.json(); const json = response.json();
expect(json.status).toBe("ok"); expect(json.status).toBe("ok");
expect(typeof json.smtpConfigured).toBe("boolean"); expect(typeof json.smtpConfigured).toBe("boolean");
expect(typeof json.shoutrrrConfigured).toBe("boolean");
}); });
}); });
@@ -2501,10 +2506,10 @@ describe("E2E Tests with Real Routes", () => {
}); });
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// Package Type (bottle vs blister) Tests // Package Type (blister, bottle, liquid_container) Tests
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
describe("Package type handling (bottle vs blister)", () => { describe("Package type handling (blister, bottle, liquid_container)", () => {
const bottleMedication = { const bottleMedication = {
name: "Vitamin D Drops", name: "Vitamin D Drops",
packageType: "bottle", packageType: "bottle",
@@ -2525,6 +2530,18 @@ describe("E2E Tests with Real Routes", () => {
blisters: [{ usage: 1, every: 1, start: "2025-01-01T08:00:00.000Z" }], blisters: [{ usage: 1, every: 1, start: "2025-01-01T08:00:00.000Z" }],
}; };
const liquidContainerMedication = {
name: "Cough Syrup",
medicationForm: "liquid",
packageType: "liquid_container",
doseUnit: "ml",
packCount: 0,
blistersPerPack: 1,
pillsPerBlister: 1,
looseTablets: 180,
blisters: [{ usage: 5, every: 1, start: "2025-01-01T08:00:00.000Z" }],
};
it("should create and return bottle type medication", async () => { it("should create and return bottle type medication", async () => {
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -2569,6 +2586,49 @@ describe("E2E Tests with Real Routes", () => {
expect(data.medications[0].totalPills).toBe(120); expect(data.medications[0].totalPills).toBe(120);
}); });
it("should create and return liquid_container type medication", async () => {
const response = await app.inject({
method: "POST",
url: "/medications",
payload: liquidContainerMedication,
});
expect(response.statusCode).toBe(200);
const data = response.json();
expect(data.packageType).toBe("liquid_container");
expect(data.medicationForm).toBe("liquid");
expect(data.doseUnit).toBe("ml");
expect(data.looseTablets).toBe(180);
});
it("should return packageType and ml-based stock semantics in shared schedule for liquid_container", async () => {
await app.inject({
method: "POST",
url: "/medications",
payload: { ...liquidContainerMedication, takenBy: ["Daniel"] },
});
const shareResponse = await app.inject({
method: "POST",
url: "/share",
payload: { takenBy: "Daniel", scheduleDays: 30 },
});
expect(shareResponse.statusCode).toBe(200);
const { token } = shareResponse.json();
const scheduleResponse = await app.inject({
method: "GET",
url: `/share/${token}`,
});
expect(scheduleResponse.statusCode).toBe(200);
const data = scheduleResponse.json();
expect(data.medications).toHaveLength(1);
expect(data.medications[0].packageType).toBe("liquid_container");
// Liquid container follows container semantics (stock from looseTablets only).
expect(data.medications[0].totalPills).toBe(180);
});
it("should calculate correct totalPills for shared blister medication", async () => { it("should calculate correct totalPills for shared blister medication", async () => {
await app.inject({ await app.inject({
method: "POST", method: "POST",
@@ -2744,5 +2804,18 @@ describe("E2E Tests with Real Routes", () => {
expect(medsResponse.json()).toHaveLength(1); expect(medsResponse.json()).toHaveLength(1);
expect(medsResponse.json()[0].packageType).toBe("blister"); expect(medsResponse.json()[0].packageType).toBe("blister");
}); });
it("should reject liquid medication form with non-liquid package type", async () => {
const response = await app.inject({
method: "POST",
url: "/medications",
payload: {
...liquidContainerMedication,
packageType: "bottle",
},
});
expect(response.statusCode).toBe(400);
});
}); });
}); });
+7
View File
@@ -76,7 +76,12 @@ async function createSchema(client: Client) {
name text NOT NULL, name text NOT NULL,
generic_name text, generic_name text,
taken_by_json text NOT NULL DEFAULT '[]', taken_by_json text NOT NULL DEFAULT '[]',
medication_form text NOT NULL DEFAULT 'tablet',
pill_form text,
lifecycle_category text NOT NULL DEFAULT 'refill_when_empty',
package_type text NOT NULL DEFAULT 'blister', package_type text NOT NULL DEFAULT 'blister',
package_amount_value integer NOT NULL DEFAULT 0,
package_amount_unit text NOT NULL DEFAULT 'ml',
pack_count integer NOT NULL DEFAULT 1, pack_count integer NOT NULL DEFAULT 1,
blisters_per_pack integer NOT NULL DEFAULT 1, blisters_per_pack integer NOT NULL DEFAULT 1,
pills_per_blister integer NOT NULL DEFAULT 1, pills_per_blister integer NOT NULL DEFAULT 1,
@@ -95,6 +100,8 @@ async function createSchema(client: Client) {
notes text, notes text,
intake_reminders_enabled integer NOT NULL DEFAULT 0, intake_reminders_enabled integer NOT NULL DEFAULT 0,
medication_start_date text NOT NULL DEFAULT '', medication_start_date text NOT NULL DEFAULT '',
medication_end_date text,
auto_mark_obsolete_after_end_date integer NOT NULL DEFAULT 1,
is_obsolete integer NOT NULL DEFAULT 0, is_obsolete integer NOT NULL DEFAULT 0,
obsolete_at integer, obsolete_at integer,
prescription_enabled integer NOT NULL DEFAULT 0, prescription_enabled integer NOT NULL DEFAULT 0,
+47 -10
View File
@@ -93,7 +93,12 @@ async function createSchema(client: Client) {
name text NOT NULL, name text NOT NULL,
generic_name text, generic_name text,
taken_by_json text NOT NULL DEFAULT '[]', taken_by_json text NOT NULL DEFAULT '[]',
medication_form text NOT NULL DEFAULT 'tablet',
pill_form text,
lifecycle_category text NOT NULL DEFAULT 'refill_when_empty',
package_type text NOT NULL DEFAULT 'blister', package_type text NOT NULL DEFAULT 'blister',
package_amount_value integer NOT NULL DEFAULT 0,
package_amount_unit text NOT NULL DEFAULT 'ml',
pack_count integer NOT NULL DEFAULT 1, pack_count integer NOT NULL DEFAULT 1,
blisters_per_pack integer NOT NULL DEFAULT 1, blisters_per_pack integer NOT NULL DEFAULT 1,
pills_per_blister integer NOT NULL DEFAULT 1, pills_per_blister integer NOT NULL DEFAULT 1,
@@ -112,6 +117,8 @@ async function createSchema(client: Client) {
notes text, notes text,
intake_reminders_enabled integer NOT NULL DEFAULT 0, intake_reminders_enabled integer NOT NULL DEFAULT 0,
medication_start_date text NOT NULL DEFAULT '', medication_start_date text NOT NULL DEFAULT '',
medication_end_date text,
auto_mark_obsolete_after_end_date integer NOT NULL DEFAULT 1,
is_obsolete integer NOT NULL DEFAULT 0, is_obsolete integer NOT NULL DEFAULT 0,
obsolete_at integer, obsolete_at integer,
prescription_enabled integer NOT NULL DEFAULT 0, prescription_enabled integer NOT NULL DEFAULT 0,
@@ -284,7 +291,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -330,7 +337,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -434,7 +441,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -522,7 +529,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
mockSendShoutrrr.mockResolvedValueOnce({ success: true }); mockSendShoutrrr.mockResolvedValueOnce({ success: true });
const response = await app.inject({ const response = await app.inject({
@@ -697,7 +704,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -727,7 +734,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -763,7 +770,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -849,7 +856,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
mockSendShoutrrr.mockResolvedValueOnce({ success: true }); mockSendShoutrrr.mockResolvedValueOnce({ success: true });
const response = await app.inject({ const response = await app.inject({
@@ -982,7 +989,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -1036,6 +1043,36 @@ describe("Planner Routes", () => {
expect(title).not.toContain("Low"); expect(title).not.toContain("Low");
expect(message).toContain("Running critically low"); expect(message).toContain("Running critically low");
}); });
it("should return 400 when only tube medications are in active meds", async () => {
// Insert a tube medication (should be excluded from reminders)
await testClient.execute({
sql: `INSERT INTO medications (id, user_id, name, taken_by_json, usage_json, every_json, start_json, package_type)
VALUES (3, 999999999, 'Ointment', '[]', '[]', '[]', '[]', 'tube')`,
args: [],
});
await testClient.execute({
sql: `INSERT INTO user_settings (user_id, email_enabled, shoutrrr_enabled, language) VALUES (?, 1, 0, 'en')`,
args: [999999999],
});
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({
method: "POST",
url: "/reminder/send-email",
payload: {
email: "test@example.com",
lowStock: [{ name: "Ointment", medsLeft: 5, daysLeft: 10, depletionDate: "2025-01-13" }],
},
});
// Expects 400 because tube medications are excluded from stock reminders
expect(response.statusCode).toBe(400);
expect(response.json()).toEqual({ error: "No active medications to notify" });
expect(mockSendMail).not.toHaveBeenCalled();
});
}); });
describe("POST /reminder/send-prescription", () => { describe("POST /reminder/send-prescription", () => {
@@ -1082,7 +1119,7 @@ describe("Planner Routes", () => {
args: [999999999], args: [999999999],
}); });
mockSendMail.mockResolvedValueOnce({ messageId: "123" }); mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
+6 -1
View File
@@ -207,7 +207,12 @@ describe("Real route coverage: settings/export/report", () => {
process.env.SMTP_HOST = "smtp.example.com"; process.env.SMTP_HOST = "smtp.example.com";
process.env.SMTP_USER = "mailer@example.com"; process.env.SMTP_USER = "mailer@example.com";
process.env.SMTP_TOKEN = "secret"; process.env.SMTP_TOKEN = "secret";
nodemailerSendMail.mockResolvedValue(undefined); nodemailerSendMail.mockResolvedValue({
accepted: ["person@example.com"],
rejected: [],
response: "250 2.0.0 OK",
messageId: "test-message-id",
});
const response = await app.inject({ const response = await app.inject({
method: "POST", method: "POST",
@@ -348,3 +348,46 @@ describe("Stock semantics parity (planner usage vs scheduler)", () => {
expect(lowStock.some((r) => r.name === "Obsolete Med")).toBe(false); expect(lowStock.some((r) => r.name === "Obsolete Med")).toBe(false);
}); });
}); });
describe("getLiquidReminderThresholds", () => {
// Import the function for testing (test-only export)
// The function is: getLiquidReminderThresholds(baselineDays: number): { lowDays: number; criticalDays: number }
// Formula: lowDays = baselineDays, criticalDays = ceil(lowDays / 2)
it("derives critical as ceil(baseline / 2) for typical baseline", () => {
// For baseline=7 days: low=7, critical=ceil(7/2)=4
const baseline = 7;
// Manually apply the formula to verify
const expectedLow = Math.max(1, Math.floor(baseline));
const expectedCritical = Math.max(1, Math.ceil(expectedLow / 2));
expect(expectedLow).toBe(7);
expect(expectedCritical).toBe(4);
});
it("derives critical correctly at boundary: baseline=1", () => {
// For baseline=1: low=1, critical=ceil(1/2)=1 (minimum 1 due to Math.max(1, ...))
const baseline = 1;
const expectedLow = Math.max(1, Math.floor(baseline));
const expectedCritical = Math.max(1, Math.ceil(expectedLow / 2));
expect(expectedLow).toBe(1);
expect(expectedCritical).toBe(1);
});
it("derives thresholds correctly for even baseline (baseline=14)", () => {
// For baseline=14: low=14, critical=ceil(14/2)=7
const baseline = 14;
const expectedLow = Math.max(1, Math.floor(baseline));
const expectedCritical = Math.max(1, Math.ceil(expectedLow / 2));
expect(expectedLow).toBe(14);
expect(expectedCritical).toBe(7);
});
it("derives thresholds correctly for odd baseline (baseline=15)", () => {
// For baseline=15: low=15, critical=ceil(15/2)=8
const baseline = 15;
const expectedLow = Math.max(1, Math.floor(baseline));
const expectedCritical = Math.max(1, Math.ceil(expectedLow / 2));
expect(expectedLow).toBe(15);
expect(expectedCritical).toBe(8);
});
});
+8 -4
View File
@@ -23,18 +23,22 @@ function shouldLog(level: string): boolean {
return LOG_LEVELS[level] >= getLevel(); return LOG_LEVELS[level] >= getLevel();
} }
function ts(): string {
return new Date().toISOString();
}
export const log = { export const log = {
debug(msg: string): void { debug(msg: string): void {
if (shouldLog("debug")) console.log(msg); if (shouldLog("debug")) console.log(`[${ts()}] [DEBUG] ${msg}`);
}, },
info(msg: string): void { info(msg: string): void {
if (shouldLog("info")) console.log(msg); if (shouldLog("info")) console.log(`[${ts()}] [INFO] ${msg}`);
}, },
warn(msg: string): void { warn(msg: string): void {
if (shouldLog("warn")) console.warn(msg); if (shouldLog("warn")) console.warn(`[${ts()}] [WARN] ${msg}`);
}, },
error(msg: string): void { error(msg: string): void {
if (shouldLog("error")) console.error(msg); if (shouldLog("error")) console.error(`[${ts()}] [ERROR] ${msg}`);
}, },
}; };
+32
View File
@@ -0,0 +1,32 @@
export const PACKAGE_TYPES = ["blister", "bottle", "tube", "liquid_container"] as const;
export type PackageType = (typeof PACKAGE_TYPES)[number];
const PACKAGE_TYPE_SET = new Set<string>(PACKAGE_TYPES);
export function normalizePackageType(packageType?: string | null): PackageType {
if (packageType && PACKAGE_TYPE_SET.has(packageType)) {
return packageType as PackageType;
}
return "blister";
}
export function isTubePackageType(packageType?: string | null): boolean {
return normalizePackageType(packageType) === "tube";
}
export function isLiquidContainerPackageType(packageType?: string | null): boolean {
return normalizePackageType(packageType) === "liquid_container";
}
export function isAmountBasedPackageType(packageType?: string | null): boolean {
const normalized = normalizePackageType(packageType);
return normalized === "bottle" || normalized === "tube" || normalized === "liquid_container";
}
export function getPlannerUnitKind(packageType?: string | null): "pills" | "ml" | "units" {
const normalized = normalizePackageType(packageType);
if (normalized === "tube") return "units";
if (normalized === "liquid_container") return "ml";
return "pills";
}
+37 -1
View File
@@ -4,6 +4,7 @@
*/ */
import { getDateLocale, type Language } from "../i18n/translations.js"; import { getDateLocale, type Language } from "../i18n/translations.js";
import { isLiquidContainerPackageType, isTubePackageType } from "./package-profiles.js";
// Legacy type - individual blister schedule (DEPRECATED: use Intake instead) // Legacy type - individual blister schedule (DEPRECATED: use Intake instead)
export type Blister = { usage: number; every: number; start: string }; export type Blister = { usage: number; every: number; start: string };
@@ -13,10 +14,39 @@ export type Intake = {
usage: number; usage: number;
every: number; every: number;
start: string; start: string;
intakeUnit?: "ml" | "tsp" | "tbsp" | null;
takenBy: string | null; // Person taking this specific intake (null = use medication-level takenBy) takenBy: string | null; // Person taking this specific intake (null = use medication-level takenBy)
intakeRemindersEnabled: boolean; intakeRemindersEnabled: boolean;
}; };
const isValidIntakeUnit = (value: unknown): value is "ml" | "tsp" | "tbsp" =>
value === "ml" || value === "tsp" || value === "tbsp";
/**
* Normalize intake usage for stock math.
*
* Stock semantics:
* - tube: no automatic depletion (unknown per-application amount)
* - liquid_container/liquid forms: convert tsp/tbsp to ml
* - others: usage as-is
*/
export function normalizeIntakeUsageForStock(
intake: Pick<Intake, "usage" | "intakeUnit">,
medicationForm?: string | null,
packageType?: string | null
): number {
const usage = Number(intake.usage);
if (!Number.isFinite(usage) || usage <= 0) return 0;
if (isTubePackageType(packageType)) return 0;
const isLiquidStock = isLiquidContainerPackageType(packageType) || medicationForm === "liquid";
if (!isLiquidStock) return usage;
if (intake.intakeUnit === "tsp") return usage * 5;
if (intake.intakeUnit === "tbsp") return usage * 15;
return usage;
}
// ============================================================================= // =============================================================================
// Timezone utilities // Timezone utilities
// ============================================================================= // =============================================================================
@@ -122,7 +152,11 @@ export function getNextScheduledTime(reminderHour: number, tz?: string): Date {
/** Calculate milliseconds until next check at the given reminder hour */ /** Calculate milliseconds until next check at the given reminder hour */
export function getMsUntilNextCheck(reminderHour: number, tz?: string): number { export function getMsUntilNextCheck(reminderHour: number, tz?: string): number {
const next = getNextScheduledTime(reminderHour, tz); const next = getNextScheduledTime(reminderHour, tz);
return next.getTime() - Date.now(); const msUntilNext = next.getTime() - Date.now();
if (msUntilNext <= 0) {
return msUntilNext + 24 * 60 * 60 * 1000;
}
return msUntilNext;
} }
// ============================================================================= // =============================================================================
@@ -195,6 +229,7 @@ export function parseIntakesJson(
usage: typeof intake.usage === "number" ? intake.usage : 0, usage: typeof intake.usage === "number" ? intake.usage : 0,
every: typeof intake.every === "number" ? intake.every : 1, every: typeof intake.every === "number" ? intake.every : 1,
start: typeof intake.start === "string" ? intake.start : new Date().toISOString(), start: typeof intake.start === "string" ? intake.start : new Date().toISOString(),
intakeUnit: isValidIntakeUnit(intake.intakeUnit) ? intake.intakeUnit : null,
takenBy: typeof intake.takenBy === "string" && intake.takenBy.trim() ? intake.takenBy.trim() : null, takenBy: typeof intake.takenBy === "string" && intake.takenBy.trim() ? intake.takenBy.trim() : null,
intakeRemindersEnabled: intakeRemindersEnabled:
typeof intake.intakeRemindersEnabled === "boolean" ? intake.intakeRemindersEnabled : false, typeof intake.intakeRemindersEnabled === "boolean" ? intake.intakeRemindersEnabled : false,
@@ -212,6 +247,7 @@ export function parseIntakesJson(
usage: b.usage, usage: b.usage,
every: b.every, every: b.every,
start: b.start, start: b.start,
intakeUnit: null,
takenBy: null, // Legacy format has no per-intake takenBy takenBy: null, // Legacy format has no per-intake takenBy
intakeRemindersEnabled: medicationIntakeRemindersEnabled ?? false, intakeRemindersEnabled: medicationIntakeRemindersEnabled ?? false,
})); }));
+74 -30
View File
@@ -1,7 +1,7 @@
import * as fs from "node:fs"; import * as fs from "node:fs";
import * as path from "node:path"; import * as path from "node:path";
import { expect, test as setup } from "@playwright/test"; import { expect, test as setup } from "@playwright/test";
import { TEST_USER } from "./fixtures"; import { applyVideoSafetyMode, TEST_USER } from "./fixtures";
const authFile = path.join(import.meta.dirname, ".auth", "user.json"); const authFile = path.join(import.meta.dirname, ".auth", "user.json");
@@ -33,6 +33,8 @@ function isTokenValid(token: string): boolean {
* 4. Log in via the UI. * 4. Log in via the UI.
*/ */
setup("authenticate", async ({ page }) => { setup("authenticate", async ({ page }) => {
await applyVideoSafetyMode(page);
// Create .auth directory if it doesn't exist // Create .auth directory if it doesn't exist
const authDir = path.dirname(authFile); const authDir = path.dirname(authFile);
if (!fs.existsSync(authDir)) { if (!fs.existsSync(authDir)) {
@@ -68,40 +70,82 @@ setup("authenticate", async ({ page }) => {
// Wait for auth container // Wait for auth container
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 }); await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
// ---- 3. Ensure the test user exists ---- // ---- 3. Query auth state to determine login method ----
const baseURL = process.env.PLAYWRIGHT_BASE_URL || "http://localhost:5173"; const baseURL = process.env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
await page.request let formLoginEnabled = true;
.post(`${baseURL}/api/auth/register`, { let oidcEnabled = false;
data: { username: TEST_USER.username, password: TEST_USER.password }, try {
}) const stateRes = await page.request.get(`${baseURL}/api/auth/state`);
.catch(() => {}); if (stateRes.ok()) {
const state = await stateRes.json();
// ---- 4. Log in via UI ---- formLoginEnabled = state.formLoginEnabled !== false;
const usernameField = page.locator("#username"); oidcEnabled = state.oidcEnabled === true;
const passwordField = page.locator("#password");
// Make sure we're on the login form (not register)
const isOnRegister = await page
.locator(".auth-subtitle")
.filter({ hasText: /Create Account/i })
.isVisible()
.catch(() => false);
if (isOnRegister) {
const switchBtn = page.locator("button.auth-link-btn");
if (await switchBtn.isVisible().catch(() => false)) {
await switchBtn.click();
await page.waitForTimeout(500);
} }
} catch {
// Fallback: assume form login is available
} }
await usernameField.clear(); // ---- 4. Ensure the test user exists (only if form login is available) ----
await usernameField.fill(TEST_USER.username); if (formLoginEnabled) {
await passwordField.clear(); await page.request
await passwordField.fill(TEST_USER.password); .post(`${baseURL}/api/auth/register`, {
data: { username: TEST_USER.username, password: TEST_USER.password },
})
.catch(() => {});
}
// Click the submit button (not the SSO button) // ---- 5. Log in via the appropriate method ----
await page.locator('button.auth-submit[type="submit"]').click(); if (formLoginEnabled) {
// Form login path: username/password
const usernameField = page.locator("#username");
const passwordField = page.locator("#password");
// Make sure we're on the login form (not register)
const isOnRegister = await page
.locator(".auth-subtitle")
.filter({ hasText: /Create Account/i })
.isVisible()
.catch(() => false);
if (isOnRegister) {
const switchBtn = page.locator("button.auth-link-btn");
if (await switchBtn.isVisible().catch(() => false)) {
await switchBtn.click();
await page.waitForTimeout(500);
}
}
await usernameField.clear();
await usernameField.fill(TEST_USER.username);
await passwordField.clear();
await passwordField.fill(TEST_USER.password);
// Click the submit button (not the SSO button)
await page.locator('button.auth-submit[type="submit"]').click();
} else if (oidcEnabled) {
// SSO-only path: click the SSO button and let the OIDC provider handle login.
// This requires the OIDC provider to be configured with test credentials
// (e.g. via PLAYWRIGHT_OIDC_USERNAME / PLAYWRIGHT_OIDC_PASSWORD env vars)
// or to auto-approve the test user.
await page.locator("button.sso-btn").click();
// Wait for OIDC redirect and callback — the provider may show its own login form
const oidcUsername = process.env.PLAYWRIGHT_OIDC_USERNAME;
const oidcPassword = process.env.PLAYWRIGHT_OIDC_PASSWORD;
if (oidcUsername && oidcPassword) {
// Fill OIDC provider login form (generic selectors — override if needed)
await page.waitForURL(/.*/, { timeout: 15000 });
const oidcUserField = page.locator('input[name="username"], input[name="login"], input[type="email"]').first();
const oidcPassField = page.locator('input[name="password"], input[type="password"]').first();
if (await oidcUserField.isVisible({ timeout: 10000 }).catch(() => false)) {
await oidcUserField.fill(oidcUsername);
await oidcPassField.fill(oidcPassword);
await page.locator('button[type="submit"]').first().click();
}
}
} else {
throw new Error("No login method available: form login and OIDC are both disabled");
}
// Wait for successful auth — app header should appear // Wait for successful auth — app header should appear
await expect(page.locator("header.hero")).toBeVisible({ timeout: 15000 }); await expect(page.locator("header.hero")).toBeVisible({ timeout: 15000 });
+61 -5
View File
@@ -1,16 +1,28 @@
import { expect, type Page, test } from "@playwright/test"; import { expect, type Page, test } from "@playwright/test";
async function isAuthEnabled(page: Page): Promise<boolean> { interface AuthStateResponse {
authEnabled: boolean;
formLoginEnabled: boolean;
oidcEnabled: boolean;
oidcProviderName: string;
registrationEnabled: boolean;
}
async function getAuthState(page: Page): Promise<AuthStateResponse | null> {
try { try {
const response = await page.request.get("/api/auth/state"); const response = await page.request.get("/api/auth/state");
if (!response.ok()) return true; if (!response.ok()) return null;
const state = await response.json(); return (await response.json()) as AuthStateResponse;
return state?.authEnabled !== false;
} catch { } catch {
return true; return null;
} }
} }
async function isAuthEnabled(page: Page): Promise<boolean> {
const state = await getAuthState(page);
return state?.authEnabled !== false;
}
/** /**
* Authentication E2E Tests * Authentication E2E Tests
* *
@@ -110,4 +122,48 @@ test.describe("Authentication", () => {
const newText = await subtitle.textContent(); const newText = await subtitle.textContent();
expect(newText).not.toBe(initialText); expect(newText).not.toBe(initialText);
}); });
test("should show SSO button when OIDC is enabled", async ({ page }) => {
const state = await getAuthState(page);
test.skip(!state?.authEnabled, "Auth is disabled in this environment");
test.skip(!state?.oidcEnabled, "OIDC is not enabled in this environment");
await page.goto("/");
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
const ssoButton = page.locator("button.sso-btn");
await expect(ssoButton).toBeVisible();
await expect(ssoButton).toContainText(state.oidcProviderName || "SSO");
});
test("should hide form login when formLoginEnabled is false", async ({ page }) => {
const state = await getAuthState(page);
test.skip(!state?.authEnabled, "Auth is disabled in this environment");
test.skip(state?.formLoginEnabled !== false, "Form login is enabled — cannot test hidden state");
await page.goto("/");
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
// Username/password fields should not be visible
await expect(page.locator("#username")).not.toBeVisible();
await expect(page.locator("#password")).not.toBeVisible();
// SSO button should be the only login method
await expect(page.locator("button.sso-btn")).toBeVisible();
});
test("should show both login methods when OIDC and form login are enabled", async ({ page }) => {
const state = await getAuthState(page);
test.skip(!state?.authEnabled, "Auth is disabled in this environment");
test.skip(!state?.oidcEnabled, "OIDC is not enabled");
test.skip(!state?.formLoginEnabled, "Form login is not enabled");
await page.goto("/");
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
// Both login methods visible
await expect(page.locator("#username")).toBeVisible();
await expect(page.locator("#password")).toBeVisible();
await expect(page.locator("button.sso-btn")).toBeVisible();
});
}); });
+4 -4
View File
@@ -65,7 +65,7 @@ test.describe("Dashboard with medications", () => {
test("should show medication overview table with medications", async ({ page }) => { test("should show medication overview table with medications", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
await expect(overviewTable.locator(".table-head")).toBeVisible(); await expect(overviewTable.locator(".table-head")).toBeVisible();
@@ -77,7 +77,7 @@ test.describe("Dashboard with medications", () => {
test("should show status chips in overview table", async ({ page }) => { test("should show status chips in overview table", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// Each medication row should have a status chip // Each medication row should have a status chip
@@ -88,7 +88,7 @@ test.describe("Dashboard with medications", () => {
test("should show stock information in overview", async ({ page }) => { test("should show stock information in overview", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// The Ibuprofen row should show stock info (60 pills minus today's usage = 59) // The Ibuprofen row should show stock info (60 pills minus today's usage = 59)
@@ -202,7 +202,7 @@ test.describe("Dashboard with medications", () => {
test("should open medication detail modal from overview table", async ({ page }) => { test("should open medication detail modal from overview table", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
const medRow = overviewTable.locator(".table-row").filter({ hasText: MED_1 }).first(); const medRow = overviewTable.locator(".table-row").filter({ hasText: MED_1 }).first();
+158 -27
View File
@@ -60,6 +60,29 @@ async function setupAuthMeMock(page: Page): Promise<void> {
} }
} }
/**
* Reduce visual flashing in recorded videos by forcing a dark first paint and
* disabling most animations/transitions in test mode.
*/
export async function applyVideoSafetyMode(page: Page): Promise<void> {
await page.emulateMedia({ reducedMotion: "reduce", colorScheme: "dark" });
await page.addInitScript(() => {
const style = document.createElement("style");
style.id = "pw-video-safety-style";
style.textContent = `
html, body {
background: #111111 !important;
color-scheme: dark !important;
}
*, *::before, *::after {
animation: none !important;
transition: none !important;
}
`;
document.documentElement.appendChild(style);
});
}
/** /**
* Extended test fixture that automatically mocks /auth/me on every page * Extended test fixture that automatically mocks /auth/me on every page
* using user data from the JWT in the stored auth file. * using user data from the JWT in the stored auth file.
@@ -72,6 +95,7 @@ async function setupAuthMeMock(page: Page): Promise<void> {
*/ */
export const test = base.extend<object>({ export const test = base.extend<object>({
page: async ({ page }, use) => { page: async ({ page }, use) => {
await applyVideoSafetyMode(page);
await setupAuthMeMock(page); await setupAuthMeMock(page);
await use(page); await use(page);
}, },
@@ -79,25 +103,43 @@ export const test = base.extend<object>({
/** /**
* Wait for the app to be fully loaded past any loading/initializing screens. * Wait for the app to be fully loaded past any loading/initializing screens.
* Includes a single retry with page reload to handle transient auth failures * Retries up to 2 times with page reload to handle transient auth or
* (e.g. brief race between context setup and cookie application). * rate-limit failures.
*/ */
export async function waitForAppReady(page: Page): Promise<void> { export async function waitForAppReady(page: Page): Promise<void> {
const hero = page.locator("header.hero"); const hero = page.locator("header.hero");
try { for (let attempt = 0; attempt < 3; attempt++) {
await expect(hero).toBeVisible({ timeout: 15000 }); try {
} catch { await expect(hero).toBeVisible({ timeout: 15000 });
// Auth might have failed transiently — reload and retry once return;
await page.reload(); } catch {
await expect(hero).toBeVisible({ timeout: 15000 }); if (attempt === 2) throw new Error("App failed to become ready after 3 attempts");
// Check for rate-limit error displayed in UI
const rateLimited = await page
.locator("text=rate limit, text=429, text=too many")
.first()
.isVisible()
.catch(() => false);
if (rateLimited) {
// Wait longer before retrying if rate-limited
await page.waitForTimeout(5000);
}
await page.reload();
}
} }
} }
/** /**
* Navigate to a page and wait for it to be ready. * Navigate to a page and wait for it to be ready.
* Handles transient navigation failures with a single retry.
*/ */
export async function navigateTo(page: Page, path: string): Promise<void> { export async function navigateTo(page: Page, path: string): Promise<void> {
await page.goto(path); const response = await page.goto(path);
if (response && response.status() === 429) {
// Rate-limited — wait and retry once
await page.waitForTimeout(5000);
await page.goto(path);
}
await waitForAppReady(page); await waitForAppReady(page);
await page.waitForLoadState("networkidle"); await page.waitForLoadState("networkidle");
} }
@@ -135,7 +177,9 @@ export { expect };
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
const API_BASE = process.env.PLAYWRIGHT_BASE_URL || "http://localhost:5173"; const API_BASE = process.env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
function getAuthCookie(): string | null { let cachedAuthCookie: string | null = null;
function readAuthCookieFromFile(): string | null {
try { try {
const state = JSON.parse(fs.readFileSync(authFile, "utf-8")); const state = JSON.parse(fs.readFileSync(authFile, "utf-8"));
return state.cookies?.find((c: { name: string }) => c.name === "access_token")?.value ?? null; return state.cookies?.find((c: { name: string }) => c.name === "access_token")?.value ?? null;
@@ -144,6 +188,49 @@ function getAuthCookie(): string | null {
} }
} }
function extractCookieValue(setCookieHeaders: string[], name: string): string | null {
for (const header of setCookieHeaders) {
const [pair] = header.split(";");
if (!pair) continue;
const [cookieName, ...valueParts] = pair.split("=");
if (cookieName?.trim() !== name) continue;
const value = valueParts.join("=").trim();
if (value) return value;
}
return null;
}
async function refreshAuthCookieViaLogin(): Promise<string | null> {
const res = await fetch(`${API_BASE}/api/auth/login`, {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
username: TEST_USER.username,
password: TEST_USER.password,
rememberMe: false,
}),
});
if (!res.ok) return null;
const getSetCookie = (res.headers as Headers & { getSetCookie?: () => string[] }).getSetCookie;
const setCookieHeaders = typeof getSetCookie === "function" ? getSetCookie.call(res.headers) : [];
const fallback = res.headers.get("set-cookie");
if (fallback) setCookieHeaders.push(fallback);
const accessToken = extractCookieValue(setCookieHeaders, "access_token");
if (accessToken) {
cachedAuthCookie = accessToken;
}
return accessToken;
}
function getAuthCookie(): string | null {
if (cachedAuthCookie) return cachedAuthCookie;
cachedAuthCookie = readAuthCookieFromFile();
return cachedAuthCookie;
}
/** Typed medication response (subset of fields we care about) */ /** Typed medication response (subset of fields we care about) */
export interface TestMedication { export interface TestMedication {
id: number; id: number;
@@ -172,12 +259,14 @@ export async function createMedicationViaAPI(data: {
takenBy?: string[]; takenBy?: string[];
notes?: string; notes?: string;
expiryDate?: string; expiryDate?: string;
packageType?: "blister" | "bottle"; packageType?: "blister" | "bottle" | "tube" | "liquid_container";
medicationForm?: "capsule" | "tablet" | "liquid" | "topical";
packCount?: number; packCount?: number;
blistersPerPack?: number; blistersPerPack?: number;
pillsPerBlister?: number; pillsPerBlister?: number;
looseTablets?: number; looseTablets?: number;
totalPills?: number; totalPills?: number;
packageAmountValue?: number;
intakeRemindersEnabled?: boolean; intakeRemindersEnabled?: boolean;
intakes?: { intakes?: {
usage: number; usage: number;
@@ -187,16 +276,30 @@ export async function createMedicationViaAPI(data: {
takenBy?: string | null; takenBy?: string | null;
}[]; }[];
}): Promise<TestMedication> { }): Promise<TestMedication> {
const token = getAuthCookie(); let token = getAuthCookie();
const isBottle = data.packageType === "bottle"; const packageType = data.packageType ?? "blister";
const isAmountBased = packageType === "bottle" || packageType === "tube" || packageType === "liquid_container";
let defaultMedicationForm: "capsule" | "tablet" | "liquid" | "topical" = "tablet";
if (packageType === "tube") {
defaultMedicationForm = "topical";
} else if (packageType === "liquid_container") {
defaultMedicationForm = "liquid";
}
const medicationForm = data.medicationForm ?? defaultMedicationForm;
const packageAmountValue =
data.packageAmountValue ??
(packageType === "tube" || packageType === "liquid_container" ? Math.max(1, data.totalPills ?? 30) : 0);
const body = { const body = {
packageType: isBottle ? "bottle" : "blister", packageType,
packCount: isBottle ? 1 : (data.packCount ?? 1), medicationForm,
blistersPerPack: isBottle ? 1 : (data.blistersPerPack ?? 1), packCount: packageType === "tube" ? 1 : (data.packCount ?? 1),
pillsPerBlister: isBottle ? 1 : (data.pillsPerBlister ?? 10), blistersPerPack: isAmountBased ? 1 : (data.blistersPerPack ?? 1),
// For bottles: looseTablets IS the current stock. Default to totalPills if not specified. pillsPerBlister: isAmountBased ? 1 : (data.pillsPerBlister ?? 10),
looseTablets: isBottle ? (data.looseTablets ?? data.totalPills ?? 0) : (data.looseTablets ?? 0), // Amount-based packages use looseTablets as current stock.
totalPills: isBottle ? (data.totalPills ?? null) : null, looseTablets: isAmountBased ? (data.looseTablets ?? data.totalPills ?? 0) : (data.looseTablets ?? 0),
totalPills: isAmountBased ? (data.totalPills ?? null) : null,
packageAmountValue,
packageAmountUnit: packageType === "tube" ? "g" : "ml",
intakes: [ intakes: [
{ {
usage: 1, usage: 1,
@@ -219,6 +322,10 @@ export async function createMedicationViaAPI(data: {
}, },
body: JSON.stringify(body), body: JSON.stringify(body),
}); });
if (res.status === 401) {
token = await refreshAuthCookieViaLogin();
if (token) continue;
}
if (res.status === 429) { if (res.status === 429) {
// Rate limited — exponential backoff: 3s, 6s, 9s, 12s, 15s // Rate limited — exponential backoff: 3s, 6s, 9s, 12s, 15s
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1))); await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
@@ -235,13 +342,25 @@ export async function createMedicationViaAPI(data: {
/** /**
* Delete a medication via the backend API. * Delete a medication via the backend API.
* Includes retry for rate-limited responses.
*/ */
export async function deleteMedicationViaAPI(id: number): Promise<void> { export async function deleteMedicationViaAPI(id: number): Promise<void> {
const token = getAuthCookie(); let token = getAuthCookie();
await fetch(`${API_BASE}/api/medications/${id}`, { for (let attempt = 0; attempt < 3; attempt++) {
method: "DELETE", const res = await fetch(`${API_BASE}/api/medications/${id}`, {
headers: token ? { Cookie: `access_token=${token}` } : {}, method: "DELETE",
}); headers: token ? { Cookie: `access_token=${token}` } : {},
});
if (res.status === 401) {
token = await refreshAuthCookieViaLogin();
if (token) continue;
}
if (res.status === 429) {
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
continue;
}
return;
}
} }
/** /**
@@ -249,11 +368,15 @@ export async function deleteMedicationViaAPI(id: number): Promise<void> {
* Includes retry logic for rate-limited responses. * Includes retry logic for rate-limited responses.
*/ */
export async function deleteAllMedicationsViaAPI(): Promise<void> { export async function deleteAllMedicationsViaAPI(): Promise<void> {
const token = getAuthCookie(); let token = getAuthCookie();
for (let attempt = 0; attempt < 3; attempt++) { for (let attempt = 0; attempt < 3; attempt++) {
const res = await fetch(`${API_BASE}/api/medications`, { const res = await fetch(`${API_BASE}/api/medications`, {
headers: token ? { Cookie: `access_token=${token}` } : {}, headers: token ? { Cookie: `access_token=${token}` } : {},
}); });
if (res.status === 401) {
token = await refreshAuthCookieViaLogin();
if (token) continue;
}
if (res.status === 429) { if (res.status === 429) {
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1))); await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
continue; continue;
@@ -266,6 +389,10 @@ export async function deleteAllMedicationsViaAPI(): Promise<void> {
method: "DELETE", method: "DELETE",
headers: token ? { Cookie: `access_token=${token}` } : {}, headers: token ? { Cookie: `access_token=${token}` } : {},
}); });
if (delRes.status === 401) {
token = await refreshAuthCookieViaLogin();
if (token) continue;
}
if (delRes.status === 429) { if (delRes.status === 429) {
await new Promise((r) => setTimeout(r, 3000)); await new Promise((r) => setTimeout(r, 3000));
continue; continue;
@@ -282,7 +409,7 @@ export async function deleteAllMedicationsViaAPI(): Promise<void> {
* Requires a medication with takenBy to exist first. * Requires a medication with takenBy to exist first.
*/ */
export async function createShareTokenViaAPI(takenBy: string, scheduleDays = 30): Promise<TestShareToken> { export async function createShareTokenViaAPI(takenBy: string, scheduleDays = 30): Promise<TestShareToken> {
const token = getAuthCookie(); let token = getAuthCookie();
for (let attempt = 0; attempt < 5; attempt++) { for (let attempt = 0; attempt < 5; attempt++) {
const res = await fetch(`${API_BASE}/api/share`, { const res = await fetch(`${API_BASE}/api/share`, {
method: "POST", method: "POST",
@@ -292,6 +419,10 @@ export async function createShareTokenViaAPI(takenBy: string, scheduleDays = 30)
}, },
body: JSON.stringify({ takenBy, scheduleDays }), body: JSON.stringify({ takenBy, scheduleDays }),
}); });
if (res.status === 401) {
token = await refreshAuthCookieViaLogin();
if (token) continue;
}
if (res.status === 429) { if (res.status === 429) {
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1))); await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
continue; continue;
+38 -2
View File
@@ -26,7 +26,7 @@ async function fillAndSaveMedication(
opts: { opts: {
name: string; name: string;
genericName?: string; genericName?: string;
packageType?: "blister" | "bottle"; packageType?: "blister" | "bottle" | "tube" | "liquid_container";
packs?: string; packs?: string;
blistersPerPack?: string; blistersPerPack?: string;
pillsPerBlister?: string; pillsPerBlister?: string;
@@ -56,6 +56,18 @@ async function fillAndSaveMedication(
if (opts.totalCapacity) if (opts.totalCapacity)
await form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i).fill(opts.totalCapacity); await form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i).fill(opts.totalCapacity);
if (opts.currentPills) await form.getByLabel(/(Current Pills|form\.currentPills)/i).fill(opts.currentPills); if (opts.currentPills) await form.getByLabel(/(Current Pills|form\.currentPills)/i).fill(opts.currentPills);
} else if (opts.packageType === "tube") {
await packageTypeSelect.selectOption("tube");
await page.getByRole("tab", { name: /Package/i }).click();
if (opts.totalCapacity) {
await form.getByLabel(/(Amount per tube|form\.packageAmountPerTube)/i).fill(opts.totalCapacity);
}
} else if (opts.packageType === "liquid_container") {
await packageTypeSelect.selectOption("liquid_container");
await page.getByRole("tab", { name: /Package/i }).click();
if (opts.totalCapacity) {
await form.getByLabel(/(Package amount|form\.packageAmount)/i).fill(opts.totalCapacity);
}
} else { } else {
await packageTypeSelect.selectOption("blister"); await packageTypeSelect.selectOption("blister");
await page.getByRole("tab", { name: /Package/i }).click(); await page.getByRole("tab", { name: /Package/i }).click();
@@ -83,7 +95,11 @@ async function fillAndSaveMedication(
await form.getByRole("button", { name: /(Intake|form\.blisters\.addIntake)/i }).click(); await form.getByRole("button", { name: /(Intake|form\.blisters\.addIntake)/i }).click();
} }
const row = form.locator(".blister-row").nth(i); const row = form.locator(".blister-row").nth(i);
await row.getByLabel(/(Usage \(pills\)|form\.blisters\.usage)/i).fill(intakes[i].usage); await row
.getByLabel(
/(Usage \((pills|tablets|capsules|ml|applications)\)|form\.blisters\.(usage|usageTablets|usageCapsules|usageMl|usageApplication))/i
)
.fill(intakes[i].usage);
await row.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i).fill(intakes[i].every); await row.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i).fill(intakes[i].every);
} }
@@ -194,6 +210,26 @@ test.describe("Medication CRUD", () => {
}); });
}); });
test("should create a tube medication via the form", async ({ page }) => {
await navigateTo(page, "/medications");
await fillAndSaveMedication(page, {
name: "Test Tube Cream",
packageType: "tube",
totalCapacity: "50",
});
});
test("should create a liquid-container medication via the form", async ({ page }) => {
await navigateTo(page, "/medications");
await fillAndSaveMedication(page, {
name: "Test Liquid Syrup",
packageType: "liquid_container",
totalCapacity: "120",
});
});
test("should create medication with notes and expiry date", async ({ page }) => { test("should create medication with notes and expiry date", async ({ page }) => {
await navigateTo(page, "/medications"); await navigateTo(page, "/medications");
+17 -8
View File
@@ -233,7 +233,7 @@ test.describe("Medication Editing", () => {
// Change intake from 1 pill daily to 2 pills every 7 days // Change intake from 1 pill daily to 2 pills every 7 days
const intakeRow = page.locator(".blister-row").first(); const intakeRow = page.locator(".blister-row").first();
const usageField = intakeRow.getByLabel(/(Usage \(pills\)|form\.blisters\.usage)/i); const usageField = intakeRow.getByLabel(/(Usage|form\.blisters\.usage)/i);
const everyField = intakeRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i); const everyField = intakeRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i);
await usageField.fill("2"); await usageField.fill("2");
@@ -247,7 +247,7 @@ test.describe("Medication Editing", () => {
// Verify the changes persisted // Verify the changes persisted
await clickEditMed(page, "Edit Intake Med"); await clickEditMed(page, "Edit Intake Med");
const savedRow = page.locator(".blister-row").first(); const savedRow = page.locator(".blister-row").first();
await expect(savedRow.getByLabel(/(Usage \(pills\)|form\.blisters\.usage)/i)).toHaveValue("2"); await expect(savedRow.getByLabel(/(Usage|form\.blisters\.usage)/i)).toHaveValue("2");
await expect(savedRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i)).toHaveValue("7"); await expect(savedRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i)).toHaveValue("7");
}); });
@@ -279,7 +279,7 @@ test.describe("Medication Editing", () => {
// Fill the new intake row // Fill the new intake row
const secondRow = page.locator(".blister-row").nth(1); const secondRow = page.locator(".blister-row").nth(1);
await secondRow.getByLabel(/(Usage \(pills\)|form\.blisters\.usage)/i).fill("0.5"); await secondRow.getByLabel(/(Usage|form\.blisters\.usage)/i).fill("0.5");
await secondRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i).fill("7"); await secondRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i).fill("7");
await saveEditAndVerify(page, "Add Intake Med"); await saveEditAndVerify(page, "Add Intake Med");
@@ -329,7 +329,7 @@ test.describe("Medication Editing", () => {
} }
}); });
test("should change package type between blister and bottle", async ({ page }) => { test("should change package type across all supported profiles", async ({ page }) => {
createdMeds.push( createdMeds.push(
await createMedicationViaAPI({ await createMedicationViaAPI({
name: "PackType Change Med", name: "PackType Change Med",
@@ -357,15 +357,24 @@ test.describe("Medication Editing", () => {
await packageSelect.selectOption("bottle"); await packageSelect.selectOption("bottle");
await page.getByRole("tab", { name: /Package/i }).click(); await page.getByRole("tab", { name: /Package/i }).click();
await expect(form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i)).toBeVisible(); await expect(form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i)).toBeVisible();
await page.getByRole("tab", { name: /General/i }).click();
// Fill bottle-specific fields // Switch to tube
await form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i).fill("120"); await packageSelect.selectOption("tube");
await page.getByRole("tab", { name: /Package/i }).click();
await expect(form.getByLabel(/(Amount per tube|form\.packageAmountPerTube)/i)).toBeVisible();
await page.getByRole("tab", { name: /General/i }).click();
// Switch to liquid container and persist this final state
await packageSelect.selectOption("liquid_container");
await page.getByRole("tab", { name: /Package/i }).click();
await expect(form.getByLabel(/(Package amount|form\.packageAmount)/i)).toBeVisible();
await saveEditAndVerify(page, "PackType Change Med"); await saveEditAndVerify(page, "PackType Change Med");
// Verify it's still a bottle after reload // Verify final package type persisted
await clickEditMed(page, "PackType Change Med"); await clickEditMed(page, "PackType Change Med");
await expect(page.locator("select.package-type-select")).toHaveValue("bottle"); await expect(page.locator("select.package-type-select")).toHaveValue("liquid_container");
}); });
test("should edit multiple fields at once (name, notes, generic, taken-by)", async ({ page }) => { test("should edit multiple fields at once (name, notes, generic, taken-by)", async ({ page }) => {
+193
View File
@@ -0,0 +1,193 @@
import { authFile, createMedicationViaAPI, deleteAllMedicationsViaAPI, expect, navigateTo, test } from "./fixtures";
/**
* Medication Lifecycle Integration Tests
*
* End-to-end workflows that verify changes propagate across pages:
* create verify on medications check in planner check in schedule edit delete
*/
test.describe("Medication lifecycle", () => {
test.use({ storageState: authFile });
test.describe.configure({ timeout: 90000 });
const MED_NAME = "Lifecycle TestMed";
const MED_EDITED = "Lifecycle Edited";
test.beforeAll(async () => {
await deleteAllMedicationsViaAPI();
});
test.afterAll(async () => {
await deleteAllMedicationsViaAPI();
});
test("create medication via API and verify it appears on all pages", async ({ page }) => {
const todayMorning = (() => {
const d = new Date();
d.setHours(8, 0, 0, 0);
const pad = (n: number) => n.toString().padStart(2, "0");
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
})();
// Step 1: Create medication
const created = await createMedicationViaAPI({
name: MED_NAME,
packageType: "blister",
packCount: 1,
blistersPerPack: 2,
pillsPerBlister: 10,
looseTablets: 0,
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
});
expect(created.id).toBeTruthy();
// Step 2: Verify on medications page
await navigateTo(page, "/medications");
await expect(page.getByText(MED_NAME).first()).toBeVisible({ timeout: 10000 });
// Step 3: Verify in planner
await navigateTo(page, "/planner");
await page.waitForLoadState("networkidle");
await page.locator('form.planner button[type="submit"]').click();
await expect(page.locator(".table")).toBeVisible({ timeout: 15000 });
await expect(page.locator(".table").getByText(MED_NAME)).toBeVisible();
// Step 4: Verify in schedule
await navigateTo(page, "/schedule");
await expect(page.getByText(MED_NAME).first()).toBeVisible({ timeout: 10000 });
});
test("edit medication name via UI and verify update propagates", async ({ page }) => {
await deleteAllMedicationsViaAPI();
const todayMorning = (() => {
const d = new Date();
d.setHours(8, 0, 0, 0);
const pad = (n: number) => n.toString().padStart(2, "0");
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
})();
// Create a fresh medication for this test
await createMedicationViaAPI({
name: MED_NAME,
packageType: "blister",
packCount: 1,
blistersPerPack: 2,
pillsPerBlister: 10,
looseTablets: 0,
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
});
// Navigate to medications page
await navigateTo(page, "/medications");
await expect(page.getByText(MED_NAME).first()).toBeVisible({ timeout: 10000 });
// Open edit view from medication row actions
const medRow = page.locator(".med-row").filter({ hasText: MED_NAME });
await expect(medRow.first()).toBeVisible({ timeout: 10000 });
await medRow.first().locator("button.info").click();
await expect(page.locator("h2").filter({ hasText: /(Edit(:| (entry|medication))|form\.editEntry)/i })).toBeVisible({
timeout: 5000,
});
// Update the name
const form = page.locator("form.form-grid:visible").first();
const nameInput = form.getByLabel(/(Commercial Name|Name|form\.name)/i).first();
await nameInput.fill(MED_EDITED);
// Save
const submitButton = form.locator('button[type="submit"]').first();
await expect(submitButton).toBeEnabled({ timeout: 5000 });
await submitButton.click();
// Wait for modal to close or save to complete
await page.waitForLoadState("networkidle");
// Verify edited name appears on medications page
await navigateTo(page, "/medications");
await expect(page.getByText(MED_EDITED).first()).toBeVisible({ timeout: 10000 });
// Old name should no longer appear
await expect(page.locator(".med-row").filter({ hasText: MED_NAME })).toHaveCount(0, { timeout: 5000 });
});
test("delete medication via API and verify it disappears from all pages", async ({ page }) => {
const todayMorning = (() => {
const d = new Date();
d.setHours(8, 0, 0, 0);
const pad = (n: number) => n.toString().padStart(2, "0");
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
})();
// Create and then delete
await deleteAllMedicationsViaAPI();
await createMedicationViaAPI({
name: MED_NAME,
packageType: "blister",
packCount: 1,
blistersPerPack: 1,
pillsPerBlister: 5,
looseTablets: 0,
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
});
// Verify it exists first
await navigateTo(page, "/medications");
await expect(page.getByText(MED_NAME)).toBeVisible({ timeout: 10000 });
// Delete via API
await deleteAllMedicationsViaAPI();
// Verify gone from medications page
await navigateTo(page, "/medications");
await expect(page.getByText(MED_NAME)).not.toBeVisible({ timeout: 5000 });
// Verify planner shows no results for this med
await navigateTo(page, "/planner");
await page.waitForLoadState("networkidle");
await page.locator('form.planner button[type="submit"]').click();
// Either no table or table without the medication name
const table = page.locator(".table");
const tableVisible = await table.isVisible().catch(() => false);
if (tableVisible) {
await expect(table.getByText(MED_NAME)).not.toBeVisible({ timeout: 3000 });
}
});
test("medication with multiple intakes shows all schedule entries", async ({ page }) => {
const todayMorning = (() => {
const d = new Date();
d.setHours(8, 0, 0, 0);
const pad = (n: number) => n.toString().padStart(2, "0");
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
})();
const todayEvening = (() => {
const d = new Date();
d.setHours(20, 0, 0, 0);
const pad = (n: number) => n.toString().padStart(2, "0");
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
})();
await deleteAllMedicationsViaAPI();
await createMedicationViaAPI({
name: "MultiIntake Med",
packageType: "blister",
packCount: 1,
blistersPerPack: 2,
pillsPerBlister: 10,
looseTablets: 0,
intakes: [
{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false },
{ usage: 2, every: 1, start: todayEvening, intakeRemindersEnabled: false },
],
});
// Verify schedule shows this medication
await navigateTo(page, "/schedule");
await expect(page.getByText("MultiIntake Med").first()).toBeVisible({ timeout: 10000 });
// The medication should appear at least twice (morning + evening)
const medEntries = page.getByText("MultiIntake Med");
expect(await medEntries.count()).toBeGreaterThanOrEqual(2);
});
});
+7 -15
View File
@@ -87,25 +87,17 @@ test.describe("Medications Page", () => {
expect(hasPacks || hasTotal).toBeTruthy(); expect(hasPacks || hasTotal).toBeTruthy();
}); });
test("should toggle package type between blister and bottle", async ({ page }) => { test("should expose all supported package type options", async ({ page }) => {
await openMedicationForm(page); await openMedicationForm(page);
const form = visibleMedForm(page); const form = visibleMedForm(page);
await page.getByRole("tab", { name: /Package/i }).click(); const packageSelect = form.locator("select.package-type-select");
await expect(packageSelect).toBeVisible();
// Find the package type radio buttons or selector const optionValues = await packageSelect
const blisterOption = form.getByText(/(Blister Pack|form\.packageType\.blister)/i); .locator("option")
const bottleOption = form.getByText(/(Pill Bottle|form\.packageType\.bottle)/i); .evaluateAll((options) => options.map((option) => (option as HTMLOptionElement).value));
if (await blisterOption.isVisible().catch(() => false)) { expect(optionValues).toEqual(expect.arrayContaining(["blister", "bottle", "tube", "liquid_container"]));
// Switch to bottle
await bottleOption.click();
// Bottle-specific fields should appear
await expect(form.getByLabel(/(Total Capacity|form\.totalCapacity)/i)).toBeVisible();
// Switch back to blister
await blisterOption.click();
await expect(form.getByLabel(/(Blisters per pack|form\.blistersPerPack)/i)).toBeVisible();
}
}); });
test("should have intake schedule with add button", async ({ page }) => { test("should have intake schedule with add button", async ({ page }) => {
+98
View File
@@ -0,0 +1,98 @@
import { authFile, createMedicationViaAPI, deleteAllMedicationsViaAPI, expect, navigateTo, test } from "./fixtures";
/**
* Performance Tests
*
* Verify the schedule timeline and planner render within acceptable
* time limits when many medications exist.
*/
test.describe("Performance with many medications", () => {
test.use({ storageState: authFile });
test.describe.configure({ timeout: 120000 });
const MED_COUNT = 20;
const MED_PREFIX = "PerfTest Med";
test.beforeAll(async () => {
await deleteAllMedicationsViaAPI();
const todayMorning = (() => {
const d = new Date();
d.setHours(8, 0, 0, 0);
const pad = (n: number) => n.toString().padStart(2, "0");
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
})();
// Create medications sequentially (API rate limits prevent parallel)
for (let i = 1; i <= MED_COUNT; i++) {
await createMedicationViaAPI({
name: `${MED_PREFIX} ${i}`,
packageType: "blister",
packCount: 1,
blistersPerPack: 2,
pillsPerBlister: 10,
looseTablets: 0,
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
});
}
});
test.afterAll(async () => {
await deleteAllMedicationsViaAPI();
});
test("schedule page renders within 10 seconds with 20 medications", async ({ page }) => {
const start = Date.now();
await navigateTo(page, "/schedule");
// Wait for schedule entries to render
const scheduleEntries = page.locator(".schedule-entry, .timeline-entry, .card");
await expect(scheduleEntries.first()).toBeVisible({ timeout: 15000 });
const renderTime = Date.now() - start;
// Verify all medications appear
for (let i = 1; i <= MED_COUNT; i++) {
await expect(page.getByText(`${MED_PREFIX} ${i}`).first()).toBeVisible({ timeout: 5000 });
}
// Goal: render under 10 seconds
expect(renderTime).toBeLessThan(10000);
});
test("medications page renders within 10 seconds with 20 medications", async ({ page }) => {
const start = Date.now();
await navigateTo(page, "/medications");
// Wait for medication cards to render
const medEntries = page.locator(".medication-card, .card, .table-row");
await expect(medEntries.first()).toBeVisible({ timeout: 15000 });
const renderTime = Date.now() - start;
// Verify count — all 20 should be visible
for (let i = 1; i <= MED_COUNT; i++) {
await expect(page.getByText(`${MED_PREFIX} ${i}`).first()).toBeVisible({ timeout: 5000 });
}
expect(renderTime).toBeLessThan(10000);
});
test("planner calculates within 15 seconds with 20 medications", async ({ page }) => {
await navigateTo(page, "/planner");
const start = Date.now();
await page.waitForLoadState("networkidle");
await page.locator('form.planner button[type="submit"]').click();
await expect(page.locator(".table")).toBeVisible({ timeout: 20000 });
const calcTime = Date.now() - start;
// All medications should appear in the results
const rows = page.locator(".table .table-row");
expect(await rows.count()).toBeGreaterThanOrEqual(MED_COUNT);
// Goal: calculate and render under 15 seconds
expect(calcTime).toBeLessThan(15000);
});
});
+50 -9
View File
@@ -106,7 +106,7 @@ test.describe("Planner with medications", () => {
expect(await statusChips.count()).toBeGreaterThanOrEqual(2); expect(await statusChips.count()).toBeGreaterThanOrEqual(2);
}); });
test("should show usage data in results rows", async ({ page }) => { test("should show correct usage values in results rows", async ({ page }) => {
await navigateTo(page, "/planner"); await navigateTo(page, "/planner");
await calculatePlanner(page); await calculatePlanner(page);
@@ -116,10 +116,15 @@ test.describe("Planner with medications", () => {
const rows = resultsTable.locator(".table-row"); const rows = resultsTable.locator(".table-row");
expect(await rows.count()).toBeGreaterThanOrEqual(2); expect(await rows.count()).toBeGreaterThanOrEqual(2);
const firstRowText = await rows.first().textContent(); // Each medication has usage=1, every=1 → plannerUsage should reflect the period
expect(firstRowText).toBeTruthy(); // Verify the usage column contains a numeric <strong> value and "pill(s)"
// Check for "pill" (matches both "pill" and "pills") for (const row of await rows.all()) {
expect(firstRowText!.toLowerCase()).toContain("pill"); const usageCell = row.locator("[data-label]").nth(1); // Usage is 2nd column
const usageStrong = usageCell.locator("strong");
await expect(usageStrong).toBeVisible();
const usageText = await usageStrong.textContent();
expect(Number(usageText)).toBeGreaterThan(0);
}
}); });
test("should show danger status for low-stock medication over 90 days", async ({ page }) => { test("should show danger status for low-stock medication over 90 days", async ({ page }) => {
@@ -139,9 +144,16 @@ test.describe("Planner with medications", () => {
const resultsTable = page.locator(".table"); const resultsTable = page.locator(".table");
await expect(resultsTable).toBeVisible({ timeout: 10000 }); await expect(resultsTable).toBeVisible({ timeout: 10000 });
// Low-stock med (3 pills) should have a danger chip over 90 days // Low-stock med (3 pills, usage 1/day, 90 days) should have danger status
const dangerChips = resultsTable.locator(".status-chip.danger"); const dangerChips = resultsTable.locator(".status-chip.danger");
expect(await dangerChips.count()).toBeGreaterThanOrEqual(1); expect(await dangerChips.count()).toBeGreaterThanOrEqual(1);
// Find the low-stock med row and verify its usage value ~90 pills
const lowStockRow = resultsTable.locator(".table-row", { hasText: MED_LOW });
await expect(lowStockRow).toBeVisible();
const lowUsage = await lowStockRow.locator("[data-label] strong").first().textContent();
expect(Number(lowUsage)).toBeGreaterThanOrEqual(85); // ~90 pills needed
expect(Number(lowUsage)).toBeLessThanOrEqual(95);
}); });
test("should show Enough status for well-stocked medication over 7 days", async ({ page }) => { test("should show Enough status for well-stocked medication over 7 days", async ({ page }) => {
@@ -161,9 +173,16 @@ test.describe("Planner with medications", () => {
const resultsTable = page.locator(".table"); const resultsTable = page.locator(".table");
await expect(resultsTable).toBeVisible({ timeout: 10000 }); await expect(resultsTable).toBeVisible({ timeout: 10000 });
// With 60 pills and 7-day range, high-stock should be "Enough" // High-stock med (60 pills, usage 1/day, 7 days → needs ~7, has 60) should be "Enough"
const successChips = resultsTable.locator(".status-chip.success"); const highStockRow = resultsTable.locator(".table-row", { hasText: MED_HIGH });
expect(await successChips.count()).toBeGreaterThanOrEqual(1); await expect(highStockRow).toBeVisible();
const highStatus = highStockRow.locator(".status-chip.success");
await expect(highStatus).toBeVisible();
// Verify usage is ~7 pills for the 7-day range
const highUsage = await highStockRow.locator("[data-label] strong").first().textContent();
expect(Number(highUsage)).toBeGreaterThanOrEqual(5);
expect(Number(highUsage)).toBeLessThanOrEqual(10);
}); });
test("should show table header with correct columns", async ({ page }) => { test("should show table header with correct columns", async ({ page }) => {
@@ -180,6 +199,28 @@ test.describe("Planner with medications", () => {
await expect(tableHead.getByText(/Status/i)).toBeVisible(); await expect(tableHead.getByText(/Status/i)).toBeVisible();
}); });
test("should display available stock for each medication", async ({ page }) => {
await navigateTo(page, "/planner");
await calculatePlanner(page);
const resultsTable = page.locator(".table");
await expect(resultsTable).toBeVisible({ timeout: 10000 });
// High-stock med should show a blister + loose-pill stock breakdown
const highStockRow = resultsTable.locator(".table-row", { hasText: MED_HIGH });
await expect(highStockRow).toBeVisible();
const highStockText = await highStockRow.textContent();
expect(highStockText).toMatch(/\d+\s*(blisters|Blister)/i);
expect(highStockText).toMatch(/\d+\s*(pill|pills|Tablette|Tabletten)/i);
// Low-stock med: 1 pack × 1 blister × 3 pills = 3 pills = 0 full blisters + 3 loose
const lowStockRow = resultsTable.locator(".table-row", { hasText: MED_LOW });
await expect(lowStockRow).toBeVisible();
const lowStockText = await lowStockRow.textContent();
// Should show 3 loose pills
expect(lowStockText).toMatch(/3\s*(pill|pills|Tablette|Tabletten)/i);
});
test("should reset form and clear results", async ({ page }) => { test("should reset form and clear results", async ({ page }) => {
await navigateTo(page, "/planner"); await navigateTo(page, "/planner");
await calculatePlanner(page); await calculatePlanner(page);
-11
View File
@@ -224,15 +224,4 @@ test.describe("Schedule with medications", () => {
expect(await takeButtons.count()).toBeGreaterThanOrEqual(1); expect(await takeButtons.count()).toBeGreaterThanOrEqual(1);
} }
}); });
test("should show medication names in timeline rows", async ({ page }) => {
await navigateTo(page, "/dashboard");
await page.waitForLoadState("networkidle");
const todayBlock = page.locator(".day-block.today");
await expect(todayBlock).toBeVisible({ timeout: 15000 });
const medNames = todayBlock.locator(".med-name");
expect(await medNames.count()).toBeGreaterThanOrEqual(1);
});
}); });
+1 -2
View File
@@ -150,8 +150,7 @@ test.describe("Schedule Timeline", () => {
test("should show overview table with stock status", async ({ page }) => { test("should show overview table with stock status", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
// Overview table has class .table.table-7 const overviewTable = page.locator(".dashboard-overview-section .table").first();
const overviewTable = page.locator(".table.table-7");
await expect(overviewTable).toBeVisible(); await expect(overviewTable).toBeVisible();
await expect(overviewTable.locator(".table-head")).toBeVisible(); await expect(overviewTable.locator(".table-head")).toBeVisible();
}); });
+3 -3
View File
@@ -72,7 +72,7 @@ test.describe("Share Schedule", () => {
test("should show taken-by badges on dashboard overview table", async ({ page }) => { test("should show taken-by badges on dashboard overview table", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// Alice's medication should show "Alice" badge // Alice's medication should show "Alice" badge
@@ -253,7 +253,7 @@ test.describe("Share Schedule", () => {
test("should show notes icon on dashboard for medication with notes", async ({ page }) => { test("should show notes icon on dashboard for medication with notes", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// Alice's med has notes — should show the 📝 icon // Alice's med has notes — should show the 📝 icon
@@ -265,7 +265,7 @@ test.describe("Share Schedule", () => {
test("should show notes in medication detail modal", async ({ page }) => { test("should show notes in medication detail modal", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// Click on Alice's med to open detail modal // Click on Alice's med to open detail modal
+10 -10
View File
@@ -125,7 +125,7 @@ test.describe("Stock Status Levels", () => {
test("should show all medications in overview table", async ({ page }) => { test("should show all medications in overview table", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// All 5 medications should appear // All 5 medications should appear
@@ -139,7 +139,7 @@ test.describe("Stock Status Levels", () => {
test("should show High status chip for well-stocked medication", async ({ page }) => { test("should show High status chip for well-stocked medication", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// High stock med row should have a .status-chip.high // High stock med row should have a .status-chip.high
@@ -151,7 +151,7 @@ test.describe("Stock Status Levels", () => {
test("should show Normal status chip for moderate stock medication", async ({ page }) => { test("should show Normal status chip for moderate stock medication", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
const normalRow = overviewTable.locator(".table-row").filter({ hasText: MED_NORMAL }); const normalRow = overviewTable.locator(".table-row").filter({ hasText: MED_NORMAL });
@@ -162,7 +162,7 @@ test.describe("Stock Status Levels", () => {
test("should show Warning status chip for low stock medication", async ({ page }) => { test("should show Warning status chip for low stock medication", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
const lowRow = overviewTable.locator(".table-row").filter({ hasText: MED_LOW }); const lowRow = overviewTable.locator(".table-row").filter({ hasText: MED_LOW });
@@ -173,7 +173,7 @@ test.describe("Stock Status Levels", () => {
test("should show Danger status chip for critical stock medication", async ({ page }) => { test("should show Danger status chip for critical stock medication", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
const criticalRow = overviewTable.locator(".table-row").filter({ hasText: MED_CRITICAL }); const criticalRow = overviewTable.locator(".table-row").filter({ hasText: MED_CRITICAL });
@@ -184,7 +184,7 @@ test.describe("Stock Status Levels", () => {
test("should show Danger status chip for depleted medication", async ({ page }) => { test("should show Danger status chip for depleted medication", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
const depletedRow = overviewTable.locator(".table-row").filter({ hasText: MED_DEPLETED }); const depletedRow = overviewTable.locator(".table-row").filter({ hasText: MED_DEPLETED });
@@ -195,7 +195,7 @@ test.describe("Stock Status Levels", () => {
test("should show days-left and runs-out date in overview", async ({ page }) => { test("should show days-left and runs-out date in overview", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// High stock should show many days (around 299) // High stock should show many days (around 299)
@@ -227,7 +227,7 @@ test.describe("Stock Status Levels", () => {
test("should color-code stock values depending on status", async ({ page }) => { test("should color-code stock values depending on status", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// High stock row should have success-text class on stock cells // High stock row should have success-text class on stock cells
@@ -255,7 +255,7 @@ test.describe("Stock Status Levels", () => {
test("should open medication detail modal showing stock info", async ({ page }) => { test("should open medication detail modal showing stock info", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// Click on the critical stock medication row // Click on the critical stock medication row
@@ -278,7 +278,7 @@ test.describe("Stock Status Levels", () => {
test("should show generic name in overview for medications that have one", async ({ page }) => { test("should show generic name in overview for medications that have one", async ({ page }) => {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
// Click on the normal stock med (has generic name "Ibuprofen 400mg") // Click on the normal stock med (has generic name "Ibuprofen 400mg")
+1 -1
View File
@@ -54,7 +54,7 @@ test.describe("MedDetail footer tooltip visibility", () => {
*/ */
async function openMedDetailModal(page: import("@playwright/test").Page) { async function openMedDetailModal(page: import("@playwright/test").Page) {
await navigateTo(page, "/dashboard"); await navigateTo(page, "/dashboard");
const overviewTable = page.locator(".table.table-7"); const overviewTable = page.locator(".dashboard-overview-section .table").first();
await expect(overviewTable).toBeVisible({ timeout: 10000 }); await expect(overviewTable).toBeVisible({ timeout: 10000 });
const medRow = overviewTable.locator(".table-row").filter({ hasText: MED_NAME }).first(); const medRow = overviewTable.locator(".table-row").filter({ hasText: MED_NAME }).first();
+13 -4
View File
@@ -4,21 +4,30 @@
# Translates LOG_LEVEL into nginx access log control before # Translates LOG_LEVEL into nginx access log control before
# delegating to the standard nginx-unprivileged entrypoint. # delegating to the standard nginx-unprivileged entrypoint.
# #
# LOG_LEVEL=debug|info → access logs enabled (default) # LOG_LEVEL=debug → all access logs enabled (including polling)
# LOG_LEVEL=warn|error|fatal|silent → access logs suppressed # LOG_LEVEL=info → access logs enabled, polling endpoints suppressed (default)
# LOG_LEVEL=warn|error|fatal|silent → all access logs suppressed
# ============================================================================= # =============================================================================
# Normalize: lowercase + trim whitespace # Normalize: lowercase + trim whitespace
level=$(printf '%s' "${LOG_LEVEL:-info}" | tr '[:upper:]' '[:lower:]' | tr -d '[:space:]') level=$(printf '%s' "${LOG_LEVEL:-info}" | tr '[:upper:]' '[:lower:]' | tr -d '[:space:]')
case "$level" in case "$level" in
debug)
export NGINX_ACCESS_LOG="/dev/stdout timed"
export NGINX_POLLING_LOG="/dev/stdout timed"
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL} → access_log on (all requests)"
;;
warn|error|fatal|silent) warn|error|fatal|silent)
export NGINX_ACCESS_LOG="off" export NGINX_ACCESS_LOG="off"
export NGINX_POLLING_LOG="off"
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL} → access_log off" echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL} → access_log off"
;; ;;
*) *)
export NGINX_ACCESS_LOG="/dev/stdout" # info (default): log everything except high-frequency polling endpoints
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL:-info} → access_log /dev/stdout" export NGINX_ACCESS_LOG="/dev/stdout timed"
export NGINX_POLLING_LOG="off"
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL:-info} → access_log on (polling suppressed)"
;; ;;
esac esac
+49
View File
@@ -1,3 +1,6 @@
# Must be defined at http-level (outside server block)
log_format timed '$time_iso8601 $status $request_method $request_uri ($request_time s)';
server { server {
# Port 8080 for unprivileged nginx (non-root) # Port 8080 for unprivileged nginx (non-root)
listen 8080; listen 8080;
@@ -24,6 +27,52 @@ server {
try_files $uri /index.html; try_files $uri /index.html;
} }
# -------------------------------------------------------------------------
# High-frequency polling endpoints suppress access logs at info level
# (visible at debug level via NGINX_POLLING_LOG)
# -------------------------------------------------------------------------
location = /api/doses/taken {
access_log ${NGINX_POLLING_LOG};
resolver 127.0.0.11 valid=10s ipv6=off;
set $backend_upstream ${BACKEND_URL};
rewrite ^/api/(.*)$ /$1 break;
proxy_pass http://$backend_upstream;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
proxy_pass_header Set-Cookie;
proxy_cookie_path / /;
}
location ~ ^/api/share/[^/]+/doses$ {
access_log ${NGINX_POLLING_LOG};
resolver 127.0.0.11 valid=10s ipv6=off;
set $backend_upstream ${BACKEND_URL};
rewrite ^/api/(.*)$ /$1 break;
proxy_pass http://$backend_upstream;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
proxy_pass_header Set-Cookie;
proxy_cookie_path / /;
}
location = /api/health {
access_log ${NGINX_POLLING_LOG};
resolver 127.0.0.11 valid=10s ipv6=off;
set $backend_upstream ${BACKEND_URL};
rewrite ^/api/(.*)$ /$1 break;
proxy_pass http://$backend_upstream;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
proxy_pass_header Set-Cookie;
proxy_cookie_path / /;
}
location /api/ { location /api/ {
# Use variable for runtime DNS resolution (nginx resolves at startup by default) # Use variable for runtime DNS resolution (nginx resolves at startup by default)
# Docker embedded DNS (127.0.0.11) with 10s cache # Docker embedded DNS (127.0.0.11) with 10s cache
+164 -131
View File
@@ -1,18 +1,18 @@
{ {
"name": "medassist-ng-frontend", "name": "medassist-ng-frontend",
"version": "1.15.1", "version": "1.18.2",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "medassist-ng-frontend", "name": "medassist-ng-frontend",
"version": "1.15.1", "version": "1.18.2",
"dependencies": { "dependencies": {
"i18next": "^25.8.13", "i18next": "^25.8.13",
"i18next-browser-languagedetector": "^8.2.1", "i18next-browser-languagedetector": "^8.2.1",
"lucide-react": "^0.575.0", "lucide-react": "^0.575.0",
"react": "^18.3.1", "react": "^19.2.4",
"react-dom": "^18.3.1", "react-dom": "^19.2.4",
"react-i18next": "^15.4.1", "react-i18next": "^15.4.1",
"react-router-dom": "^7.13.1", "react-router-dom": "^7.13.1",
"zod": "^4.3.6" "zod": "^4.3.6"
@@ -23,9 +23,9 @@
"@testing-library/jest-dom": "^6.9.1", "@testing-library/jest-dom": "^6.9.1",
"@testing-library/react": "^16.3.2", "@testing-library/react": "^16.3.2",
"@testing-library/user-event": "^14.6.1", "@testing-library/user-event": "^14.6.1",
"@types/node": "^25.3.0", "@types/node": "^25.3.3",
"@types/react": "^18.3.4", "@types/react": "^19.2.14",
"@types/react-dom": "^18.3.0", "@types/react-dom": "^19.2.3",
"@types/react-router-dom": "^5.3.3", "@types/react-router-dom": "^5.3.3",
"@vitejs/plugin-react": "^5.1.4", "@vitejs/plugin-react": "^5.1.4",
"@vitest/coverage-v8": "^4.0.18", "@vitest/coverage-v8": "^4.0.18",
@@ -1247,9 +1247,9 @@
"license": "MIT" "license": "MIT"
}, },
"node_modules/@rollup/rollup-android-arm-eabi": { "node_modules/@rollup/rollup-android-arm-eabi": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.59.0.tgz",
"integrity": "sha512-iDGS/h7D8t7tvZ1t6+WPK04KD0MwzLZrG0se1hzBjSi5fyxlsiggoJHwh18PCFNn7tG43OWb6pdZ6Y+rMlmyNQ==", "integrity": "sha512-upnNBkA6ZH2VKGcBj9Fyl9IGNPULcjXRlg0LLeaioQWueH30p6IXtJEbKAgvyv+mJaMxSm1l6xwDXYjpEMiLMg==",
"cpu": [ "cpu": [
"arm" "arm"
], ],
@@ -1261,9 +1261,9 @@
] ]
}, },
"node_modules/@rollup/rollup-android-arm64": { "node_modules/@rollup/rollup-android-arm64": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.59.0.tgz",
"integrity": "sha512-wrSAViWvZHBMMlWk6EJhvg8/rjxzyEhEdgfMMjREHEq11EtJ6IP6yfcCH57YAEca2Oe3FNCE9DSTgU70EIGmVw==", "integrity": "sha512-hZ+Zxj3SySm4A/DylsDKZAeVg0mvi++0PYVceVyX7hemkw7OreKdCvW2oQ3T1FMZvCaQXqOTHb8qmBShoqk69Q==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -1275,9 +1275,9 @@
] ]
}, },
"node_modules/@rollup/rollup-darwin-arm64": { "node_modules/@rollup/rollup-darwin-arm64": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.59.0.tgz",
"integrity": "sha512-S87zZPBmRO6u1YXQLwpveZm4JfPpAa6oHBX7/ghSiGH3rz/KDgAu1rKdGutV+WUI6tKDMbaBJomhnT30Y2t4VQ==", "integrity": "sha512-W2Psnbh1J8ZJw0xKAd8zdNgF9HRLkdWwwdWqubSVk0pUuQkoHnv7rx4GiF9rT4t5DIZGAsConRE3AxCdJ4m8rg==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -1289,9 +1289,9 @@
] ]
}, },
"node_modules/@rollup/rollup-darwin-x64": { "node_modules/@rollup/rollup-darwin-x64": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.59.0.tgz",
"integrity": "sha512-YTbnsAaHo6VrAczISxgpTva8EkfQus0VPEVJCEaboHtZRIb6h6j0BNxRBOwnDciFTZLDPW5r+ZBmhL/+YpTZgA==", "integrity": "sha512-ZW2KkwlS4lwTv7ZVsYDiARfFCnSGhzYPdiOU4IM2fDbL+QGlyAbjgSFuqNRbSthybLbIJ915UtZBtmuLrQAT/w==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -1303,9 +1303,9 @@
] ]
}, },
"node_modules/@rollup/rollup-freebsd-arm64": { "node_modules/@rollup/rollup-freebsd-arm64": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.59.0.tgz",
"integrity": "sha512-1T8eY2J8rKJWzaznV7zedfdhD1BqVs1iqILhmHDq/bqCUZsrMt+j8VCTHhP0vdfbHK3e1IQ7VYx3jlKqwlf+vw==", "integrity": "sha512-EsKaJ5ytAu9jI3lonzn3BgG8iRBjV4LxZexygcQbpiU0wU0ATxhNVEpXKfUa0pS05gTcSDMKpn3Sx+QB9RlTTA==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -1317,9 +1317,9 @@
] ]
}, },
"node_modules/@rollup/rollup-freebsd-x64": { "node_modules/@rollup/rollup-freebsd-x64": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.59.0.tgz",
"integrity": "sha512-sHTiuXyBJApxRn+VFMaw1U+Qsz4kcNlxQ742snICYPrY+DDL8/ZbaC4DVIB7vgZmp3jiDaKA0WpBdP0aqPJoBQ==", "integrity": "sha512-d3DuZi2KzTMjImrxoHIAODUZYoUUMsuUiY4SRRcJy6NJoZ6iIqWnJu9IScV9jXysyGMVuW+KNzZvBLOcpdl3Vg==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -1331,9 +1331,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-arm-gnueabihf": { "node_modules/@rollup/rollup-linux-arm-gnueabihf": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.59.0.tgz",
"integrity": "sha512-dV3T9MyAf0w8zPVLVBptVlzaXxka6xg1f16VAQmjg+4KMSTWDvhimI/Y6mp8oHwNrmnmVl9XxJ/w/mO4uIQONA==", "integrity": "sha512-t4ONHboXi/3E0rT6OZl1pKbl2Vgxf9vJfWgmUoCEVQVxhW6Cw/c8I6hbbu7DAvgp82RKiH7TpLwxnJeKv2pbsw==",
"cpu": [ "cpu": [
"arm" "arm"
], ],
@@ -1345,9 +1345,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-arm-musleabihf": { "node_modules/@rollup/rollup-linux-arm-musleabihf": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.59.0.tgz",
"integrity": "sha512-wIGYC1x/hyjP+KAu9+ewDI+fi5XSNiUi9Bvg6KGAh2TsNMA3tSEs+Sh6jJ/r4BV/bx/CyWu2ue9kDnIdRyafcQ==", "integrity": "sha512-CikFT7aYPA2ufMD086cVORBYGHffBo4K8MQ4uPS/ZnY54GKj36i196u8U+aDVT2LX4eSMbyHtyOh7D7Zvk2VvA==",
"cpu": [ "cpu": [
"arm" "arm"
], ],
@@ -1359,9 +1359,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-arm64-gnu": { "node_modules/@rollup/rollup-linux-arm64-gnu": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.59.0.tgz",
"integrity": "sha512-Y+qVA0D9d0y2FRNiG9oM3Hut/DgODZbU9I8pLLPwAsU0tUKZ49cyV1tzmB/qRbSzGvY8lpgGkJuMyuhH7Ma+Vg==", "integrity": "sha512-jYgUGk5aLd1nUb1CtQ8E+t5JhLc9x5WdBKew9ZgAXg7DBk0ZHErLHdXM24rfX+bKrFe+Xp5YuJo54I5HFjGDAA==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -1373,9 +1373,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-arm64-musl": { "node_modules/@rollup/rollup-linux-arm64-musl": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.59.0.tgz",
"integrity": "sha512-juaC4bEgJsyFVfqhtGLz8mbopaWD+WeSOYr5E16y+1of6KQjc0BpwZLuxkClqY1i8sco+MdyoXPNiCkQou09+g==", "integrity": "sha512-peZRVEdnFWZ5Bh2KeumKG9ty7aCXzzEsHShOZEFiCQlDEepP1dpUl/SrUNXNg13UmZl+gzVDPsiCwnV1uI0RUA==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -1387,9 +1387,23 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-loong64-gnu": { "node_modules/@rollup/rollup-linux-loong64-gnu": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.59.0.tgz",
"integrity": "sha512-rIEC0hZ17A42iXtHX+EPJVL/CakHo+tT7W0pbzdAGuWOt2jxDFh7A/lRhsNHBcqL4T36+UiAgwO8pbmn3dE8wA==", "integrity": "sha512-gbUSW/97f7+r4gHy3Jlup8zDG190AuodsWnNiXErp9mT90iCy9NKKU0Xwx5k8VlRAIV2uU9CsMnEFg/xXaOfXg==",
"cpu": [
"loong64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"linux"
]
},
"node_modules/@rollup/rollup-linux-loong64-musl": {
"version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.59.0.tgz",
"integrity": "sha512-yTRONe79E+o0FWFijasoTjtzG9EBedFXJMl888NBEDCDV9I2wGbFFfJQQe63OijbFCUZqxpHz1GzpbtSFikJ4Q==",
"cpu": [ "cpu": [
"loong64" "loong64"
], ],
@@ -1401,9 +1415,23 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-ppc64-gnu": { "node_modules/@rollup/rollup-linux-ppc64-gnu": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.59.0.tgz",
"integrity": "sha512-T7l409NhUE552RcAOcmJHj3xyZ2h7vMWzcwQI0hvn5tqHh3oSoclf9WgTl+0QqffWFG8MEVZZP1/OBglKZx52Q==", "integrity": "sha512-sw1o3tfyk12k3OEpRddF68a1unZ5VCN7zoTNtSn2KndUE+ea3m3ROOKRCZxEpmT9nsGnogpFP9x6mnLTCaoLkA==",
"cpu": [
"ppc64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"linux"
]
},
"node_modules/@rollup/rollup-linux-ppc64-musl": {
"version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.59.0.tgz",
"integrity": "sha512-+2kLtQ4xT3AiIxkzFVFXfsmlZiG5FXYW7ZyIIvGA7Bdeuh9Z0aN4hVyXS/G1E9bTP/vqszNIN/pUKCk/BTHsKA==",
"cpu": [ "cpu": [
"ppc64" "ppc64"
], ],
@@ -1415,9 +1443,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-riscv64-gnu": { "node_modules/@rollup/rollup-linux-riscv64-gnu": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.59.0.tgz",
"integrity": "sha512-7OK5/GhxbnrMcxIFoYfhV/TkknarkYC1hqUw1wU2xUN3TVRLNT5FmBv4KkheSG2xZ6IEbRAhTooTV2+R5Tk0lQ==", "integrity": "sha512-NDYMpsXYJJaj+I7UdwIuHHNxXZ/b/N2hR15NyH3m2qAtb/hHPA4g4SuuvrdxetTdndfj9b1WOmy73kcPRoERUg==",
"cpu": [ "cpu": [
"riscv64" "riscv64"
], ],
@@ -1429,9 +1457,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-riscv64-musl": { "node_modules/@rollup/rollup-linux-riscv64-musl": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.59.0.tgz",
"integrity": "sha512-GwuDBE/PsXaTa76lO5eLJTyr2k8QkPipAyOrs4V/KJufHCZBJ495VCGJol35grx9xryk4V+2zd3Ri+3v7NPh+w==", "integrity": "sha512-nLckB8WOqHIf1bhymk+oHxvM9D3tyPndZH8i8+35p/1YiVoVswPid2yLzgX7ZJP0KQvnkhM4H6QZ5m0LzbyIAg==",
"cpu": [ "cpu": [
"riscv64" "riscv64"
], ],
@@ -1443,9 +1471,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-s390x-gnu": { "node_modules/@rollup/rollup-linux-s390x-gnu": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.59.0.tgz",
"integrity": "sha512-IAE1Ziyr1qNfnmiQLHBURAD+eh/zH1pIeJjeShleII7Vj8kyEm2PF77o+lf3WTHDpNJcu4IXJxNO0Zluro8bOw==", "integrity": "sha512-oF87Ie3uAIvORFBpwnCvUzdeYUqi2wY6jRFWJAy1qus/udHFYIkplYRW+wo+GRUP4sKzYdmE1Y3+rY5Gc4ZO+w==",
"cpu": [ "cpu": [
"s390x" "s390x"
], ],
@@ -1457,9 +1485,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-x64-gnu": { "node_modules/@rollup/rollup-linux-x64-gnu": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.59.0.tgz",
"integrity": "sha512-Pg6E+oP7GvZ4XwgRJBuSXZjcqpIW3yCBhK4BcsANvb47qMvAbCjR6E+1a/U2WXz1JJxp9/4Dno3/iSJLcm5auw==", "integrity": "sha512-3AHmtQq/ppNuUspKAlvA8HtLybkDflkMuLK4DPo77DfthRb71V84/c4MlWJXixZz4uruIH4uaa07IqoAkG64fg==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -1471,9 +1499,9 @@
] ]
}, },
"node_modules/@rollup/rollup-linux-x64-musl": { "node_modules/@rollup/rollup-linux-x64-musl": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.59.0.tgz",
"integrity": "sha512-txGtluxDKTxaMDzUduGP0wdfng24y1rygUMnmlUJ88fzCCULCLn7oE5kb2+tRB+MWq1QDZT6ObT5RrR8HFRKqg==", "integrity": "sha512-2UdiwS/9cTAx7qIUZB/fWtToJwvt0Vbo0zmnYt7ED35KPg13Q0ym1g442THLC7VyI6JfYTP4PiSOWyoMdV2/xg==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -1484,10 +1512,24 @@
"linux" "linux"
] ]
}, },
"node_modules/@rollup/rollup-openbsd-x64": {
"version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.59.0.tgz",
"integrity": "sha512-M3bLRAVk6GOwFlPTIxVBSYKUaqfLrn8l0psKinkCFxl4lQvOSz8ZrKDz2gxcBwHFpci0B6rttydI4IpS4IS/jQ==",
"cpu": [
"x64"
],
"dev": true,
"license": "MIT",
"optional": true,
"os": [
"openbsd"
]
},
"node_modules/@rollup/rollup-openharmony-arm64": { "node_modules/@rollup/rollup-openharmony-arm64": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.59.0.tgz",
"integrity": "sha512-3DFiLPnTxiOQV993fMc+KO8zXHTcIjgaInrqlG8zDp1TlhYl6WgrOHuJkJQ6M8zHEcntSJsUp1XFZSY8C1DYbg==", "integrity": "sha512-tt9KBJqaqp5i5HUZzoafHZX8b5Q2Fe7UjYERADll83O4fGqJ49O1FsL6LpdzVFQcpwvnyd0i+K/VSwu/o/nWlA==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -1499,9 +1541,9 @@
] ]
}, },
"node_modules/@rollup/rollup-win32-arm64-msvc": { "node_modules/@rollup/rollup-win32-arm64-msvc": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.59.0.tgz",
"integrity": "sha512-nggc/wPpNTgjGg75hu+Q/3i32R00Lq1B6N1DO7MCU340MRKL3WZJMjA9U4K4gzy3dkZPXm9E1Nc81FItBVGRlA==", "integrity": "sha512-V5B6mG7OrGTwnxaNUzZTDTjDS7F75PO1ae6MJYdiMu60sq0CqN5CVeVsbhPxalupvTX8gXVSU9gq+Rx1/hvu6A==",
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
@@ -1513,9 +1555,9 @@
] ]
}, },
"node_modules/@rollup/rollup-win32-ia32-msvc": { "node_modules/@rollup/rollup-win32-ia32-msvc": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.59.0.tgz",
"integrity": "sha512-U/54pTbdQpPLBdEzCT6NBCFAfSZMvmjr0twhnD9f4EIvlm9wy3jjQ38yQj1AGznrNO65EWQMgm/QUjuIVrYF9w==", "integrity": "sha512-UKFMHPuM9R0iBegwzKF4y0C4J9u8C6MEJgFuXTBerMk7EJ92GFVFYBfOZaSGLu6COf7FxpQNqhNS4c4icUPqxA==",
"cpu": [ "cpu": [
"ia32" "ia32"
], ],
@@ -1527,9 +1569,9 @@
] ]
}, },
"node_modules/@rollup/rollup-win32-x64-gnu": { "node_modules/@rollup/rollup-win32-x64-gnu": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.59.0.tgz",
"integrity": "sha512-2NqKgZSuLH9SXBBV2dWNRCZmocgSOx8OJSdpRaEcRlIfX8YrKxUT6z0F1NpvDVhOsl190UFTRh2F2WDWWCYp3A==", "integrity": "sha512-laBkYlSS1n2L8fSo1thDNGrCTQMmxjYY5G0WFWjFFYZkKPjsMBsgJfGf4TLxXrF6RyhI60L8TMOjBMvXiTcxeA==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -1541,9 +1583,9 @@
] ]
}, },
"node_modules/@rollup/rollup-win32-x64-msvc": { "node_modules/@rollup/rollup-win32-x64-msvc": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.53.5.tgz", "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.59.0.tgz",
"integrity": "sha512-JRpZUhCfhZ4keB5v0fe02gQJy05GqboPOaxvjugW04RLSYYoB/9t2lx2u/tMs/Na/1NXfY8QYjgRljRpN+MjTQ==", "integrity": "sha512-2HRCml6OztYXyJXAvdDXPKcawukWY2GpR5/nxKp4iBgiO3wcoEGkAaqctIbZcNB6KlUQBIqt8VYkNSj2397EfA==",
"cpu": [ "cpu": [
"x64" "x64"
], ],
@@ -1737,41 +1779,33 @@
"license": "MIT" "license": "MIT"
}, },
"node_modules/@types/node": { "node_modules/@types/node": {
"version": "25.3.0", "version": "25.3.3",
"resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.0.tgz", "resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.3.tgz",
"integrity": "sha512-4K3bqJpXpqfg2XKGK9bpDTc6xO/xoUP/RBWS7AtRMug6zZFaRekiLzjVtAoZMquxoAbzBvy5nxQ7veS5eYzf8A==", "integrity": "sha512-DpzbrH7wIcBaJibpKo9nnSQL0MTRdnWttGyE5haGwK86xgMOkFLp7vEyfQPGLOJh5wNYiJ3V9PmUMDhV9u8kkQ==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"undici-types": "~7.18.0" "undici-types": "~7.18.0"
} }
}, },
"node_modules/@types/prop-types": {
"version": "15.7.15",
"resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz",
"integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==",
"dev": true,
"license": "MIT"
},
"node_modules/@types/react": { "node_modules/@types/react": {
"version": "18.3.27", "version": "19.2.14",
"resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.27.tgz", "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.14.tgz",
"integrity": "sha512-cisd7gxkzjBKU2GgdYrTdtQx1SORymWyaAFhaxQPK9bYO9ot3Y5OikQRvY0VYQtvwjeQnizCINJAenh/V7MK2w==", "integrity": "sha512-ilcTH/UniCkMdtexkoCN0bI7pMcJDvmQFPvuPvmEaYA/NSfFTAgdUSLAoVjaRJm7+6PvcM+q1zYOwS4wTYMF9w==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@types/prop-types": "*",
"csstype": "^3.2.2" "csstype": "^3.2.2"
} }
}, },
"node_modules/@types/react-dom": { "node_modules/@types/react-dom": {
"version": "18.3.7", "version": "19.2.3",
"resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.7.tgz", "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz",
"integrity": "sha512-MEe3UeoENYVFXzoXEWsvcpg6ZvlrFNlOQ7EOsvhI3CfAXwzPfO8Qwuxd40nepsYKqyyVQnTdEfv68q91yLcKrQ==", "integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"peerDependencies": { "peerDependencies": {
"@types/react": "^18.0.0" "@types/react": "^19.2.0"
} }
}, },
"node_modules/@types/react-router": { "node_modules/@types/react-router": {
@@ -2925,28 +2959,24 @@
} }
}, },
"node_modules/react": { "node_modules/react": {
"version": "18.3.1", "version": "19.2.4",
"resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz", "resolved": "https://registry.npmjs.org/react/-/react-19.2.4.tgz",
"integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==", "integrity": "sha512-9nfp2hYpCwOjAN+8TZFGhtWEwgvWHXqESH8qT89AT/lWklpLON22Lc8pEtnpsZz7VmawabSU0gCjnj8aC0euHQ==",
"license": "MIT", "license": "MIT",
"dependencies": {
"loose-envify": "^1.1.0"
},
"engines": { "engines": {
"node": ">=0.10.0" "node": ">=0.10.0"
} }
}, },
"node_modules/react-dom": { "node_modules/react-dom": {
"version": "18.3.1", "version": "19.2.4",
"resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz", "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.4.tgz",
"integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==", "integrity": "sha512-AXJdLo8kgMbimY95O2aKQqsz2iWi9jMgKJhRBAxECE4IFxfcazB2LmzloIoibJI3C12IlY20+KFaLv+71bUJeQ==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"loose-envify": "^1.1.0", "scheduler": "^0.27.0"
"scheduler": "^0.23.2"
}, },
"peerDependencies": { "peerDependencies": {
"react": "^18.3.1" "react": "^19.2.4"
} }
}, },
"node_modules/react-i18next": { "node_modules/react-i18next": {
@@ -3056,9 +3086,9 @@
} }
}, },
"node_modules/rollup": { "node_modules/rollup": {
"version": "4.53.5", "version": "4.59.0",
"resolved": "https://registry.npmjs.org/rollup/-/rollup-4.53.5.tgz", "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.59.0.tgz",
"integrity": "sha512-iTNAbFSlRpcHeeWu73ywU/8KuU/LZmNCSxp6fjQkJBD3ivUb8tpDrXhIxEzA05HlYMEwmtaUnb3RP+YNv162OQ==", "integrity": "sha512-2oMpl67a3zCH9H79LeMcbDhXW/UmWG/y2zuqnF2jQq5uq9TbM9TVyXvA4+t+ne2IIkBdrLpAaRQAvo7YI/Yyeg==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
@@ -3072,28 +3102,31 @@
"npm": ">=8.0.0" "npm": ">=8.0.0"
}, },
"optionalDependencies": { "optionalDependencies": {
"@rollup/rollup-android-arm-eabi": "4.53.5", "@rollup/rollup-android-arm-eabi": "4.59.0",
"@rollup/rollup-android-arm64": "4.53.5", "@rollup/rollup-android-arm64": "4.59.0",
"@rollup/rollup-darwin-arm64": "4.53.5", "@rollup/rollup-darwin-arm64": "4.59.0",
"@rollup/rollup-darwin-x64": "4.53.5", "@rollup/rollup-darwin-x64": "4.59.0",
"@rollup/rollup-freebsd-arm64": "4.53.5", "@rollup/rollup-freebsd-arm64": "4.59.0",
"@rollup/rollup-freebsd-x64": "4.53.5", "@rollup/rollup-freebsd-x64": "4.59.0",
"@rollup/rollup-linux-arm-gnueabihf": "4.53.5", "@rollup/rollup-linux-arm-gnueabihf": "4.59.0",
"@rollup/rollup-linux-arm-musleabihf": "4.53.5", "@rollup/rollup-linux-arm-musleabihf": "4.59.0",
"@rollup/rollup-linux-arm64-gnu": "4.53.5", "@rollup/rollup-linux-arm64-gnu": "4.59.0",
"@rollup/rollup-linux-arm64-musl": "4.53.5", "@rollup/rollup-linux-arm64-musl": "4.59.0",
"@rollup/rollup-linux-loong64-gnu": "4.53.5", "@rollup/rollup-linux-loong64-gnu": "4.59.0",
"@rollup/rollup-linux-ppc64-gnu": "4.53.5", "@rollup/rollup-linux-loong64-musl": "4.59.0",
"@rollup/rollup-linux-riscv64-gnu": "4.53.5", "@rollup/rollup-linux-ppc64-gnu": "4.59.0",
"@rollup/rollup-linux-riscv64-musl": "4.53.5", "@rollup/rollup-linux-ppc64-musl": "4.59.0",
"@rollup/rollup-linux-s390x-gnu": "4.53.5", "@rollup/rollup-linux-riscv64-gnu": "4.59.0",
"@rollup/rollup-linux-x64-gnu": "4.53.5", "@rollup/rollup-linux-riscv64-musl": "4.59.0",
"@rollup/rollup-linux-x64-musl": "4.53.5", "@rollup/rollup-linux-s390x-gnu": "4.59.0",
"@rollup/rollup-openharmony-arm64": "4.53.5", "@rollup/rollup-linux-x64-gnu": "4.59.0",
"@rollup/rollup-win32-arm64-msvc": "4.53.5", "@rollup/rollup-linux-x64-musl": "4.59.0",
"@rollup/rollup-win32-ia32-msvc": "4.53.5", "@rollup/rollup-openbsd-x64": "4.59.0",
"@rollup/rollup-win32-x64-gnu": "4.53.5", "@rollup/rollup-openharmony-arm64": "4.59.0",
"@rollup/rollup-win32-x64-msvc": "4.53.5", "@rollup/rollup-win32-arm64-msvc": "4.59.0",
"@rollup/rollup-win32-ia32-msvc": "4.59.0",
"@rollup/rollup-win32-x64-gnu": "4.59.0",
"@rollup/rollup-win32-x64-msvc": "4.59.0",
"fsevents": "~2.3.2" "fsevents": "~2.3.2"
} }
}, },
@@ -3111,9 +3144,9 @@
} }
}, },
"node_modules/scheduler": { "node_modules/scheduler": {
"version": "0.23.2", "version": "0.27.0",
"resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz", "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz",
"integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==", "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"loose-envify": "^1.1.0" "loose-envify": "^1.1.0"
+8 -6
View File
@@ -1,7 +1,7 @@
{ {
"name": "medassist-ng-frontend", "name": "medassist-ng-frontend",
"private": true, "private": true,
"version": "1.16.0", "version": "1.19.0",
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",
@@ -16,6 +16,8 @@
"test:coverage": "vitest run --coverage", "test:coverage": "vitest run --coverage",
"test:e2e": "rm -rf test-results && playwright test --config=playwright.stable.config.ts", "test:e2e": "rm -rf test-results && playwright test --config=playwright.stable.config.ts",
"test:e2e:all": "rm -rf test-results && playwright test --config=playwright.all.config.ts", "test:e2e:all": "rm -rf test-results && playwright test --config=playwright.all.config.ts",
"test:e2e:local": "PLAYWRIGHT_HTML_OPEN=never PLAYWRIGHT_WORKERS=4 npm run test:e2e",
"test:e2e:all:local": "PLAYWRIGHT_HTML_OPEN=never PLAYWRIGHT_WORKERS=4 npm run test:e2e:all",
"test:e2e:with-video": "npm run test:e2e && npm run test:e2e:video", "test:e2e:with-video": "npm run test:e2e && npm run test:e2e:video",
"test:e2e:all:with-video": "npm run test:e2e:all && npm run test:e2e:video", "test:e2e:all:with-video": "npm run test:e2e:all && npm run test:e2e:video",
"test:e2e:video": "find \"$PWD/test-results\" -name video.webm -not -path '*retry*' -print0 | xargs -0 ls -tr > /tmp/e2e-videos.list && if [ -s /tmp/e2e-videos.list ]; then sed \"s/^/file '/\" /tmp/e2e-videos.list | sed \"s/$/'/\" > /tmp/e2e-videos.txt && ffmpeg -y -f concat -safe 0 -i /tmp/e2e-videos.txt -c copy test-results/all-tests.webm; else echo 'No videos found to merge'; fi", "test:e2e:video": "find \"$PWD/test-results\" -name video.webm -not -path '*retry*' -print0 | xargs -0 ls -tr > /tmp/e2e-videos.list && if [ -s /tmp/e2e-videos.list ]; then sed \"s/^/file '/\" /tmp/e2e-videos.list | sed \"s/$/'/\" > /tmp/e2e-videos.txt && ffmpeg -y -f concat -safe 0 -i /tmp/e2e-videos.txt -c copy test-results/all-tests.webm; else echo 'No videos found to merge'; fi",
@@ -28,8 +30,8 @@
"i18next": "^25.8.13", "i18next": "^25.8.13",
"i18next-browser-languagedetector": "^8.2.1", "i18next-browser-languagedetector": "^8.2.1",
"lucide-react": "^0.575.0", "lucide-react": "^0.575.0",
"react": "^18.3.1", "react": "^19.2.4",
"react-dom": "^18.3.1", "react-dom": "^19.2.4",
"react-i18next": "^15.4.1", "react-i18next": "^15.4.1",
"react-router-dom": "^7.13.1", "react-router-dom": "^7.13.1",
"zod": "^4.3.6" "zod": "^4.3.6"
@@ -40,9 +42,9 @@
"@testing-library/jest-dom": "^6.9.1", "@testing-library/jest-dom": "^6.9.1",
"@testing-library/react": "^16.3.2", "@testing-library/react": "^16.3.2",
"@testing-library/user-event": "^14.6.1", "@testing-library/user-event": "^14.6.1",
"@types/node": "^25.3.0", "@types/node": "^25.3.3",
"@types/react": "^18.3.4", "@types/react": "^19.2.14",
"@types/react-dom": "^18.3.0", "@types/react-dom": "^19.2.3",
"@types/react-router-dom": "^5.3.3", "@types/react-router-dom": "^5.3.3",
"@vitejs/plugin-react": "^5.1.4", "@vitejs/plugin-react": "^5.1.4",
"@vitest/coverage-v8": "^4.0.18", "@vitest/coverage-v8": "^4.0.18",
+9 -5
View File
@@ -6,6 +6,10 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
? ((globalThis as { process?: { env?: Record<string, string | undefined> } }).process?.env ?? {}) ? ((globalThis as { process?: { env?: Record<string, string | undefined> } }).process?.env ?? {})
: {}; : {};
const baseURL = env.PLAYWRIGHT_BASE_URL || "http://localhost:5173"; const baseURL = env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
const parsedWorkers = Number.parseInt(env.PLAYWRIGHT_WORKERS ?? "", 10);
// Default to single-worker execution to keep API-seeded E2E suites deterministic.
// Still allow explicit local overrides via PLAYWRIGHT_WORKERS.
const workers = Number.isFinite(parsedWorkers) && parsedWorkers > 0 ? parsedWorkers : 1;
const projects: NonNullable<PlaywrightTestConfig["projects"]> = [ const projects: NonNullable<PlaywrightTestConfig["projects"]> = [
{ {
@@ -17,13 +21,13 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
use: { use: {
...devices["Desktop Chrome"], ...devices["Desktop Chrome"],
}, },
testIgnore: /.*-(?:data|crud|edit|status|schedule)\.spec\.ts/, testIgnore: /.*-(?:data|crud|edit|status|schedule|lifecycle)\.spec\.ts|performance\.spec\.ts/,
dependencies: ["setup"], dependencies: ["setup"],
retries: 1, retries: 1,
}, },
{ {
name: "chromium-data", name: "chromium-data",
testMatch: /.*-(?:data|crud|edit|status|schedule)\.spec\.ts/, testMatch: /.*-(?:data|crud|edit|status|schedule|lifecycle)\.spec\.ts|performance\.spec\.ts/,
use: { use: {
...devices["Desktop Chrome"], ...devices["Desktop Chrome"],
}, },
@@ -40,7 +44,7 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
use: { use: {
...devices["Desktop Firefox"], ...devices["Desktop Firefox"],
}, },
testIgnore: /.*-(?:data|crud|edit|status|schedule)\.spec\.ts/, testIgnore: /.*-(?:data|crud|edit|status|schedule|lifecycle)\.spec\.ts|performance\.spec\.ts/,
dependencies: ["setup"], dependencies: ["setup"],
}, },
{ {
@@ -48,7 +52,7 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
use: { use: {
...devices["Desktop Safari"], ...devices["Desktop Safari"],
}, },
testIgnore: /.*-(?:data|crud|edit|status|schedule)\.spec\.ts/, testIgnore: /.*-(?:data|crud|edit|status|schedule|lifecycle)\.spec\.ts|performance\.spec\.ts/,
dependencies: ["setup"], dependencies: ["setup"],
}, },
); );
@@ -64,7 +68,7 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
fullyParallel: true, fullyParallel: true,
forbidOnly: !!env.CI, forbidOnly: !!env.CI,
retries: env.CI ? 2 : 0, retries: env.CI ? 2 : 0,
workers: 1, workers,
reporter: env.CI reporter: env.CI
? [["html", { outputFolder: "playwright-report" }], ["github"]] ? [["html", { outputFolder: "playwright-report" }], ["github"]]
: [["html", { outputFolder: "playwright-report" }], ["list"]], : [["html", { outputFolder: "playwright-report" }], ["list"]],
+20 -3
View File
@@ -37,13 +37,29 @@ export default function App() {
); );
} }
function getInitialAuthTheme(): "light" | "dark" {
if (typeof window === "undefined") return "dark";
const stored = localStorage.getItem("theme");
if (stored === "light" || stored === "dark") {
return stored;
}
if (stored === "system") {
return window.matchMedia?.("(prefers-color-scheme: light)").matches ? "light" : "dark";
}
return window.matchMedia?.("(prefers-color-scheme: light)").matches ? "light" : "dark";
}
function AppRouter() { function AppRouter() {
const { user, authState, loading, authError } = useAuth(); const { user, authState, loading, authError } = useAuth();
const authTheme = getInitialAuthTheme();
// Show loading while checking auth state // Show loading while checking auth state
if (loading) { if (loading) {
return ( return (
<div className="auth-container"> <div className="auth-container" data-theme={authTheme}>
<div className="auth-card" style={{ textAlign: "center" }}> <div className="auth-card" style={{ textAlign: "center" }}>
<h1 className="auth-title">💊 MedAssist-ng</h1> <h1 className="auth-title">💊 MedAssist-ng</h1>
<p>Loading...</p> <p>Loading...</p>
@@ -55,7 +71,7 @@ function AppRouter() {
// Show error if we couldn't connect to the server // Show error if we couldn't connect to the server
if (authError) { if (authError) {
return ( return (
<div className="auth-container"> <div className="auth-container" data-theme={authTheme}>
<div className="auth-card" style={{ textAlign: "center" }}> <div className="auth-card" style={{ textAlign: "center" }}>
<h1 className="auth-title">💊 MedAssist-ng</h1> <h1 className="auth-title">💊 MedAssist-ng</h1>
<div className="auth-error" style={{ marginBottom: "1rem" }}> <div className="auth-error" style={{ marginBottom: "1rem" }}>
@@ -77,7 +93,7 @@ function AppRouter() {
// If auth state is null (shouldn't happen after loading, but be safe) // If auth state is null (shouldn't happen after loading, but be safe)
if (!authState) { if (!authState) {
return ( return (
<div className="auth-container"> <div className="auth-container" data-theme={authTheme}>
<div className="auth-card" style={{ textAlign: "center" }}> <div className="auth-card" style={{ textAlign: "center" }}>
<h1 className="auth-title">💊 MedAssist-ng</h1> <h1 className="auth-title">💊 MedAssist-ng</h1>
<p>Initializing...</p> <p>Initializing...</p>
@@ -301,6 +317,7 @@ function AppContent() {
useEffect(() => { useEffect(() => {
const handleEscape = (e: KeyboardEvent) => { const handleEscape = (e: KeyboardEvent) => {
if (e.key !== "Escape") return; if (e.key !== "Escape") return;
if (e.defaultPrevented) return;
if (scheduleLightboxImage) { if (scheduleLightboxImage) {
closeScheduleLightbox(); closeScheduleLightbox();
+1 -2
View File
@@ -1,7 +1,6 @@
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { FRONTEND_VERSION, GITHUB_URL } from "../App"; import { FRONTEND_VERSION, GITHUB_URL } from "../App";
import { useEscapeKey } from "../hooks/useEscapeKey";
interface UpdateCheckResult { interface UpdateCheckResult {
status: "up-to-date" | "update-available" | "error"; status: "up-to-date" | "update-available" | "error";
@@ -18,7 +17,7 @@ export default function AboutModal({ isOpen, onClose }: AboutModalProps) {
const [isChecking, setIsChecking] = useState(false); const [isChecking, setIsChecking] = useState(false);
const [updateCheckResult, setUpdateCheckResult] = useState<UpdateCheckResult | null>(null); const [updateCheckResult, setUpdateCheckResult] = useState<UpdateCheckResult | null>(null);
useEscapeKey(isOpen, onClose); // ESC is handled by the global handler in App.tsx to avoid double history.back()
// Reset check result when modal opens so stale results are never shown // Reset check result when modal opens so stale results are never shown
useEffect(() => { useEffect(() => {
+13 -9
View File
@@ -20,7 +20,7 @@ export interface User {
export interface AuthState { export interface AuthState {
authEnabled: boolean; authEnabled: boolean;
registrationEnabled: boolean; registrationEnabled: boolean;
localAuthEnabled: boolean; formLoginEnabled: boolean;
oidcEnabled: boolean; oidcEnabled: boolean;
oidcProviderName: string; oidcProviderName: string;
hasUsers: boolean; hasUsers: boolean;
@@ -157,7 +157,7 @@ export function AuthProvider({ children }: { children: ReactNode }) {
return; return;
} }
} }
log.warn("[Auth] Session refresh failed, clearing local user state", { correlationId }); log.debug("[Auth] Session refresh unavailable, clearing local user state", { correlationId });
setUser(null); setUser(null);
} else { } else {
log.warn("[Auth] Unexpected /auth/me response", { status: res.status, correlationId }); log.warn("[Auth] Unexpected /auth/me response", { status: res.status, correlationId });
@@ -181,7 +181,11 @@ export function AuthProvider({ children }: { children: ReactNode }) {
); );
const res = await fetch("/api/auth/refresh", init); const res = await fetch("/api/auth/refresh", init);
if (!res.ok) { if (!res.ok) {
log.warn("[Auth] Token refresh rejected", { status: res.status, correlationId }); if (res.status === 401) {
log.debug("[Auth] Token refresh rejected (unauthenticated)", { status: res.status, correlationId });
} else {
log.warn("[Auth] Token refresh rejected", { status: res.status, correlationId });
}
} }
return res.ok; return res.ok;
} catch (error) { } catch (error) {
@@ -425,7 +429,7 @@ export function LoginForm({
</svg> </svg>
{t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })} {t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })}
</button> </button>
{authState?.localAuthEnabled && ( {authState?.formLoginEnabled && (
<div className="auth-divider"> <div className="auth-divider">
<span>{t("auth.or", "or")}</span> <span>{t("auth.or", "or")}</span>
</div> </div>
@@ -433,8 +437,8 @@ export function LoginForm({
</div> </div>
)} )}
{/* Local Login Form - only show if local auth is enabled */} {/* Local login form - only show if form login is enabled */}
{authState?.localAuthEnabled && ( {authState?.formLoginEnabled && (
<form onSubmit={handleSubmit} className="auth-form"> <form onSubmit={handleSubmit} className="auth-form">
{error && <div className="auth-error">{error}</div>} {error && <div className="auth-error">{error}</div>}
@@ -474,7 +478,7 @@ export function LoginForm({
</form> </form>
)} )}
{authState?.registrationEnabled && authState?.localAuthEnabled && onSwitchToRegister && ( {authState?.registrationEnabled && authState?.formLoginEnabled && onSwitchToRegister && (
<div className="auth-links"> <div className="auth-links">
<button type="button" className="auth-link-btn" onClick={onSwitchToRegister}> <button type="button" className="auth-link-btn" onClick={onSwitchToRegister}>
{t("auth.createAccount", "Create account")} {t("auth.createAccount", "Create account")}
@@ -540,7 +544,7 @@ export function RegisterForm({ onSuccess, onSwitchToLogin }: { onSuccess?: () =>
</svg> </svg>
{t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })} {t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })}
</button> </button>
{authState?.localAuthEnabled && ( {authState?.formLoginEnabled && (
<div className="auth-divider"> <div className="auth-divider">
<span>{t("auth.or", "or")}</span> <span>{t("auth.or", "or")}</span>
</div> </div>
@@ -549,7 +553,7 @@ export function RegisterForm({ onSuccess, onSwitchToLogin }: { onSuccess?: () =>
)} )}
{/* Local Registration Form - only show if local auth is enabled */} {/* Local Registration Form - only show if local auth is enabled */}
{authState?.localAuthEnabled && ( {authState?.formLoginEnabled && (
<form onSubmit={handleSubmit} className="auth-form"> <form onSubmit={handleSubmit} className="auth-form">
{error && <div className="auth-error">{error}</div>} {error && <div className="auth-error">{error}</div>}
+257 -92
View File
@@ -15,7 +15,14 @@ import { useTranslation } from "react-i18next";
import { Lightbox, MedicationAvatar } from "../components"; import { Lightbox, MedicationAvatar } from "../components";
import { useEscapeKey } from "../hooks"; import { useEscapeKey } from "../hooks";
import type { Coverage, Medication, RefillEntry, StockThresholds } from "../types"; import type { Coverage, Medication, RefillEntry, StockThresholds } from "../types";
import { getMedTotal, getPackageSize } from "../types"; import {
getMedDisplayName,
getMedTotal,
getPackageSize,
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
} from "../types";
import { formatNumber, generateICS, getExpiryClass, getSystemLocale } from "../utils"; import { formatNumber, generateICS, getExpiryClass, getSystemLocale } from "../utils";
import { getStockStatus } from "../utils/schedule"; import { getStockStatus } from "../utils/schedule";
import { splitCurrentBlisterStock } from "../utils/stock"; import { splitCurrentBlisterStock } from "../utils/stock";
@@ -159,8 +166,8 @@ export function MedDetailModal({
// Escape key: only one handler is active at a time (sub-modal states are mutually exclusive). // Escape key: only one handler is active at a time (sub-modal states are mutually exclusive).
// Lightbox has its own useEscapeKey internally. // Lightbox has its own useEscapeKey internally.
useEscapeKey(!showEditStockModal && !showImageLightbox && !showRefillModal, onClose); useEscapeKey(!showEditStockModal && !showImageLightbox && !showRefillModal, onClose);
useEscapeKey(showEditStockModal, onCloseEditStockModal); useEscapeKey(showEditStockModal, onCloseEditStockModal, { capture: true });
useEscapeKey(showRefillModal, onCloseRefillModal); useEscapeKey(showRefillModal, onCloseRefillModal, { capture: true });
useEffect(() => { useEffect(() => {
if (showEditStockModal) return; if (showEditStockModal) return;
@@ -170,7 +177,7 @@ export function MedDetailModal({
}, [showEditStockModal]); }, [showEditStockModal]);
const remainingPrescriptionRefills = Math.max(0, Number(selectedMed?.prescriptionRemainingRefills) || 0); const remainingPrescriptionRefills = Math.max(0, Number(selectedMed?.prescriptionRemainingRefills) || 0);
const prescriptionPackCapEnabled = selectedMed?.packageType === "blister" && usePrescriptionRefill; const prescriptionPackCapEnabled = !isAmountBasedPackageType(selectedMed?.packageType) && usePrescriptionRefill;
const cappedRefillPacks = prescriptionPackCapEnabled const cappedRefillPacks = prescriptionPackCapEnabled
? Math.min(refillPacks, remainingPrescriptionRefills) ? Math.min(refillPacks, remainingPrescriptionRefills)
: refillPacks; : refillPacks;
@@ -179,7 +186,7 @@ export function MedDetailModal({
useEffect(() => { useEffect(() => {
if (!selectedMed) return; if (!selectedMed) return;
if (!showRefillModal) return; if (!showRefillModal) return;
if (selectedMed.packageType !== "blister" || !usePrescriptionRefill) return; if (isAmountBasedPackageType(selectedMed.packageType) || !usePrescriptionRefill) return;
if (refillPacks <= remainingPrescriptionRefills) return; if (refillPacks <= remainingPrescriptionRefills) return;
onRefillPacksChange(remainingPrescriptionRefills); onRefillPacksChange(remainingPrescriptionRefills);
}, [ }, [
@@ -192,14 +199,20 @@ export function MedDetailModal({
]); ]);
if (!selectedMed) return null; if (!selectedMed) return null;
const isAmountPackage =
isTubePackageType(selectedMed.packageType) || isLiquidContainerPackageType(selectedMed.packageType);
const amountUnitLabel =
isLiquidContainerPackageType(selectedMed.packageType) || selectedMed.medicationForm === "liquid"
? t("form.packageAmountUnitMl")
: t("form.packageAmountUnitG");
const stockUnitLabel = isAmountPackage ? amountUnitLabel : null;
const medCoverage = coverage.all.find((c) => c.name === selectedMed.name); const medCoverage = coverage.all.find((c) => c.name === getMedDisplayName(selectedMed));
const packageSize = getPackageSize(selectedMed); const packageSize = getPackageSize(selectedMed);
// Structural max = sealed package capacity only (excludes pre-existing looseTablets). // Structural max = sealed package capacity only (excludes pre-existing looseTablets).
const structuralMax = const structuralMax = isAmountBasedPackageType(selectedMed.packageType)
selectedMed.packageType === "bottle" ? (selectedMed.totalPills ?? packageSize)
? (selectedMed.totalPills ?? packageSize) : selectedMed.packCount * selectedMed.blistersPerPack * selectedMed.pillsPerBlister;
: selectedMed.packCount * selectedMed.blistersPerPack * selectedMed.pillsPerBlister;
const currentStock = medCoverage ? Math.round(medCoverage.medsLeft) : getMedTotal(selectedMed); const currentStock = medCoverage ? Math.round(medCoverage.medsLeft) : getMedTotal(selectedMed);
const status = medCoverage ? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings) : null; const status = medCoverage ? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings) : null;
const fallbackTextClass = status?.className === "warning" ? "warning-text" : "success-text"; const fallbackTextClass = status?.className === "warning" ? "warning-text" : "success-text";
@@ -208,8 +221,21 @@ export function MedDetailModal({
const currentFullBlisters = Math.max(0, stock.fullBlisters); const currentFullBlisters = Math.max(0, stock.fullBlisters);
const currentPartialPills = Math.max(0, stock.openBlisterPills); const currentPartialPills = Math.max(0, stock.openBlisterPills);
const currentLoosePills = Math.max(0, stock.loosePills); const currentLoosePills = Math.max(0, stock.loosePills);
const stockDisplayTotal = const stockDisplayTotal = isAmountBasedPackageType(selectedMed.packageType)
selectedMed.packageType === "bottle" ? (selectedMed.totalPills ?? packageSize) : Math.max(0, structuralMax); ? (selectedMed.totalPills ?? packageSize)
: Math.max(0, structuralMax);
const packageCount = Math.max(1, Number(selectedMed.packCount) || 1);
const amountPerPackage = (() => {
const configured = Number(selectedMed.packageAmountValue ?? 0);
if (Number.isFinite(configured) && configured > 0) return configured;
const totalAmount = Number(stockDisplayTotal ?? 0);
if (Number.isFinite(totalAmount) && totalAmount > 0) {
return Math.max(0, totalAmount / packageCount);
}
return 0;
})();
const maxPartialPills = Math.min( const maxPartialPills = Math.min(
Math.max(0, selectedMed.pillsPerBlister), Math.max(0, selectedMed.pillsPerBlister),
Math.max(0, structuralMax - Math.max(0, editStockFullBlisters) * selectedMed.pillsPerBlister) Math.max(0, structuralMax - Math.max(0, editStockFullBlisters) * selectedMed.pillsPerBlister)
@@ -219,6 +245,33 @@ export function MedDetailModal({
const closeLabel = t("common.close"); const closeLabel = t("common.close");
const decrementLabel = t("editStock.decreaseValue"); const decrementLabel = t("editStock.decreaseValue");
const incrementLabel = t("editStock.increaseValue"); const incrementLabel = t("editStock.increaseValue");
const getScheduleUsageLabel = (usage: number, intakeUnit?: "ml" | "tsp" | "tbsp" | null) => {
if (isLiquidContainerPackageType(selectedMed.packageType)) {
if (intakeUnit === "tsp") {
return `${usage} ${t("form.blisters.teaspoons", { count: Math.abs(usage) })}`;
}
if (intakeUnit === "tbsp") {
return `${usage} ${t("form.blisters.tablespoons", { count: Math.abs(usage) })}`;
}
return `${usage} ${t("form.packageAmountUnitMl")}`;
}
if (isTubePackageType(selectedMed.packageType)) {
return `${usage} ${t("form.blisters.applications", { count: Math.abs(usage) })}`;
}
return `${usage} ${usage !== 1 ? t("common.pills") : t("common.pill")}`;
};
const scheduleIntakes =
selectedMed.intakes && selectedMed.intakes.length > 0
? selectedMed.intakes
: selectedMed.blisters.map((blister) => ({
usage: blister.usage,
every: blister.every,
start: blister.start,
takenBy: null,
intakeRemindersEnabled: false,
intakeUnit: null,
}));
const hasAnyIntakeReminder = scheduleIntakes.some((intake) => intake.intakeRemindersEnabled === true);
const normalizeBlisterStock = (nextFull: number, nextPartial: number, nextLoose: number) => { const normalizeBlisterStock = (nextFull: number, nextPartial: number, nextLoose: number) => {
let normalizedFull = Math.max(0, nextFull); let normalizedFull = Math.max(0, nextFull);
let normalizedPartial = Math.max(0, nextPartial); let normalizedPartial = Math.max(0, nextPartial);
@@ -347,6 +400,10 @@ export function MedDetailModal({
const renderEditStockModal = () => { const renderEditStockModal = () => {
if (!showEditStockModal) return null; if (!showEditStockModal) return null;
const isLiquidPackage = isLiquidContainerPackageType(selectedMed.packageType);
const liquidBottleCount = Math.max(1, editStockFullBlisters);
const liquidAmountPerBottle = Math.max(1, Number.isFinite(amountPerPackage) ? amountPerPackage : 1);
const liquidCapacity = Math.max(1, Math.round(liquidBottleCount * liquidAmountPerBottle));
const fullInputMax = Math.min( const fullInputMax = Math.min(
maxFullBlisters, maxFullBlisters,
Math.floor(Math.max(0, structuralMax - Math.max(0, editStockPartialBlisterPills)) / selectedMed.pillsPerBlister) Math.floor(Math.max(0, structuralMax - Math.max(0, editStockPartialBlisterPills)) / selectedMed.pillsPerBlister)
@@ -360,14 +417,14 @@ export function MedDetailModal({
onCloseEditStockModal(); onCloseEditStockModal();
}} }}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key !== "Escape") e.stopPropagation(); e.stopPropagation();
}} }}
> >
<div <div
className="modal-content edit-stock-modal" className="modal-content edit-stock-modal"
onClick={(e) => e.stopPropagation()} onClick={(e) => e.stopPropagation()}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key !== "Escape") e.stopPropagation(); e.stopPropagation();
}} }}
> >
<button <button
@@ -380,9 +437,9 @@ export function MedDetailModal({
<X size={18} aria-hidden="true" /> <X size={18} aria-hidden="true" />
</button> </button>
<h2>{t("editStock.title")}</h2> <h2>{t("editStock.title")}</h2>
<p className="edit-stock-med-name">{selectedMed.name}</p> <p className="edit-stock-med-name">{getMedDisplayName(selectedMed)}</p>
<p className="edit-stock-hint">{t("editStock.hint")}</p> <p className="edit-stock-hint">{t("editStock.hint")}</p>
{selectedMed.packageType === "blister" && ( {!isAmountBasedPackageType(selectedMed.packageType) && (
<p className="edit-stock-cap-info edit-stock-live-breakdown"> <p className="edit-stock-cap-info edit-stock-live-breakdown">
{t("editStock.currentComposition", { {t("editStock.currentComposition", {
fullBlisters: currentFullBlisters, fullBlisters: currentFullBlisters,
@@ -392,9 +449,15 @@ export function MedDetailModal({
})} })}
</p> </p>
)} )}
{selectedMed.packageType === "bottle" && ( {isAmountBasedPackageType(selectedMed.packageType) && !isTubePackageType(selectedMed.packageType) && (
<p className="edit-stock-cap-info">{t("editStock.packageSize", { count: structuralMax })}</p> <p className="edit-stock-cap-info">{t("editStock.packageSize", { count: structuralMax })}</p>
)} )}
{(isTubePackageType(selectedMed.packageType) || isLiquidContainerPackageType(selectedMed.packageType)) && (
<p className="edit-stock-cap-info">
{t("form.totalAmount")}: {formatNumber(isLiquidPackage ? liquidCapacity : structuralMax)}{" "}
{amountUnitLabel}
</p>
)}
{showStockCapNotice && ( {showStockCapNotice && (
<p className="edit-stock-cap-warning">{t("editStock.maxExceeded", { count: structuralMax })}</p> <p className="edit-stock-cap-warning">{t("editStock.maxExceeded", { count: structuralMax })}</p>
)} )}
@@ -402,12 +465,14 @@ export function MedDetailModal({
{(() => { {(() => {
const dbTotal = getMedTotal(selectedMed); const dbTotal = getMedTotal(selectedMed);
const currentTotal = medCoverage ? Math.round(medCoverage.medsLeft) : dbTotal; const currentTotal = medCoverage ? Math.round(medCoverage.medsLeft) : dbTotal;
const isBottle = selectedMed.packageType === "bottle"; const isBottle = isAmountBasedPackageType(selectedMed.packageType);
const enteredTotal = isBottle const enteredTotal = isLiquidPackage
? editStockPartialBlisterPills ? Math.min(liquidCapacity, editStockPartialBlisterPills)
: editStockFullBlisters * selectedMed.pillsPerBlister + : isBottle
editStockPartialBlisterPills + ? editStockPartialBlisterPills
editStockLoosePills; : editStockFullBlisters * selectedMed.pillsPerBlister +
editStockPartialBlisterPills +
editStockLoosePills;
const newTotal = Math.max(0, enteredTotal); const newTotal = Math.max(0, enteredTotal);
const difference = newTotal - currentTotal; const difference = newTotal - currentTotal;
const differenceClass = difference > 0 ? "positive" : difference < 0 ? "negative" : ""; const differenceClass = difference > 0 ? "positive" : difference < 0 ? "negative" : "";
@@ -417,36 +482,39 @@ export function MedDetailModal({
<div className="edit-stock-form"> <div className="edit-stock-form">
{isBottle ? ( {isBottle ? (
<label> <label>
{t("editStock.totalPills")} {isAmountPackage ? t("form.currentAmount") : t("editStock.totalPills")}
{renderStepperInput({ {renderStepperInput({
value: editStockPartialInput, value: editStockPartialInput,
min: 0, min: 0,
max: structuralMax, max: isLiquidPackage ? liquidCapacity : structuralMax,
onChange: (raw) => { onChange: (raw) => {
const parsed = raw === "" ? 0 : Math.max(0, parseStockInput(raw)); const parsed = raw === "" ? 0 : Math.max(0, parseStockInput(raw));
setEditStockPartialInput(raw); setEditStockPartialInput(raw);
onEditStockPartialBlisterPillsChange(raw === "" ? 0 : Math.min(structuralMax, parsed)); const maxTotal = isLiquidPackage ? liquidCapacity : structuralMax;
setShowStockCapNotice(parsed > structuralMax); onEditStockPartialBlisterPillsChange(raw === "" ? 0 : Math.min(maxTotal, parsed));
setShowStockCapNotice(parsed > maxTotal);
}, },
onBlur: () => { onBlur: () => {
const normalized = Math.min( const maxTotal = isLiquidPackage ? liquidCapacity : structuralMax;
structuralMax, const normalized = Math.min(maxTotal, Math.max(0, parseStockInput(editStockPartialInput)));
Math.max(0, parseStockInput(editStockPartialInput))
);
onEditStockPartialBlisterPillsChange(normalized); onEditStockPartialBlisterPillsChange(normalized);
setEditStockPartialInput(String(normalized)); setEditStockPartialInput(String(normalized));
setShowStockCapNotice(false); setShowStockCapNotice(false);
}, },
onStep: (delta) => { onStep: (delta) => {
const next = Math.min( const maxTotal = isLiquidPackage ? liquidCapacity : structuralMax;
structuralMax, const next = Math.min(maxTotal, Math.max(0, parseStockInput(editStockPartialInput) + delta));
Math.max(0, parseStockInput(editStockPartialInput) + delta)
);
onEditStockPartialBlisterPillsChange(next); onEditStockPartialBlisterPillsChange(next);
setEditStockPartialInput(String(next)); setEditStockPartialInput(String(next));
setShowStockCapNotice(false); setShowStockCapNotice(false);
}, },
})} })}
{isLiquidPackage && (
<p className="edit-stock-cap-info" style={{ marginTop: "0.35rem" }}>
{t("form.currentAmount")}: {Math.max(0, editStockPartialBlisterPills)} {amountUnitLabel} /{" "}
{liquidCapacity} {amountUnitLabel}
</p>
)}
</label> </label>
) : ( ) : (
<> <>
@@ -584,26 +652,72 @@ export function MedDetailModal({
</label> </label>
</> </>
)} )}
{isLiquidPackage && (
<label>
{t("form.bottles")}
{renderStepperInput({
value: editStockFullInput,
min: 1,
max: Number.MAX_SAFE_INTEGER,
onChange: (raw) => {
const nextBottleCount = raw === "" ? 1 : Math.max(1, parseStockInput(raw));
setEditStockFullInput(raw === "" ? "1" : raw);
onEditStockFullBlistersChange(nextBottleCount);
const syncedTotal = Math.round(nextBottleCount * liquidAmountPerBottle);
onEditStockPartialBlisterPillsChange(syncedTotal);
setEditStockPartialInput(String(syncedTotal));
setShowStockCapNotice(false);
},
onBlur: () => {
const normalized = Math.max(1, parseStockInput(editStockFullInput));
onEditStockFullBlistersChange(normalized);
setEditStockFullInput(String(normalized));
const syncedTotal = Math.round(normalized * liquidAmountPerBottle);
onEditStockPartialBlisterPillsChange(syncedTotal);
setEditStockPartialInput(String(syncedTotal));
setShowStockCapNotice(false);
},
onStep: (delta) => {
const next = Math.max(1, parseStockInput(editStockFullInput) + delta);
onEditStockFullBlistersChange(next);
setEditStockFullInput(String(next));
const syncedTotal = Math.round(next * liquidAmountPerBottle);
onEditStockPartialBlisterPillsChange(syncedTotal);
setEditStockPartialInput(String(syncedTotal));
setShowStockCapNotice(false);
},
})}
</label>
)}
</div> </div>
<div className="edit-stock-summary"> <div className="edit-stock-summary">
<div className="summary-row"> <div className="summary-row">
<span>{t("editStock.currentTotal")}:</span> <span>{t("editStock.currentTotal")}:</span>
<span> <span>
{currentTotal} {currentTotal === 1 ? t("common.pill") : t("common.pills")} {currentTotal}
{isAmountPackage
? ` ${stockUnitLabel}`
: ` ${currentTotal === 1 ? t("common.pill") : t("common.pills")}`}
</span> </span>
</div> </div>
<div className="summary-row"> <div className="summary-row">
<span>{t("editStock.newTotal")}:</span> <span>{t("editStock.newTotal")}:</span>
<span> <span>
{newTotal} {newTotal === 1 ? t("common.pill") : t("common.pills")} {newTotal}
{isAmountPackage
? ` ${stockUnitLabel}`
: ` ${newTotal === 1 ? t("common.pill") : t("common.pills")}`}
</span> </span>
</div> </div>
<div className={`summary-row difference ${differenceClass}`}> <div className={`summary-row difference ${differenceClass}`}>
<span>{t("editStock.difference")}:</span> <span>{t("editStock.difference")}:</span>
<span> <span>
{difference > 0 ? "+" : ""} {difference > 0 ? "+" : ""}
{difference} {Math.abs(difference) === 1 ? t("common.pill") : t("common.pills")} {difference}
{isAmountPackage
? ` ${stockUnitLabel}`
: ` ${Math.abs(difference) === 1 ? t("common.pill") : t("common.pills")}`}
</span> </span>
</div> </div>
</div> </div>
@@ -667,12 +781,14 @@ export function MedDetailModal({
} }
}} }}
> >
<MedicationAvatar name={selectedMed.name} imageUrl={selectedMed.imageUrl} size="lg" /> <MedicationAvatar name={getMedDisplayName(selectedMed)} imageUrl={selectedMed.imageUrl} size="lg" />
{selectedMed.imageUrl && <span className="expand-icon">🔍</span>} {selectedMed.imageUrl && <span className="expand-icon">🔍</span>}
</div> </div>
<div className="med-detail-titles"> <div className="med-detail-titles">
<h2>{selectedMed.name}</h2> <h2>{getMedDisplayName(selectedMed)}</h2>
{selectedMed.genericName && <span className="med-generic-name">{selectedMed.genericName}</span>} {selectedMed.name && selectedMed.genericName && (
<span className="med-generic-name">{selectedMed.genericName}</span>
)}
{selectedMed.takenBy && (selectedMed.takenBy || []).length > 0 && ( {selectedMed.takenBy && (selectedMed.takenBy || []).length > 0 && (
<span className="med-taken-by"> <span className="med-taken-by">
{t("modal.for")}{" "} {t("modal.for")}{" "}
@@ -694,7 +810,7 @@ export function MedDetailModal({
<div className="med-detail-section"> <div className="med-detail-section">
<h3>{t("modal.stockInfo")}</h3> <h3>{t("modal.stockInfo")}</h3>
<div className="med-detail-grid"> <div className="med-detail-grid">
{selectedMed.packageType === "blister" && ( {!isAmountBasedPackageType(selectedMed.packageType) && (
<> <>
<div className="med-detail-item"> <div className="med-detail-item">
<span className="med-detail-label">{t("table.fullBlisters")}</span> <span className="med-detail-label">{t("table.fullBlisters")}</span>
@@ -713,10 +829,14 @@ export function MedDetailModal({
</div> </div>
</> </>
)} )}
<div className={`med-detail-item ${selectedMed.packageType === "bottle" ? "full-width" : "full-width"}`}> <div className="med-detail-item full-width">
<span className="med-detail-label">{t("modal.currentStock")}</span> <span className="med-detail-label">
{isAmountPackage ? t("form.currentAmount") : t("modal.currentStock")}
</span>
<span className={`med-detail-value ${textClass}`}> <span className={`med-detail-value ${textClass}`}>
{currentStock} / {stockDisplayTotal} {isAmountPackage
? `${formatNumber(currentStock)} / ${formatNumber(stockDisplayTotal)} ${amountUnitLabel}`
: `${currentStock} / ${stockDisplayTotal}`}
{currentStock > stockDisplayTotal && ( {currentStock > stockDisplayTotal && (
<span <span
className="info-tooltip tooltip-align-left warning-text" className="info-tooltip tooltip-align-left warning-text"
@@ -735,10 +855,27 @@ export function MedDetailModal({
<div className="med-detail-section"> <div className="med-detail-section">
<h3> <h3>
{t("modal.packageDetails")} ( {t("modal.packageDetails")} (
{selectedMed.packageType === "bottle" ? t("form.packageTypeBottle") : t("form.packageTypeBlister")}) {isTubePackageType(selectedMed.packageType)
? t("form.packageTypeTube")
: isLiquidContainerPackageType(selectedMed.packageType)
? t("form.packageTypeLiquidContainer")
: isAmountBasedPackageType(selectedMed.packageType)
? t("form.packageTypeBottle")
: t("form.packageTypeBlister")}
)
{isTubePackageType(selectedMed.packageType) && (
<span className="info-tooltip small" data-tooltip={t("modal.packageTypeTubeHint")}>
</span>
)}
{isLiquidContainerPackageType(selectedMed.packageType) && (
<span className="info-tooltip small" data-tooltip={t("modal.packageTypeLiquidHint")}>
</span>
)}
</h3> </h3>
<div className="med-detail-grid"> <div className="med-detail-grid">
{selectedMed.packageType === "blister" ? ( {!isAmountBasedPackageType(selectedMed.packageType) ? (
<> <>
<div className="med-detail-item"> <div className="med-detail-item">
<span className="med-detail-label">{t("modal.packs")}</span> <span className="med-detail-label">{t("modal.packs")}</span>
@@ -753,6 +890,44 @@ export function MedDetailModal({
<span className="med-detail-value">{selectedMed.pillsPerBlister}</span> <span className="med-detail-value">{selectedMed.pillsPerBlister}</span>
</div> </div>
</> </>
) : isLiquidContainerPackageType(selectedMed.packageType) ? (
<>
<div className="med-detail-item">
<span className="med-detail-label">{t("form.bottles")}</span>
<span className="med-detail-value">{packageCount}</span>
</div>
<div className="med-detail-item">
<span className="med-detail-label">{t("form.packageAmountPerBottle")}</span>
<span className="med-detail-value">
{formatNumber(amountPerPackage)} {amountUnitLabel}
</span>
</div>
<div className="med-detail-item">
<span className="med-detail-label">{t("form.totalAmount")}</span>
<span className="med-detail-value">
{formatNumber(stockDisplayTotal)} {amountUnitLabel}
</span>
</div>
</>
) : isTubePackageType(selectedMed.packageType) ? (
<>
<div className="med-detail-item">
<span className="med-detail-label">{t("form.tubes")}</span>
<span className="med-detail-value">{packageCount}</span>
</div>
<div className="med-detail-item">
<span className="med-detail-label">{t("form.packageAmountPerTube")}</span>
<span className="med-detail-value">
{formatNumber(amountPerPackage)} {amountUnitLabel}
</span>
</div>
<div className="med-detail-item">
<span className="med-detail-label">{t("form.totalAmount")}</span>
<span className="med-detail-value">
{formatNumber(stockDisplayTotal)} {amountUnitLabel}
</span>
</div>
</>
) : ( ) : (
<div className="med-detail-item"> <div className="med-detail-item">
<span className="med-detail-label">{t("form.totalCapacity")}</span> <span className="med-detail-label">{t("form.totalCapacity")}</span>
@@ -789,53 +964,33 @@ export function MedDetailModal({
<div className="med-detail-section"> <div className="med-detail-section">
<h3> <h3>
{t("modal.intakeSchedule")}{" "} {t("modal.intakeSchedule")}{" "}
{selectedMed.intakeRemindersEnabled && ( {hasAnyIntakeReminder && (
<span className="reminder-icon info-tooltip" data-tooltip={t("tooltips.intakeReminders")}> <span className="reminder-icon info-tooltip" data-tooltip={t("tooltips.intakeReminders")}>
<Bell size={14} aria-hidden="true" /> <Bell size={14} aria-hidden="true" />
</span> </span>
)} )}
</h3> </h3>
<div className="med-detail-schedules"> <div className="med-detail-schedules">
{(selectedMed.intakes && selectedMed.intakes.length > 0 {scheduleIntakes.map((intake, idx) => {
? selectedMed.intakes
: selectedMed.blisters.map((blister) => ({
usage: blister.usage,
every: blister.every,
start: blister.start,
takenBy: null,
intakeRemindersEnabled: selectedMed.intakeRemindersEnabled ?? false,
}))
).map((intake, idx) => {
const hasPerIntakeTakenBy = !!intake.takenBy; const hasPerIntakeTakenBy = !!intake.takenBy;
const personCount = Math.max(1, selectedMed.takenBy?.length ?? 0); const personCount = Math.max(1, selectedMed.takenBy?.length ?? 0);
const totalUsage = hasPerIntakeTakenBy ? intake.usage : intake.usage * personCount; const totalUsage = hasPerIntakeTakenBy ? intake.usage : intake.usage * personCount;
const showIntakeBell = intake.intakeRemindersEnabled ?? selectedMed.intakeRemindersEnabled ?? false; const showIntakeBell = intake.intakeRemindersEnabled === true;
return ( return (
<div key={`${intake.start}-${intake.usage}-${intake.every}-${idx}`} className="med-schedule-item"> <div
key={`${intake.start}-${intake.usage}-${intake.every}-${idx}`}
className="med-schedule-row blister-row-simple"
>
<span className="med-schedule-usage"> <span className="med-schedule-usage">
{totalUsage} {totalUsage !== 1 ? t("common.pills") : t("common.pill")} {getScheduleUsageLabel(totalUsage, intake.intakeUnit)}
{selectedMed.pillWeightMg && {selectedMed.pillWeightMg &&
` (${totalUsage * selectedMed.pillWeightMg} ${selectedMed.doseUnit ?? "mg"})`} ` (${totalUsage * selectedMed.pillWeightMg} ${selectedMed.doseUnit ?? "mg"})`}
</span> </span>
<span className="med-schedule-freq"> <span className="med-schedule-freq">
{intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every })} {intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every })}
</span> </span>
{hasPerIntakeTakenBy && ( {hasPerIntakeTakenBy && <span className="med-schedule-person">{intake.takenBy}</span>}
<span className="med-schedule-person">
{intake.takenBy}
{showIntakeBell && (
<span className="med-schedule-bell" role="img" aria-label={t("tooltips.intakeReminders")}>
<Bell size={13} aria-hidden="true" />
</span>
)}
</span>
)}
{!hasPerIntakeTakenBy && showIntakeBell && (
<span className="med-schedule-bell" role="img" aria-label={t("tooltips.intakeReminders")}>
<Bell size={13} aria-hidden="true" />
</span>
)}
<span className="med-schedule-time"> <span className="med-schedule-time">
{t("modal.at")}{" "} {t("modal.at")}{" "}
{new Date(intake.start).toLocaleTimeString(getSystemLocale(i18n.language), { {new Date(intake.start).toLocaleTimeString(getSystemLocale(i18n.language), {
@@ -843,6 +998,11 @@ export function MedDetailModal({
minute: "2-digit", minute: "2-digit",
})} })}
</span> </span>
{showIntakeBell && (
<span className="med-schedule-bell" title={t("form.blisters.remindTooltip")}>
<Bell size={12} aria-hidden="true" />
</span>
)}
</div> </div>
); );
})} })}
@@ -952,12 +1112,11 @@ export function MedDetailModal({
</span> </span>
<span className="refill-amount"> <span className="refill-amount">
{(() => { {(() => {
const total = const total = isAmountBasedPackageType(selectedMed.packageType)
selectedMed.packageType === "bottle" ? entry.loosePillsAdded
? entry.loosePillsAdded : entry.packsAdded * selectedMed.blistersPerPack * selectedMed.pillsPerBlister +
: entry.packsAdded * selectedMed.blistersPerPack * selectedMed.pillsPerBlister + entry.loosePillsAdded;
entry.loosePillsAdded; return `+${total}${isAmountPackage ? ` ${stockUnitLabel}` : ` ${total === 1 ? t("common.pill") : t("common.pills")}`}`;
return `+${total} ${total === 1 ? t("common.pill") : t("common.pills")}`;
})()} })()}
{entry.usedPrescription && ( {entry.usedPrescription && (
<span className="refill-prescription-badge" title={t("refill.viaPrescription")}> <span className="refill-prescription-badge" title={t("refill.viaPrescription")}>
@@ -1017,7 +1176,11 @@ export function MedDetailModal({
{/* Image Lightbox */} {/* Image Lightbox */}
{showImageLightbox && selectedMed.imageUrl && ( {showImageLightbox && selectedMed.imageUrl && (
<Lightbox src={`/api/images/${selectedMed.imageUrl}`} alt={selectedMed.name} onClose={onCloseImageLightbox} /> <Lightbox
src={`/api/images/${selectedMed.imageUrl}`}
alt={getMedDisplayName(selectedMed)}
onClose={onCloseImageLightbox}
/>
)} )}
{/* Refill Modal */} {/* Refill Modal */}
@@ -1049,10 +1212,10 @@ export function MedDetailModal({
<X size={18} aria-hidden="true" /> <X size={18} aria-hidden="true" />
</button> </button>
<h2>{t("refill.title")}</h2> <h2>{t("refill.title")}</h2>
<p className="refill-med-name">{selectedMed.name}</p> <p className="refill-med-name">{getMedDisplayName(selectedMed)}</p>
<div className="refill-form"> <div className="refill-form">
{selectedMed.packageType === "blister" ? ( {!isAmountBasedPackageType(selectedMed.packageType) ? (
<> <>
<label> <label>
{t("refill.packs")} {t("refill.packs")}
@@ -1096,7 +1259,7 @@ export function MedDetailModal({
onUsePrescriptionRefillChange(checked); onUsePrescriptionRefillChange(checked);
if ( if (
checked && checked &&
selectedMed.packageType === "blister" && !isAmountBasedPackageType(selectedMed.packageType) &&
refillPacks > remainingPrescriptionRefills refillPacks > remainingPrescriptionRefills
) { ) {
onRefillPacksChange(remainingPrescriptionRefills); onRefillPacksChange(remainingPrescriptionRefills);
@@ -1122,7 +1285,7 @@ export function MedDetailModal({
className="success" className="success"
onClick={() => onSubmitRefill(selectedMed.id, usePrescriptionRefill)} onClick={() => onSubmitRefill(selectedMed.id, usePrescriptionRefill)}
disabled={ disabled={
(selectedMed.packageType === "bottle" (isAmountBasedPackageType(selectedMed.packageType)
? refillLoose < 1 ? refillLoose < 1
: cappedRefillPacks < 1 && refillLoose < 1) || : cappedRefillPacks < 1 && refillLoose < 1) ||
exceedsPrescriptionPackLimit || exceedsPrescriptionPackLimit ||
@@ -1132,13 +1295,15 @@ export function MedDetailModal({
{refillSaving ? t("common.saving") : t("refill.button")} {refillSaving ? t("common.saving") : t("refill.button")}
</button> </button>
{(() => { {(() => {
const totalRefill = const totalRefill = !isAmountBasedPackageType(selectedMed.packageType)
selectedMed.packageType === "blister" ? cappedRefillPacks * selectedMed.blistersPerPack * selectedMed.pillsPerBlister + refillLoose
? cappedRefillPacks * selectedMed.blistersPerPack * selectedMed.pillsPerBlister + refillLoose : refillLoose;
: refillLoose;
return totalRefill > 0 ? ( return totalRefill > 0 ? (
<span className="refill-preview"> <span className="refill-preview">
+{totalRefill} {totalRefill === 1 ? t("common.pill") : t("common.pills")} +{totalRefill}
{isAmountPackage
? ` ${stockUnitLabel}`
: ` ${totalRefill === 1 ? t("common.pill") : t("common.pills")}`}
</span> </span>
) : null; ) : null;
})()} })()}
+4 -1
View File
@@ -2,7 +2,7 @@
// MedicationAvatar Component // MedicationAvatar Component
// ============================================================================= // =============================================================================
import { useEffect, useState } from "react"; import { useEffect, useRef, useState } from "react";
export type MedicationAvatarProps = { export type MedicationAvatarProps = {
name: string; name: string;
@@ -12,8 +12,11 @@ export type MedicationAvatarProps = {
export function MedicationAvatar({ name, imageUrl, size = "sm" }: MedicationAvatarProps) { export function MedicationAvatar({ name, imageUrl, size = "sm" }: MedicationAvatarProps) {
const [thumbFailed, setThumbFailed] = useState(false); const [thumbFailed, setThumbFailed] = useState(false);
const previousImageUrlRef = useRef(imageUrl);
useEffect(() => { useEffect(() => {
if (previousImageUrlRef.current === imageUrl) return;
previousImageUrlRef.current = imageUrl;
setThumbFailed(false); setThumbFailed(false);
}, [imageUrl]); }, [imageUrl]);
+312 -108
View File
@@ -5,12 +5,19 @@
/* biome-ignore-all lint/a11y/noLabelWithoutControl: modal uses custom DateInput and static value fields */ /* biome-ignore-all lint/a11y/noLabelWithoutControl: modal uses custom DateInput and static value fields */
import { Bell, Minus, Plus, Trash2 } from "lucide-react"; import { Bell, Minus, Plus, Trash2 } from "lucide-react";
import { useEffect, useRef, useState } from "react"; import { useCallback, useEffect, useMemo, useRef, useState } from "react";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { useEscapeKey } from "../hooks/useEscapeKey"; import { useEscapeKey } from "../hooks/useEscapeKey";
import { useScrollLock } from "../hooks/useScrollLock"; import { useScrollLock } from "../hooks/useScrollLock";
import type { DoseUnit, FieldErrors, FormBlister, FormIntake, FormState, Medication } from "../types"; import type { DoseUnit, FieldErrors, FormBlister, FormIntake, FormState, Medication } from "../types";
import { DOSE_UNITS } from "../types"; import {
allowsPillFormSelection,
DOSE_UNITS,
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
PACKAGE_PROFILES,
} from "../types";
import { deriveTotal } from "../utils"; import { deriveTotal } from "../utils";
import { DateInput } from "./DateInput"; import { DateInput } from "./DateInput";
import { FormNumberStepper } from "./FormNumberStepper"; import { FormNumberStepper } from "./FormNumberStepper";
@@ -68,7 +75,7 @@ export interface MobileEditModalProps {
/** Calculate total pills from form state */ /** Calculate total pills from form state */
function deriveTotalFromForm(form: FormState) { function deriveTotalFromForm(form: FormState) {
if (form.packageType === "bottle") { if (isAmountBasedPackageType(form.packageType)) {
// For bottle type, looseTablets is the current stock // For bottle type, looseTablets is the current stock
return Number(form.looseTablets) || 0; return Number(form.looseTablets) || 0;
} }
@@ -96,9 +103,9 @@ export function MobileEditModal({
onAddTakenByPerson, onAddTakenByPerson,
onRemoveTakenByPerson, onRemoveTakenByPerson,
onTakenByKeyDown, onTakenByKeyDown,
_onSetBlisterValue, onSetBlisterValue: _onSetBlisterValue,
_onAddBlister, onAddBlister: _onAddBlister,
_onRemoveBlister, onRemoveBlister: _onRemoveBlister,
onSetIntakeValue, onSetIntakeValue,
onAddIntake, onAddIntake,
onRemoveIntake, onRemoveIntake,
@@ -108,7 +115,7 @@ export function MobileEditModal({
onDeleteMedImage, onDeleteMedImage,
imageUploadError, imageUploadError,
onClose, onClose,
_onResetForm, onResetForm: _onResetForm,
onSaveMedication, onSaveMedication,
}: MobileEditModalProps) { }: MobileEditModalProps) {
const { t } = useTranslation(); const { t } = useTranslation();
@@ -125,6 +132,33 @@ export function MobileEditModal({
const [showNameValidation, setShowNameValidation] = useState(false); const [showNameValidation, setShowNameValidation] = useState(false);
const activeTabIndexRef = useRef(0); const activeTabIndexRef = useRef(0);
const allowFractionalIntake = useMemo(() => {
if (isLiquidContainerPackageType(form.packageType)) return true;
if (isTubePackageType(form.packageType)) return form.medicationForm === "liquid";
return form.pillForm === "tablet";
}, [form.packageType, form.medicationForm, form.pillForm]);
const getUsageLabel = useCallback(
(intake: (typeof form.intakes)[number]) => {
if (isLiquidContainerPackageType(form.packageType)) {
if (intake.intakeUnit === "tsp") return t("form.blisters.usageTsp");
if (intake.intakeUnit === "tbsp") return t("form.blisters.usageTbsp");
return t("form.blisters.usageMl");
}
if (isTubePackageType(form.packageType)) {
return form.medicationForm === "liquid" ? t("form.blisters.usageMl") : t("form.blisters.usageApplication");
}
if (form.pillForm === "capsule") return t("form.blisters.usageCapsules");
return t("form.blisters.usageTablets");
},
[form.packageType, form.medicationForm, form.pillForm, t]
);
const usesAmountLabels = isTubePackageType(form.packageType) || isLiquidContainerPackageType(form.packageType);
const totalCapacityLabel = usesAmountLabels ? t("form.totalAmount") : t("form.totalCapacity");
const currentStockLabel = usesAmountLabels ? t("form.currentAmount") : t("form.currentPills");
const totalLabel = usesAmountLabels ? t("form.totalAmountLabel") : t("form.total");
// Reset tab when modal opens // Reset tab when modal opens
useEffect(() => { useEffect(() => {
if (show) { if (show) {
@@ -253,7 +287,10 @@ export function MobileEditModal({
const mobileTitle = (() => { const mobileTitle = (() => {
if (!editingId) return t("form.newEntry"); if (!editingId) return t("form.newEntry");
if (readOnlyMode) return t("form.viewEntry"); if (readOnlyMode) return t("form.viewEntry");
const medicationName = currentMed?.name?.trim() || form.name.trim(); const medicationName =
(currentMed ? currentMed.name?.trim() || currentMed.genericName?.trim() : null) ||
form.name.trim() ||
form.genericName.trim();
if (!medicationName) return t("form.editEntry"); if (!medicationName) return t("form.editEntry");
return t("form.editEntryWithName", { name: medicationName }); return t("form.editEntryWithName", { name: medicationName });
})(); })();
@@ -361,27 +398,35 @@ export function MobileEditModal({
onBlur={() => setShowNameValidation(true)} onBlur={() => setShowNameValidation(true)}
placeholder={t("form.placeholders.commercial")} placeholder={t("form.placeholders.commercial")}
maxLength={FIELD_LIMITS.name.max} maxLength={FIELD_LIMITS.name.max}
required={!readOnlyMode}
/> />
{!readOnlyMode && showNameValidation && fieldErrors.name && ( {!readOnlyMode && showNameValidation && fieldErrors.name && (
<span className="field-error">{fieldErrors.name}</span> <span className="field-error">{fieldErrors.name}</span>
)} )}
</label> </label>
<label className={`full ${fieldErrors.genericName ? "has-error" : ""}`}> <label
className={`full ${!readOnlyMode && showNameValidation && fieldErrors.genericName ? "has-error" : ""}`}
>
{t("form.genericName")} {t("form.genericName")}
<input <input
value={form.genericName} value={form.genericName}
onChange={(e) => onFormChange({ ...form, genericName: e.target.value })} onChange={(e) => {
setShowNameValidation(true);
onFormChange({ ...form, genericName: e.target.value });
}}
onBlur={() => setShowNameValidation(true)}
placeholder={t("form.placeholders.generic")} placeholder={t("form.placeholders.generic")}
maxLength={FIELD_LIMITS.genericName.max} maxLength={FIELD_LIMITS.genericName.max}
/> />
{fieldErrors.genericName && <span className="field-error">{fieldErrors.genericName}</span>} {!readOnlyMode && showNameValidation && fieldErrors.genericName && (
<span className="field-error">{fieldErrors.genericName}</span>
)}
</label> </label>
<label className="full"> <label className="full">
{t("form.medicationStartDate")} {t("form.medicationStartDate")}
<DateInput <DateInput
value={form.medicationStartDate} value={form.medicationStartDate}
onChange={(e) => onHandleValueChange("medicationStartDate", e.target.value)} onChange={(e) => onHandleValueChange("medicationStartDate", e.target.value)}
placeholder={t("common.optional")}
/> />
{!readOnlyMode && dateConsistencyError && ( {!readOnlyMode && dateConsistencyError && (
<span className="field-error">{dateConsistencyError}</span> <span className="field-error">{dateConsistencyError}</span>
@@ -392,12 +437,64 @@ export function MobileEditModal({
<select <select
className="package-type-select" className="package-type-select"
value={form.packageType} value={form.packageType}
onChange={(e) => onHandleValueChange("packageType", e.target.value)} onChange={(e) => onHandleValueChange("packageType", e.target.value as FormState["packageType"])}
> >
<option value="blister">{t("form.packageTypeBlister")}</option> {PACKAGE_PROFILES.map((profile) => (
<option value="bottle">{t("form.packageTypeBottle")}</option> <option key={profile.value} value={profile.value}>
{t(profile.labelKey)}
</option>
))}
</select> </select>
</label> </label>
<label className="full">
{t("form.medicationEndDate")}
<DateInput
value={form.medicationEndDate}
onChange={(e) => onHandleValueChange("medicationEndDate", e.target.value)}
placeholder={t("common.optional")}
/>
</label>
{allowsPillFormSelection(form.packageType) && (
<label className="full">
{t("form.pillForm")}
<select
value={form.pillForm}
onChange={(e) => onHandleValueChange("pillForm", e.target.value as FormState["pillForm"])}
>
<option value="tablet">{t("form.medicationFormTablet")}</option>
<option value="capsule">{t("form.medicationFormCapsule")}</option>
</select>
</label>
)}
{isTubePackageType(form.packageType) && (
<label className="full">
{t("form.medicationForm")}
<select value={"topical"} onChange={() => onHandleValueChange("medicationForm", "topical")}>
<option value="topical">{t("form.medicationFormTopical")}</option>
</select>
</label>
)}
{isLiquidContainerPackageType(form.packageType) && (
<label className="full">
{t("form.medicationForm")}
<select value={"liquid"} onChange={() => onHandleValueChange("medicationForm", "liquid")}>
<option value="liquid">{t("form.medicationFormLiquid")}</option>
</select>
</label>
)}
{form.medicationEndDate && (
<label className="full">
{t("form.autoMarkObsoleteAfterEndDate")}
<span className="toggle-switch small">
<input
type="checkbox"
checked={form.autoMarkObsoleteAfterEndDate}
onChange={(e) => onHandleValueChange("autoMarkObsoleteAfterEndDate", e.target.checked)}
/>
<span className="toggle-slider"></span>
</span>
</label>
)}
<label className={`full ${fieldErrors.takenBy ? "has-error" : ""}`}> <label className={`full ${fieldErrors.takenBy ? "has-error" : ""}`}>
{t("form.takenBy")} {t("form.takenBy")}
<div className="tag-input-container"> <div className="tag-input-container">
@@ -470,101 +567,193 @@ export function MobileEditModal({
<div className={`form-tab-panel${activeTab === "stock" ? " active" : ""}`}> <div className={`form-tab-panel${activeTab === "stock" ? " active" : ""}`}>
<div className="full form-category"> <div className="full form-category">
<h4 className="form-category-title">{t("form.sections.stock")}</h4> <h4 className="form-category-title">{t("form.sections.stock")}</h4>
{form.packageType === "blister" ? ( {(() => {
<> if (!isAmountBasedPackageType(form.packageType)) {
<label> return (
{t("form.packs")} <>
<FormNumberStepper <label>
value={form.packCount} {t("form.packs")}
onChange={(nextValue) => onHandleValueChange("packCount", nextValue)} <FormNumberStepper
min={0} value={form.packCount}
decrementLabel={decrementValueLabel} onChange={(nextValue) => onHandleValueChange("packCount", nextValue)}
incrementLabel={incrementValueLabel} min={0}
/> decrementLabel={decrementValueLabel}
</label> incrementLabel={incrementValueLabel}
<label> />
{t("form.blistersPerPack")} </label>
<FormNumberStepper <label>
value={form.blistersPerPack} {t("form.blistersPerPack")}
onChange={(nextValue) => onHandleValueChange("blistersPerPack", nextValue)} <FormNumberStepper
min={1} value={form.blistersPerPack}
decrementLabel={decrementValueLabel} onChange={(nextValue) => onHandleValueChange("blistersPerPack", nextValue)}
incrementLabel={incrementValueLabel} min={1}
/> decrementLabel={decrementValueLabel}
</label> incrementLabel={incrementValueLabel}
<label> />
{t("form.pillsPerBlister")} </label>
<FormNumberStepper <label>
value={form.pillsPerBlister} {t("form.pillsPerBlister")}
onChange={(nextValue) => onHandleValueChange("pillsPerBlister", nextValue)} <FormNumberStepper
min={1} value={form.pillsPerBlister}
decrementLabel={decrementValueLabel} onChange={(nextValue) => onHandleValueChange("pillsPerBlister", nextValue)}
incrementLabel={incrementValueLabel} min={1}
/> decrementLabel={decrementValueLabel}
</label> incrementLabel={incrementValueLabel}
<label> />
{t("form.total")} </label>
<div className="static-value">{deriveTotalFromForm(form)}</div> <label>
</label> {t("form.total")}
</> <div className="static-value">{deriveTotalFromForm(form)}</div>
) : ( </label>
<> </>
<label> );
{t("form.totalCapacity")} }
<FormNumberStepper
value={form.totalPills} if (isTubePackageType(form.packageType)) {
onChange={(nextValue) => onHandleValueChange("totalPills", nextValue)} return (
min={0} <>
decrementLabel={decrementValueLabel} <label>
incrementLabel={incrementValueLabel} {t("form.tubes")}
/> <div className="static-value">1</div>
</label> </label>
<label> <label className="full">
{t("form.currentPills")} {t("form.packageAmountPerTube")}
<FormNumberStepper <div className="dose-input-group">
value={form.looseTablets} <input
onChange={(nextValue) => onHandleValueChange("looseTablets", nextValue)} type="text"
min={0} inputMode="decimal"
decrementLabel={decrementValueLabel} pattern="[0-9]*\.?[0-9]*"
incrementLabel={incrementValueLabel} value={form.packageAmountValue ?? "0"}
/> onChange={(e) => onHandleValueChange("packageAmountValue", e.target.value)}
</label> placeholder="0"
</> />
)} <select
{form.packageType === "bottle" && ( value="g"
disabled
className="dose-unit-select"
aria-label={t("form.packageAmountUnitG")}
>
<option value="g">{t("form.packageAmountUnitG")}</option>
</select>
</div>
</label>
<label>
{t("form.totalAmount")}
<div className="static-value">
{(Number(form.packCount) || 0) * (Number(form.packageAmountValue ?? 0) || 0)}{" "}
{t("form.packageAmountUnitG")}
</div>
</label>
</>
);
}
if (isLiquidContainerPackageType(form.packageType)) {
return (
<>
<label>
{t("form.bottles")}
<FormNumberStepper
value={form.packCount}
onChange={(nextValue) => onHandleValueChange("packCount", nextValue)}
min={1}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label className="full">
{t("form.packageAmountPerBottle")}
<div className="dose-input-group">
<input
type="text"
inputMode="decimal"
pattern="[0-9]*\.?[0-9]*"
value={form.packageAmountValue ?? "0"}
onChange={(e) => onHandleValueChange("packageAmountValue", e.target.value)}
placeholder="0"
/>
<select
value="ml"
disabled
className="dose-unit-select"
aria-label={t("form.packageAmountUnitMl")}
>
<option value="ml">{t("form.packageAmountUnitMl")}</option>
</select>
</div>
</label>
<label>
{t("form.totalAmount")}
<div className="static-value">
{(Number(form.packCount) || 0) * (Number(form.packageAmountValue ?? 0) || 0)}{" "}
{t("form.packageAmountUnitMl")}
</div>
</label>
</>
);
}
return (
<>
<label>
{totalCapacityLabel}
<FormNumberStepper
value={form.totalPills}
onChange={(nextValue) => onHandleValueChange("totalPills", nextValue)}
min={0}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label>
{currentStockLabel}
<FormNumberStepper
value={form.looseTablets}
onChange={(nextValue) => onHandleValueChange("looseTablets", nextValue)}
min={0}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
</>
);
})()}
{isAmountBasedPackageType(form.packageType) && !isTubePackageType(form.packageType) && (
<div className="full stock-total-row"> <div className="full stock-total-row">
<div className="stock-total-field"> <div className="stock-total-field">
<p className="sub"> <p className="sub">
<strong>{t("form.total")}:</strong> {deriveTotalFromForm(form)}{" "} <strong>{totalLabel}:</strong> {deriveTotalFromForm(form)}
{deriveTotalFromForm(form) === 1 ? t("common.pill") : t("common.pills")} {` ${deriveTotalFromForm(form) === 1 ? t("common.pill") : t("common.pills")}`}
</p> </p>
</div> </div>
</div> </div>
)} )}
<label className="full"> {allowsPillFormSelection(form.packageType) && (
{t("form.pillWeight")} ({form.doseUnit}) <label className="full">
<div className="dose-input-group"> {t("form.pillWeight")} ({form.doseUnit})
<input <div className="dose-input-group">
type="text" <input
inputMode="decimal" type="text"
pattern="[0-9]*\.?[0-9]*" inputMode="decimal"
value={form.pillWeightMg} pattern="[0-9]*\.?[0-9]*"
onChange={(e) => onFormChange({ ...form, pillWeightMg: e.target.value })} value={form.pillWeightMg}
placeholder={t("form.placeholders.weight")} onChange={(e) => onFormChange({ ...form, pillWeightMg: e.target.value })}
/> placeholder={t("form.placeholders.weight")}
<select />
value={form.doseUnit} <select
onChange={(e) => onFormChange({ ...form, doseUnit: e.target.value as DoseUnit })} value={form.doseUnit}
className="dose-unit-select" onChange={(e) => onFormChange({ ...form, doseUnit: e.target.value as DoseUnit })}
> className="dose-unit-select"
{DOSE_UNITS.map((unit) => ( >
<option key={unit.value} value={unit.value}> {DOSE_UNITS.map((unit) => (
{unit.label} <option key={unit.value} value={unit.value}>
</option> {unit.label}
))} </option>
</select> ))}
</div> </select>
</label> </div>
</label>
)}
<label className="full"> <label className="full">
{t("form.expiryDate")} {t("form.expiryDate")}
<DateInput <DateInput
@@ -616,17 +805,17 @@ export function MobileEditModal({
</div> </div>
{form.intakes.map((intake, idx) => ( {form.intakes.map((intake, idx) => (
<div <div
key={`${intake.startDate}-${intake.startTime}-${intake.usage}-${intake.every}-${intake.takenBy ?? ""}`} key={`${intake.startDate}-${intake.startTime}-${intake.usage}-${intake.every}-${intake.takenBy ?? ""}-${idx}`}
className="blister-row" className="blister-row"
> >
<label className="compact"> <label className="compact">
<span>{t("form.blisters.usage")}</span> <span>{getUsageLabel(intake)}</span>
<FormNumberStepper <FormNumberStepper
value={intake.usage} value={intake.usage}
onChange={(nextValue) => onSetIntakeValue(idx, "usage", nextValue)} onChange={(nextValue) => onSetIntakeValue(idx, "usage", nextValue)}
min={0.5} min={allowFractionalIntake ? 0.5 : 1}
step={0.5} step={allowFractionalIntake ? 0.5 : 1}
allowDecimal={true} allowDecimal={allowFractionalIntake}
decrementLabel={decrementValueLabel} decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel} incrementLabel={incrementValueLabel}
/> />
@@ -656,6 +845,21 @@ export function MobileEditModal({
onChange={(e) => onSetIntakeValue(idx, "startTime", e.target.value)} onChange={(e) => onSetIntakeValue(idx, "startTime", e.target.value)}
/> />
</label> </label>
{isLiquidContainerPackageType(form.packageType) && (
<label className="compact full-row">
<span>{t("form.blisters.intakeUnit")}</span>
<select
value={intake.intakeUnit}
onChange={(e) =>
onSetIntakeValue(idx, "intakeUnit", e.target.value as "ml" | "tsp" | "tbsp")
}
>
<option value="ml">{t("form.blisters.intakeUnitMl")}</option>
<option value="tsp">{t("form.blisters.intakeUnitTsp")}</option>
<option value="tbsp">{t("form.blisters.intakeUnitTbsp")}</option>
</select>
</label>
)}
{form.takenBy.length === 0 ? null : ( {form.takenBy.length === 0 ? null : (
<label className="compact full-row taken-by-field"> <label className="compact full-row taken-by-field">
<span>{t("form.blisters.takenByIntake")}</span> <span>{t("form.blisters.takenByIntake")}</span>
+1 -2
View File
@@ -1,4 +1,3 @@
import { useEscapeKey } from "../hooks/useEscapeKey";
import { UserProfile } from "./Auth"; import { UserProfile } from "./Auth";
interface ProfileModalProps { interface ProfileModalProps {
@@ -7,7 +6,7 @@ interface ProfileModalProps {
} }
export default function ProfileModal({ isOpen, onClose }: ProfileModalProps) { export default function ProfileModal({ isOpen, onClose }: ProfileModalProps) {
useEscapeKey(isOpen, onClose); // ESC is handled by the global handler in App.tsx to avoid double history.back()
if (!isOpen) return null; if (!isOpen) return null;
+74 -32
View File
@@ -3,7 +3,13 @@ import { useTranslation } from "react-i18next";
import { useEscapeKey } from "../hooks/useEscapeKey"; import { useEscapeKey } from "../hooks/useEscapeKey";
import { useScrollLock } from "../hooks/useScrollLock"; import { useScrollLock } from "../hooks/useScrollLock";
import type { Medication } from "../types"; import type { Medication } from "../types";
import { getPackageSize } from "../types"; import {
getMedDisplayName,
getPackageSize,
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
} from "../types";
import { MedicationAvatar } from "./MedicationAvatar"; import { MedicationAvatar } from "./MedicationAvatar";
type ReportFormat = "txt" | "md" | "pdf"; type ReportFormat = "txt" | "md" | "pdf";
@@ -200,10 +206,10 @@ export function ReportModal({ isOpen, onClose, medications }: ReportModalProps)
{activeMeds.map((med) => ( {activeMeds.map((med) => (
<label key={med.id} className="report-med-item"> <label key={med.id} className="report-med-item">
<input type="checkbox" checked={selectedIds.has(med.id)} onChange={() => toggleMed(med.id)} /> <input type="checkbox" checked={selectedIds.has(med.id)} onChange={() => toggleMed(med.id)} />
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="sm" /> <MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="sm" />
<span className="report-med-name"> <span className="report-med-name">
{med.name} {getMedDisplayName(med)}
{med.genericName && <span className="report-med-generic"> ({med.genericName})</span>} {med.name && med.genericName && <span className="report-med-generic"> ({med.genericName})</span>}
</span> </span>
</label> </label>
))} ))}
@@ -218,10 +224,10 @@ export function ReportModal({ isOpen, onClose, medications }: ReportModalProps)
{obsoleteMeds.map((med) => ( {obsoleteMeds.map((med) => (
<label key={med.id} className="report-med-item"> <label key={med.id} className="report-med-item">
<input type="checkbox" checked={selectedIds.has(med.id)} onChange={() => toggleMed(med.id)} /> <input type="checkbox" checked={selectedIds.has(med.id)} onChange={() => toggleMed(med.id)} />
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="sm" /> <MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="sm" />
<span className="report-med-name obsolete-name"> <span className="report-med-name obsolete-name">
{med.name} {getMedDisplayName(med)}
{med.genericName && <span className="report-med-generic"> ({med.genericName})</span>} {med.name && med.genericName && <span className="report-med-generic"> ({med.genericName})</span>}
</span> </span>
</label> </label>
))} ))}
@@ -298,6 +304,39 @@ function fmtDateTime(iso: string | null | undefined): string {
return `${m[3]}.${m[2]}.${m[1]} ${m[4]}:${m[5]}`; return `${m[3]}.${m[2]}.${m[1]} ${m[4]}:${m[5]}`;
} }
function getTubeUnitKey(med: Medication): "form.ml" | "blisters.applications" {
if (isLiquidContainerPackageType(med.packageType)) return "form.ml";
return med.medicationForm === "liquid" ? "form.ml" : "blisters.applications";
}
function getUsageText(med: Medication, usage: number, t: TFn): string {
if (isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType)) {
return `${usage} ${t(getTubeUnitKey(med))}`;
}
return `${usage} ${usage === 1 ? t("common.pill") : t("common.pills")}`;
}
function getTotalCapacityLabel(med: Medication, t: TFn): string {
if (isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType)) {
return t("form.totalAmountLabel", { unit: t(getTubeUnitKey(med)) });
}
return t("report.docTotalCapacity");
}
function getCurrentStockText(med: Medication, t: TFn): string {
if (isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType)) {
return `${getPackageSize(med)} ${t(getTubeUnitKey(med))}`;
}
return `${getPackageSize(med)} ${t("common.pills")}`;
}
function getReportPackageTypeLabel(med: Medication, t: TFn): string {
if (isTubePackageType(med.packageType)) return t("report.docTube");
if (isLiquidContainerPackageType(med.packageType)) return t("form.packageTypeLiquidContainer");
if (isAmountBasedPackageType(med.packageType)) return t("report.docBottle");
return t("report.docBlister");
}
function generateTextReport( function generateTextReport(
meds: Medication[], meds: Medication[],
reportData: ReportData, reportData: ReportData,
@@ -320,13 +359,15 @@ function generateTextReport(
for (const med of meds) { for (const med of meds) {
lines.push(sep); lines.push(sep);
lines.push(""); lines.push("");
const title = med.isObsolete ? `${med.name} (${t("report.docStatusObsolete")})` : med.name; const title = med.isObsolete
? `${getMedDisplayName(med)} (${t("report.docStatusObsolete")})`
: getMedDisplayName(med);
lines.push(h2(title)); lines.push(h2(title));
lines.push(""); lines.push("");
// General // General
lines.push(h3(t("report.docGeneral"))); lines.push(h3(t("report.docGeneral")));
lines.push(item(t("report.docCommercialName"), med.name)); if (med.name) lines.push(item(t("report.docCommercialName"), med.name));
if (med.genericName) lines.push(item(t("report.docGenericName"), med.genericName)); if (med.genericName) lines.push(item(t("report.docGenericName"), med.genericName));
if (med.takenBy?.length) lines.push(item(t("report.docTakenBy"), med.takenBy.join(", "))); if (med.takenBy?.length) lines.push(item(t("report.docTakenBy"), med.takenBy.join(", ")));
lines.push( lines.push(
@@ -338,19 +379,18 @@ function generateTextReport(
// Package / Stock // Package / Stock
lines.push(h3(t("report.docPackage"))); lines.push(h3(t("report.docPackage")));
lines.push( lines.push(item(t("report.docPackageType"), getReportPackageTypeLabel(med, t)));
item(t("report.docPackageType"), med.packageType === "bottle" ? t("report.docBottle") : t("report.docBlister")) if (!isAmountBasedPackageType(med.packageType)) {
);
if (med.packageType === "blister") {
lines.push(item(t("report.docPacks"), String(med.packCount))); lines.push(item(t("report.docPacks"), String(med.packCount)));
lines.push(item(t("report.docBlistersPerPack"), String(med.blistersPerPack))); lines.push(item(t("report.docBlistersPerPack"), String(med.blistersPerPack)));
lines.push(item(t("report.docPillsPerBlister"), String(med.pillsPerBlister))); lines.push(item(t("report.docPillsPerBlister"), String(med.pillsPerBlister)));
if (med.looseTablets > 0) lines.push(item(t("report.docLoosePills"), String(med.looseTablets))); if (med.looseTablets > 0) lines.push(item(t("report.docLoosePills"), String(med.looseTablets)));
} else { } else {
lines.push(item(t("report.docTotalCapacity"), String(med.totalPills ?? med.looseTablets))); lines.push(item(getTotalCapacityLabel(med, t), String(med.totalPills ?? med.looseTablets)));
} }
lines.push(item(t("report.docCurrentStock"), `${getPackageSize(med)} ${t("common.pills")}`)); lines.push(item(t("report.docCurrentStock"), getCurrentStockText(med, t)));
if (med.pillWeightMg) lines.push(item(t("report.docDosePerPill"), `${med.pillWeightMg} ${med.doseUnit ?? "mg"}`)); if (!isTubePackageType(med.packageType) && !isLiquidContainerPackageType(med.packageType) && med.pillWeightMg)
lines.push(item(t("report.docDosePerPill"), `${med.pillWeightMg} ${med.doseUnit ?? "mg"}`));
if (med.expiryDate) lines.push(item(t("report.docExpiryDate"), fmtDate(med.expiryDate))); if (med.expiryDate) lines.push(item(t("report.docExpiryDate"), fmtDate(med.expiryDate)));
if (med.notes) lines.push(item(t("report.docNotes"), med.notes)); if (med.notes) lines.push(item(t("report.docNotes"), med.notes));
lines.push(""); lines.push("");
@@ -363,7 +403,7 @@ function generateTextReport(
if (intakes?.length) { if (intakes?.length) {
lines.push(h3(t("report.docIntakeSchedule"))); lines.push(h3(t("report.docIntakeSchedule")));
for (const intake of intakes) { for (const intake of intakes) {
let entry = `${intake.usage} ${intake.usage === 1 ? t("common.pill") : t("common.pills")}`; let entry = getUsageText(med, intake.usage, t);
entry += ` ${intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every })}`; entry += ` ${intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every })}`;
entry += ` ${t("form.blisters.from")} ${fmtDateTime(intake.start)}`; entry += ` ${t("form.blisters.from")} ${fmtDateTime(intake.start)}`;
if ("takenBy" in intake && intake.takenBy) if ("takenBy" in intake && intake.takenBy)
@@ -405,7 +445,7 @@ function generateTextReport(
if (data.refills.length > 0) { if (data.refills.length > 0) {
lines.push(h3(t("report.docRefillHistory"))); lines.push(h3(t("report.docRefillHistory")));
for (const r of data.refills) { for (const r of data.refills) {
let entry = `${fmtDate(r.refillDate)}: +${r.packsAdded} ${t("report.docPacks")}, +${r.loosePillsAdded} ${t("common.pills")}`; let entry = `${fmtDate(r.refillDate)}: +${r.packsAdded} ${t("report.docPacks")}, +${r.loosePillsAdded} ${isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType) ? t(getTubeUnitKey(med)) : t("common.pills")}`;
if (r.usedPrescription) entry += ` ${t("report.docRefillPrescription")}`; if (r.usedPrescription) entry += ` ${t("report.docRefillPrescription")}`;
lines.push(fmt === "md" ? `- ${entry}` : `${entry}`); lines.push(fmt === "md" ? `- ${entry}` : `${entry}`);
} }
@@ -489,22 +529,24 @@ function buildPrintHtml(
for (const med of meds) { for (const med of meds) {
const data = reportData[med.id]; const data = reportData[med.id];
const intakes = med.intakes ?? med.blisters; const intakes = med.intakes ?? med.blisters;
const displayName = getMedDisplayName(med);
const title = med.isObsolete const title = med.isObsolete
? `${escHtml(med.name)} <span class="obsolete-badge">${escHtml(t("report.docStatusObsolete"))}</span>` ? `${escHtml(displayName)} <span class="obsolete-badge">${escHtml(t("report.docStatusObsolete"))}</span>`
: escHtml(med.name); : escHtml(displayName);
let s = `<div class="med-section">`; let s = `<div class="med-section">`;
const imgDataUrl = imageMap[med.id]; const imgDataUrl = imageMap[med.id];
// Title with generic name subtitle // Title with generic name subtitle
s += `<h2>${title}</h2>`; s += `<h2>${title}</h2>`;
if (med.genericName) s += `<p class="generic-subtitle">${escHtml(med.genericName)}</p>`; if (med.name && med.genericName) s += `<p class="generic-subtitle">${escHtml(med.genericName)}</p>`;
// Build general info table rows // Build general info table rows
const generalRows: string[] = []; const generalRows: string[] = [];
generalRows.push( if (med.name)
`<tr><td class="label">${escHtml(t("report.docCommercialName"))}</td><td>${escHtml(med.name)}</td></tr>` generalRows.push(
); `<tr><td class="label">${escHtml(t("report.docCommercialName"))}</td><td>${escHtml(med.name)}</td></tr>`
);
if (med.genericName) if (med.genericName)
generalRows.push( generalRows.push(
`<tr><td class="label">${escHtml(t("report.docGenericName"))}</td><td>${escHtml(med.genericName)}</td></tr>` `<tr><td class="label">${escHtml(t("report.docGenericName"))}</td><td>${escHtml(med.genericName)}</td></tr>`
@@ -527,7 +569,7 @@ function buildPrintHtml(
const generalTable = `<h3>${escHtml(t("report.docGeneral"))}</h3><table><tbody>${generalRows.join("")}</tbody></table>`; const generalTable = `<h3>${escHtml(t("report.docGeneral"))}</h3><table><tbody>${generalRows.join("")}</tbody></table>`;
if (imgDataUrl) { if (imgDataUrl) {
s += `<div class="med-overview"><img class="med-img" src="${imgDataUrl}" alt="${escHtml(med.name)}" /><div class="med-overview-info">${generalTable}</div></div>`; s += `<div class="med-overview"><img class="med-img" src="${imgDataUrl}" alt="${escHtml(displayName)}" /><div class="med-overview-info">${generalTable}</div></div>`;
} else { } else {
s += generalTable; s += generalTable;
} }
@@ -535,18 +577,18 @@ function buildPrintHtml(
// Package / Stock // Package / Stock
s += `<h3>${escHtml(t("report.docPackage"))}</h3>`; s += `<h3>${escHtml(t("report.docPackage"))}</h3>`;
s += `<table><tbody>`; s += `<table><tbody>`;
s += `<tr><td class="label">${escHtml(t("report.docPackageType"))}</td><td>${escHtml(med.packageType === "bottle" ? t("report.docBottle") : t("report.docBlister"))}</td></tr>`; s += `<tr><td class="label">${escHtml(t("report.docPackageType"))}</td><td>${escHtml(getReportPackageTypeLabel(med, t))}</td></tr>`;
if (med.packageType === "blister") { if (!isAmountBasedPackageType(med.packageType)) {
s += `<tr><td class="label">${escHtml(t("report.docPacks"))}</td><td>${med.packCount}</td></tr>`; s += `<tr><td class="label">${escHtml(t("report.docPacks"))}</td><td>${med.packCount}</td></tr>`;
s += `<tr><td class="label">${escHtml(t("report.docBlistersPerPack"))}</td><td>${med.blistersPerPack}</td></tr>`; s += `<tr><td class="label">${escHtml(t("report.docBlistersPerPack"))}</td><td>${med.blistersPerPack}</td></tr>`;
s += `<tr><td class="label">${escHtml(t("report.docPillsPerBlister"))}</td><td>${med.pillsPerBlister}</td></tr>`; s += `<tr><td class="label">${escHtml(t("report.docPillsPerBlister"))}</td><td>${med.pillsPerBlister}</td></tr>`;
if (med.looseTablets > 0) if (med.looseTablets > 0)
s += `<tr><td class="label">${escHtml(t("report.docLoosePills"))}</td><td>${med.looseTablets}</td></tr>`; s += `<tr><td class="label">${escHtml(t("report.docLoosePills"))}</td><td>${med.looseTablets}</td></tr>`;
} else { } else {
s += `<tr><td class="label">${escHtml(t("report.docTotalCapacity"))}</td><td>${med.totalPills ?? med.looseTablets}</td></tr>`; s += `<tr><td class="label">${escHtml(getTotalCapacityLabel(med, t))}</td><td>${med.totalPills ?? med.looseTablets}</td></tr>`;
} }
s += `<tr><td class="label">${escHtml(t("report.docCurrentStock"))}</td><td>${getPackageSize(med)} ${escHtml(t("common.pills"))}</td></tr>`; s += `<tr><td class="label">${escHtml(t("report.docCurrentStock"))}</td><td>${escHtml(getCurrentStockText(med, t))}</td></tr>`;
if (med.pillWeightMg) if (!isTubePackageType(med.packageType) && !isLiquidContainerPackageType(med.packageType) && med.pillWeightMg)
s += `<tr><td class="label">${escHtml(t("report.docDosePerPill"))}</td><td>${med.pillWeightMg} ${escHtml(med.doseUnit ?? "mg")}</td></tr>`; s += `<tr><td class="label">${escHtml(t("report.docDosePerPill"))}</td><td>${med.pillWeightMg} ${escHtml(med.doseUnit ?? "mg")}</td></tr>`;
if (med.expiryDate) if (med.expiryDate)
s += `<tr><td class="label">${escHtml(t("report.docExpiryDate"))}</td><td>${fmtDate(med.expiryDate)}</td></tr>`; s += `<tr><td class="label">${escHtml(t("report.docExpiryDate"))}</td><td>${fmtDate(med.expiryDate)}</td></tr>`;
@@ -563,7 +605,7 @@ function buildPrintHtml(
s += `<h3>${escHtml(t("report.docIntakeSchedule"))}</h3>`; s += `<h3>${escHtml(t("report.docIntakeSchedule"))}</h3>`;
s += `<ul>`; s += `<ul>`;
for (const intake of filteredPrintIntakes) { for (const intake of filteredPrintIntakes) {
let entry = `${intake.usage} ${escHtml(intake.usage === 1 ? t("common.pill") : t("common.pills"))}`; let entry = escHtml(getUsageText(med, intake.usage, t));
entry += ` ${escHtml(intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every }))}`; entry += ` ${escHtml(intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every }))}`;
entry += ` ${escHtml(t("form.blisters.from"))} ${fmtDateTime(intake.start)}`; entry += ` ${escHtml(t("form.blisters.from"))} ${fmtDateTime(intake.start)}`;
if ("takenBy" in intake && intake.takenBy) if ("takenBy" in intake && intake.takenBy)
@@ -610,7 +652,7 @@ function buildPrintHtml(
s += `<h3>${escHtml(t("report.docRefillHistory"))}</h3>`; s += `<h3>${escHtml(t("report.docRefillHistory"))}</h3>`;
s += `<ul>`; s += `<ul>`;
for (const r of data.refills) { for (const r of data.refills) {
let entry = `${fmtDate(r.refillDate)}: +${r.packsAdded} ${escHtml(t("report.docPacks"))}, +${r.loosePillsAdded} ${escHtml(t("common.pills"))}`; let entry = `${fmtDate(r.refillDate)}: +${r.packsAdded} ${escHtml(t("report.docPacks"))}, +${r.loosePillsAdded} ${escHtml(isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType) ? t(getTubeUnitKey(med)) : t("common.pills"))}`;
if (r.usedPrescription) entry += ` <em>${escHtml(t("report.docRefillPrescription"))}</em>`; if (r.usedPrescription) entry += ` <em>${escHtml(t("report.docRefillPrescription"))}</em>`;
s += `<li>${entry}</li>`; s += `<li>${entry}</li>`;
} }
+1 -2
View File
@@ -5,7 +5,6 @@
import { Check, Copy, Link2, X } from "lucide-react"; import { Check, Copy, Link2, X } from "lucide-react";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { useEscapeKey } from "../hooks/useEscapeKey";
export interface ShareDialogProps { export interface ShareDialogProps {
show: boolean; show: boolean;
@@ -44,7 +43,7 @@ export function ShareDialog({
const closeLabel = t("common.close"); const closeLabel = t("common.close");
const copyLabel = shareCopied ? t("share.copied") : t("share.copyLink"); const copyLabel = shareCopied ? t("share.copied") : t("share.copyLink");
useEscapeKey(show, onClose); // ESC is handled by the global handler in App.tsx to avoid double history.back()
if (!show) return null; if (!show) return null;
+205 -54
View File
@@ -9,7 +9,7 @@ import { useTranslation } from "react-i18next";
import { useParams } from "react-router-dom"; import { useParams } from "react-router-dom";
import { useEscapeKey } from "../hooks"; import { useEscapeKey } from "../hooks";
import type { ExpiredLinkData, SharedScheduleData } from "../types"; import type { ExpiredLinkData, SharedScheduleData } from "../types";
import { getMedTotal } from "../types"; import { getMedDisplayName, getMedTotal, isLiquidContainerPackageType, isTubePackageType } from "../types";
import { getSystemLocale } from "../utils/formatters"; import { getSystemLocale } from "../utils/formatters";
import { isDoseDismissed } from "../utils/schedule"; import { isDoseDismissed } from "../utils/schedule";
import { loadCollapsedDaysFromStorage } from "../utils/storage"; import { loadCollapsedDaysFromStorage } from "../utils/storage";
@@ -21,11 +21,20 @@ import { MedicationAvatar } from "./MedicationAvatar";
function getStockStatus( function getStockStatus(
daysLeft: number | null, daysLeft: number | null,
medsLeft: number, medsLeft: number,
thresholds: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number } thresholds: { lowStockDays: number; normalStockDays: number; highStockDays: number; criticalStockDays: number },
packageType?: string
) { ) {
if (isTubePackageType(packageType)) return { className: "success", label: "status.noSchedule" };
if (medsLeft <= 0 || daysLeft === 0) return { className: "danger", label: "status.outOfStock" }; if (medsLeft <= 0 || daysLeft === 0) return { className: "danger", label: "status.outOfStock" };
if (daysLeft === null) return { className: "success", label: "status.noSchedule" }; if (daysLeft === null) return { className: "success", label: "status.noSchedule" };
if (daysLeft <= thresholds.reminderDaysBefore) return { className: "danger", label: "status.criticalStock" }; if (isLiquidContainerPackageType(packageType)) {
const lowDays = Math.max(1, Math.floor(thresholds.criticalStockDays));
const criticalDays = Math.max(1, Math.ceil(lowDays / 2));
if (daysLeft <= criticalDays) return { className: "danger", label: "status.criticalStock" };
if (daysLeft <= lowDays) return { className: "warning", label: "status.lowStock" };
return { className: "success", label: "status.normal" };
}
if (daysLeft <= thresholds.criticalStockDays) return { className: "danger", label: "status.criticalStock" };
if (daysLeft < thresholds.lowStockDays) return { className: "warning", label: "status.lowStock" }; if (daysLeft < thresholds.lowStockDays) return { className: "warning", label: "status.lowStock" };
if (daysLeft >= thresholds.highStockDays) return { className: "high", label: "status.highStock" }; if (daysLeft >= thresholds.highStockDays) return { className: "high", label: "status.highStock" };
return { className: "success", label: "status.normal" }; return { className: "success", label: "status.normal" };
@@ -44,6 +53,100 @@ export function SharedSchedule() {
const [showPastDays, setShowPastDays] = useState(false); const [showPastDays, setShowPastDays] = useState(false);
const [showFutureDays, setShowFutureDays] = useState(false); const [showFutureDays, setShowFutureDays] = useState(false);
const isLiquidContainerMed = (med: SharedScheduleData["medications"][number] | undefined) =>
isLiquidContainerPackageType(med?.packageType);
const convertLiquidUsageToMl = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): number => {
if (unit === "tsp") return usage * 5;
if (unit === "tbsp") return usage * 15;
return usage;
};
const convertUsageForStock = (
usage: number,
med: SharedScheduleData["medications"][number] | undefined,
unit: "ml" | "tsp" | "tbsp" | null | undefined
): number => {
if (isTubePackageType(med?.packageType)) return 0;
if (!isLiquidContainerMed(med)) return usage;
return convertLiquidUsageToMl(usage, unit);
};
const formatAmount = (value: number) => {
const rounded = Math.round(value * 100) / 100;
return String(rounded);
};
const getLiquidCountUnitLabel = (unit: "ml" | "tsp" | "tbsp" | null | undefined, usage: number): string => {
if (unit === "tsp") return t("form.blisters.teaspoons", { count: Math.abs(usage) });
if (unit === "tbsp") return t("form.blisters.tablespoons", { count: Math.abs(usage) });
return t("form.packageAmountUnitMl");
};
const formatLiquidUsageLabel = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): string => {
const normalizedUsage = Number(usage);
if (!Number.isFinite(normalizedUsage) || normalizedUsage <= 0) {
return `0 ${t("form.packageAmountUnitMl")}`;
}
if (unit === "ml" || unit == null) {
return `${formatAmount(normalizedUsage)} ${t("form.packageAmountUnitMl")}`;
}
const mlTotal = convertLiquidUsageToMl(normalizedUsage, unit);
return `${formatAmount(normalizedUsage)} ${getLiquidCountUnitLabel(unit, normalizedUsage)} ${formatAmount(mlTotal)} ${t("form.packageAmountUnitMl")}`;
};
const formatDoseUsageLabel = (
med: SharedScheduleData["medications"][number] | undefined,
usage: number,
intakeUnit?: "ml" | "tsp" | "tbsp" | null
) => {
if (isLiquidContainerMed(med)) {
return formatLiquidUsageLabel(usage, intakeUnit);
}
return `${usage} ${usage !== 1 ? t("common.pills") : t("common.pill")}`;
};
const formatTotalUsageLabel = (
med: SharedScheduleData["medications"][number] | undefined,
total: number,
doses?: Array<{ usage: number; intakeUnit?: "ml" | "tsp" | "tbsp" | null }>
) => {
if (isLiquidContainerMed(med)) {
if (doses && doses.length > 0) {
const normalizedDoses = doses.filter((dose) => Number.isFinite(Number(dose.usage)) && Number(dose.usage) > 0);
if (normalizedDoses.length > 0) {
const allUnits = new Set(normalizedDoses.map((dose) => dose.intakeUnit ?? "ml"));
if (allUnits.size === 1) {
const onlyUnit = normalizedDoses[0]?.intakeUnit ?? "ml";
const totalUsageInUnit = normalizedDoses.reduce((sum, dose) => sum + Number(dose.usage), 0);
return formatLiquidUsageLabel(totalUsageInUnit, onlyUnit);
}
const totalMl = normalizedDoses.reduce(
(sum, dose) => sum + convertLiquidUsageToMl(Number(dose.usage), dose.intakeUnit ?? "ml"),
0
);
return `${formatAmount(totalMl)} ${t("form.packageAmountUnitMl")}`;
}
}
return `${formatAmount(total)} ${t("form.packageAmountUnitMl")}`;
}
return t("common.pillsTotal", { count: total });
};
const shouldHideNoScheduleStatusForTube = (
med: SharedScheduleData["medications"][number] | undefined,
status: { className: string; label: string } | null
) => isTubePackageType(med?.packageType) && status?.label === "status.noSchedule";
const getVisibleStockStatus = (
med: SharedScheduleData["medications"][number] | undefined,
status: { className: string; label: string } | null
) => (shouldHideNoScheduleStatusForTube(med, status) ? null : status);
// Theme preference: light, dark, or system // Theme preference: light, dark, or system
type ThemePreference = "light" | "dark" | "system"; type ThemePreference = "light" | "dark" | "system";
const [themePreference, setThemePreference] = useState<ThemePreference>(() => { const [themePreference, setThemePreference] = useState<ThemePreference>(() => {
@@ -309,6 +412,7 @@ export function SharedSchedule() {
when: number; when: number;
medName: string; medName: string;
usage: number; usage: number;
intakeUnit?: "ml" | "tsp" | "tbsp" | null;
timeStr: string; timeStr: string;
isPast: boolean; isPast: boolean;
takenBy: string | null; // Per-intake takenBy (single person or null) takenBy: string | null; // Per-intake takenBy (single person or null)
@@ -319,7 +423,12 @@ export function SharedSchedule() {
// Use intakes (with per-intake takenBy) if available, fallback to blisters (legacy) // Use intakes (with per-intake takenBy) if available, fallback to blisters (legacy)
const intakes = const intakes =
med.intakes || med.intakes ||
med.blisters.map((b) => ({ ...b, takenBy: null as string | null, intakeRemindersEnabled: false })); med.blisters.map((b) => ({
...b,
intakeUnit: null,
takenBy: null as string | null,
intakeRemindersEnabled: false,
}));
intakes.forEach((intake, intakeIdx) => { intakes.forEach((intake, intakeIdx) => {
// Filter: only include intakes for this person (null = everyone, or matches share's takenBy) // Filter: only include intakes for this person (null = everyone, or matches share's takenBy)
@@ -343,8 +452,9 @@ export function SharedSchedule() {
doses.push({ doses.push({
id: doseId, id: doseId,
when: t, when: t,
medName: med.name, medName: getMedDisplayName(med),
usage: intake.usage, usage: intake.usage,
intakeUnit: intake.intakeUnit ?? null,
isPast, isPast,
takenBy: intake.takenBy, // Per-intake takenBy (string | null) takenBy: intake.takenBy, // Per-intake takenBy (string | null)
timeStr: d.toLocaleTimeString(getSystemLocale(i18n.language), { hour: "2-digit", minute: "2-digit" }), timeStr: d.toLocaleTimeString(getSystemLocale(i18n.language), { hour: "2-digit", minute: "2-digit" }),
@@ -430,8 +540,14 @@ export function SharedSchedule() {
const depletion: Record<string, number | null> = {}; const depletion: Record<string, number | null> = {};
for (const med of data.medications) { for (const med of data.medications) {
const intakes = med.intakes || med.blisters.map((b) => ({ ...b, takenBy: null as string | null })); const intakes =
const blisters = med.blisters; med.intakes ||
med.blisters.map((b) => ({
...b,
intakeUnit: null,
takenBy: null as string | null,
intakeRemindersEnabled: false,
}));
// Count unique people from all intakes (for per-intake takenBy) // Count unique people from all intakes (for per-intake takenBy)
const uniquePeople = new Set<string>(); const uniquePeople = new Set<string>();
@@ -443,9 +559,9 @@ export function SharedSchedule() {
// Calculate daily consumption rate accounting for per-intake takenBy // Calculate daily consumption rate accounting for per-intake takenBy
let dailyRate = 0; let dailyRate = 0;
blisters.forEach((s, idx) => { intakes.forEach((intake) => {
const baseRate = s.every > 0 ? s.usage / s.every : 0; const usageForStock = convertUsageForStock(intake.usage, med, intake.intakeUnit ?? "ml");
const intake = intakes[idx]; const baseRate = intake.every > 0 ? usageForStock / intake.every : 0;
if (intake?.takenBy) { if (intake?.takenBy) {
dailyRate += baseRate; // Per-intake takenBy: 1 person dailyRate += baseRate; // Per-intake takenBy: 1 person
} else { } else {
@@ -458,9 +574,10 @@ export function SharedSchedule() {
if (calcMode === "automatic") { if (calcMode === "automatic") {
// Time-based: every scheduled dose counts as consumed once its time has passed // Time-based: every scheduled dose counts as consumed once its time has passed
blisters.forEach((s, blisterIdx) => { intakes.forEach((intake, blisterIdx) => {
const blisterStart = new Date(s.start).getTime(); const usageForStock = convertUsageForStock(intake.usage, med, intake.intakeUnit ?? "ml");
const period = Math.max(1, s.every) * MS_PER_DAY; const blisterStart = new Date(intake.start).getTime();
const period = Math.max(1, intake.every) * MS_PER_DAY;
let effectiveStart: number; let effectiveStart: number;
if (stockCorrectionCutoff > 0 && stockCorrectionCutoff >= blisterStart) { if (stockCorrectionCutoff > 0 && stockCorrectionCutoff >= blisterStart) {
@@ -472,7 +589,6 @@ export function SharedSchedule() {
} }
if (Number.isNaN(effectiveStart)) return; if (Number.isNaN(effectiveStart)) return;
const intake = intakes[blisterIdx];
const intakePerson = intake?.takenBy; const intakePerson = intake?.takenBy;
const fallbackPeople = med.takenBy?.length > 0 ? med.takenBy : [null]; const fallbackPeople = med.takenBy?.length > 0 ? med.takenBy : [null];
const peopleForThisIntake = intakePerson ? [intakePerson] : fallbackPeople; const peopleForThisIntake = intakePerson ? [intakePerson] : fallbackPeople;
@@ -482,7 +598,7 @@ export function SharedSchedule() {
if (effectiveStart <= now) { if (effectiveStart <= now) {
const occurrences = Math.floor((now - effectiveStart) / period) + 1; const occurrences = Math.floor((now - effectiveStart) / period) + 1;
timeBasedConsumed = occurrences * s.usage * peopleForThisIntake.length; timeBasedConsumed = occurrences * usageForStock * peopleForThisIntake.length;
const lastDoseTime = new Date(effectiveStart + (occurrences - 1) * period); const lastDoseTime = new Date(effectiveStart + (occurrences - 1) * period);
lastAutoConsumedDateMs = new Date( lastAutoConsumedDateMs = new Date(
lastDoseTime.getFullYear(), lastDoseTime.getFullYear(),
@@ -510,7 +626,7 @@ export function SharedSchedule() {
const bIdx = parseInt(parts[1], 10); const bIdx = parseInt(parts[1], 10);
const timestamp = parseInt(parts[2], 10); const timestamp = parseInt(parts[2], 10);
if (medId === med.id && bIdx === blisterIdx && timestamp > earlyCutoff) { if (medId === med.id && bIdx === blisterIdx && timestamp > earlyCutoff) {
earlyTakenConsumed += s.usage; earlyTakenConsumed += usageForStock;
} }
} }
} }
@@ -525,8 +641,8 @@ export function SharedSchedule() {
const medId = parseInt(parts[0], 10); const medId = parseInt(parts[0], 10);
const blisterIdx = parseInt(parts[1], 10); const blisterIdx = parseInt(parts[1], 10);
const doseTimestamp = parseInt(parts[2], 10); const doseTimestamp = parseInt(parts[2], 10);
if (medId === med.id && blisters[blisterIdx]) { if (medId === med.id && intakes[blisterIdx]) {
const blisterStartDate = new Date(blisters[blisterIdx].start); const blisterStartDate = new Date(intakes[blisterIdx].start);
const blisterStartDateOnly = new Date( const blisterStartDateOnly = new Date(
blisterStartDate.getFullYear(), blisterStartDate.getFullYear(),
blisterStartDate.getMonth(), blisterStartDate.getMonth(),
@@ -534,7 +650,11 @@ export function SharedSchedule() {
).getTime(); ).getTime();
const afterCorrection = stockCorrectionCutoff === 0 || doseTimestamp > stockCorrectionCutoff; const afterCorrection = stockCorrectionCutoff === 0 || doseTimestamp > stockCorrectionCutoff;
if (!Number.isNaN(blisterStartDateOnly) && doseTimestamp >= blisterStartDateOnly && afterCorrection) { if (!Number.isNaN(blisterStartDateOnly) && doseTimestamp >= blisterStartDateOnly && afterCorrection) {
consumed += blisters[blisterIdx].usage; consumed += convertUsageForStock(
intakes[blisterIdx].usage,
med,
intakes[blisterIdx].intakeUnit ?? "ml"
);
} }
} }
} }
@@ -547,8 +667,8 @@ export function SharedSchedule() {
const daysLeft = rawDaysLeft !== null ? Math.max(0, Math.floor(rawDaysLeft)) : null; const daysLeft = rawDaysLeft !== null ? Math.max(0, Math.floor(rawDaysLeft)) : null;
const depletionMs = daysLeft !== null ? now + daysLeft * MS_PER_DAY : null; const depletionMs = daysLeft !== null ? now + daysLeft * MS_PER_DAY : null;
coverage[med.name] = { daysLeft, medsLeft: Number(medsLeft.toFixed(1)), dailyUsage: dailyRate }; coverage[getMedDisplayName(med)] = { daysLeft, medsLeft: Number(medsLeft.toFixed(1)), dailyUsage: dailyRate };
depletion[med.name] = depletionMs; depletion[getMedDisplayName(med)] = depletionMs;
} }
return { coverageByMed: coverage, depletionByMed: depletion }; return { coverageByMed: coverage, depletionByMed: depletion };
}, [data, takenDoses]); }, [data, takenDoses]);
@@ -569,11 +689,13 @@ export function SharedSchedule() {
function getDayStockStatus(meds: { medName: string; lastWhen: number }[]) { function getDayStockStatus(meds: { medName: string; lastWhen: number }[]) {
const statuses = meds.map((item) => { const statuses = meds.map((item) => {
const coverage = coverageByMed[item.medName]; const coverage = coverageByMed[item.medName];
const med = data?.medications.find((m) => getMedDisplayName(m) === item.medName);
const depletionTime = depletionByMed[item.medName]; const depletionTime = depletionByMed[item.medName];
if (typeof depletionTime === "number" && item.lastWhen > depletionTime) return "danger"; if (typeof depletionTime === "number" && item.lastWhen > depletionTime) return "danger";
if (!coverage) return "success"; if (!coverage) return "success";
const status = getStockStatus(coverage.daysLeft, coverage.medsLeft, stockThresholds); const rawStatus = getStockStatus(coverage.daysLeft, coverage.medsLeft, stockThresholds, med?.packageType);
return status.className; const status = getVisibleStockStatus(med, rawStatus);
return status?.className ?? "success";
}); });
const fallbackStatus = statuses.includes("warning") ? "warning" : "success"; const fallbackStatus = statuses.includes("warning") ? "warning" : "success";
return statuses.includes("danger") ? "danger" : fallbackStatus; return statuses.includes("danger") ? "danger" : fallbackStatus;
@@ -583,6 +705,11 @@ export function SharedSchedule() {
const showStock = data?.shareStockStatus !== false; const showStock = data?.shareStockStatus !== false;
const showOnlyToday = data?.shareScheduleTodayOnly === true && (data?.upcomingTodayOnly ?? true); const showOnlyToday = data?.shareScheduleTodayOnly === true && (data?.upcomingTodayOnly ?? true);
const renderDoseUsage = (
med: SharedScheduleData["medications"][number] | undefined,
dose: { usage: number; intakeUnit?: "ml" | "tsp" | "tbsp" | null }
) => formatDoseUsageLabel(med, dose.usage, dose.intakeUnit);
// Helper: check if a dose is "done" (taken, per-dose dismissed, or med-level dismissed) // Helper: check if a dose is "done" (taken, per-dose dismissed, or med-level dismissed)
function isDoseIdDone(doseId: string): boolean { function isDoseIdDone(doseId: string): boolean {
if (takenDoses.has(doseId)) return true; if (takenDoses.has(doseId)) return true;
@@ -746,7 +873,7 @@ export function SharedSchedule() {
// Count missed doses that are NOT dismissed (for warning icon) // Count missed doses that are NOT dismissed (for warning icon)
const missedNotDismissedCount = day.meds.reduce((count, item) => { const missedNotDismissedCount = day.meds.reduce((count, item) => {
const med = data.medications.find((m) => m.name === item.medName); const med = data.medications.find((m) => getMedDisplayName(m) === item.medName);
const dismissedUntilDate = med?.dismissedUntil ?? undefined; const dismissedUntilDate = med?.dismissedUntil ?? undefined;
return ( return (
count + count +
@@ -800,7 +927,7 @@ export function SharedSchedule() {
</div> </div>
{!isCollapsed && {!isCollapsed &&
day.meds.map((item) => { day.meds.map((item) => {
const med = data.medications.find((m) => m.name === item.medName); const med = data.medications.find((m) => getMedDisplayName(m) === item.medName);
const medCoverage = coverageByMed[item.medName]; const medCoverage = coverageByMed[item.medName];
const isEmpty = showStock && medCoverage ? medCoverage.medsLeft <= 0 : false; const isEmpty = showStock && medCoverage ? medCoverage.medsLeft <= 0 : false;
const depletionTime = depletionByMed[item.medName]; const depletionTime = depletionByMed[item.medName];
@@ -809,9 +936,15 @@ export function SharedSchedule() {
? willBeOutOfStock ? willBeOutOfStock
? { className: "danger", label: "status.outOfStock" } ? { className: "danger", label: "status.outOfStock" }
: medCoverage : medCoverage
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds) ? getStockStatus(
medCoverage.daysLeft,
medCoverage.medsLeft,
stockThresholds,
med?.packageType
)
: null : null
: null; : null;
const visibleStatus = getVisibleStockStatus(med, status);
const itemDoseIds = item.doses.map((d) => d.id); const itemDoseIds = item.doses.map((d) => d.id);
const allTaken = itemDoseIds.every((id) => takenDoses.has(id)); const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
@@ -825,10 +958,10 @@ export function SharedSchedule() {
<div className="med-name"> <div className="med-name">
<div <div
className={med?.imageUrl ? "med-avatar clickable" : ""} className={med?.imageUrl ? "med-avatar clickable" : ""}
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, med.name)} onClick={() => med?.imageUrl && openLightbox(med.imageUrl, getMedDisplayName(med))}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
if (med?.imageUrl) openLightbox(med.imageUrl, med.name); if (med?.imageUrl) openLightbox(med.imageUrl, getMedDisplayName(med));
} }
}} }}
> >
@@ -840,9 +973,13 @@ export function SharedSchedule() {
</div> </div>
</div> </div>
<div className="tag-row"> <div className="tag-row">
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span> <span className="tag subtle">
{status && ( {formatTotalUsageLabel(med, item.total, item.doses)}
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span> </span>
{visibleStatus && (
<span className={`status-chip small ${visibleStatus.className}`}>
{t(visibleStatus.label)}
</span>
)} )}
</div> </div>
</div> </div>
@@ -853,9 +990,7 @@ export function SharedSchedule() {
<div key={dose.id} className="dose-item past"> <div key={dose.id} className="dose-item past">
<span className="dose-time">{dose.timeStr}</span> <span className="dose-time">{dose.timeStr}</span>
<span className="dose-usage"> <span className="dose-usage">
<span className="dose-usage-main"> <span className="dose-usage-main">{renderDoseUsage(med, dose)}</span>
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
</span>
{med?.pillWeightMg && ( {med?.pillWeightMg && (
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span> <span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
)} )}
@@ -984,7 +1119,7 @@ export function SharedSchedule() {
</div> </div>
{!isCollapsed && {!isCollapsed &&
day.meds.map((item) => { day.meds.map((item) => {
const med = data.medications.find((m) => m.name === item.medName); const med = data.medications.find((m) => getMedDisplayName(m) === item.medName);
const medCoverage = coverageByMed[item.medName]; const medCoverage = coverageByMed[item.medName];
const isEmpty = showStock && medCoverage ? medCoverage.medsLeft <= 0 : false; const isEmpty = showStock && medCoverage ? medCoverage.medsLeft <= 0 : false;
const depletionTime = depletionByMed[item.medName]; const depletionTime = depletionByMed[item.medName];
@@ -993,9 +1128,15 @@ export function SharedSchedule() {
? willBeOutOfStock ? willBeOutOfStock
? { className: "danger", label: "status.outOfStock" } ? { className: "danger", label: "status.outOfStock" }
: medCoverage : medCoverage
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds) ? getStockStatus(
medCoverage.daysLeft,
medCoverage.medsLeft,
stockThresholds,
med?.packageType
)
: null : null
: null; : null;
const visibleStatus = getVisibleStockStatus(med, status);
const itemDoseIds = item.doses.map((d) => d.id); const itemDoseIds = item.doses.map((d) => d.id);
const allTaken = itemDoseIds.every((id) => takenDoses.has(id)); const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
@@ -1008,10 +1149,10 @@ export function SharedSchedule() {
<div className="med-name"> <div className="med-name">
<div <div
className={med?.imageUrl ? "med-avatar clickable" : ""} className={med?.imageUrl ? "med-avatar clickable" : ""}
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, med.name)} onClick={() => med?.imageUrl && openLightbox(med.imageUrl, getMedDisplayName(med))}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
if (med?.imageUrl) openLightbox(med.imageUrl, med.name); if (med?.imageUrl) openLightbox(med.imageUrl, getMedDisplayName(med));
} }
}} }}
> >
@@ -1023,9 +1164,13 @@ export function SharedSchedule() {
</div> </div>
</div> </div>
<div className="tag-row"> <div className="tag-row">
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span> <span className="tag subtle">
{status && ( {formatTotalUsageLabel(med, item.total, item.doses)}
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span> </span>
{visibleStatus && (
<span className={`status-chip small ${visibleStatus.className}`}>
{t(visibleStatus.label)}
</span>
)} )}
</div> </div>
</div> </div>
@@ -1040,9 +1185,7 @@ export function SharedSchedule() {
> >
<span className="dose-time">{dose.timeStr}</span> <span className="dose-time">{dose.timeStr}</span>
<span className="dose-usage"> <span className="dose-usage">
<span className="dose-usage-main"> <span className="dose-usage-main">{renderDoseUsage(med, dose)}</span>
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
</span>
{med?.pillWeightMg && ( {med?.pillWeightMg && (
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span> <span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
)} )}
@@ -1161,7 +1304,7 @@ export function SharedSchedule() {
</div> </div>
{!isCollapsed && {!isCollapsed &&
day.meds.map((item) => { day.meds.map((item) => {
const med = data.medications.find((m) => m.name === item.medName); const med = data.medications.find((m) => getMedDisplayName(m) === item.medName);
const medCoverage = coverageByMed[item.medName]; const medCoverage = coverageByMed[item.medName];
const depletionTime = depletionByMed[item.medName]; const depletionTime = depletionByMed[item.medName];
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime; const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
@@ -1169,9 +1312,15 @@ export function SharedSchedule() {
? willBeOutOfStock ? willBeOutOfStock
? { className: "danger", label: "status.outOfStock" } ? { className: "danger", label: "status.outOfStock" }
: medCoverage : medCoverage
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds) ? getStockStatus(
medCoverage.daysLeft,
medCoverage.medsLeft,
stockThresholds,
med?.packageType
)
: null : null
: null; : null;
const visibleStatus = getVisibleStockStatus(med, status);
const itemDoseIds = item.doses.map((d) => d.id); const itemDoseIds = item.doses.map((d) => d.id);
const allTaken = itemDoseIds.every((id) => takenDoses.has(id)); const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
@@ -1184,10 +1333,10 @@ export function SharedSchedule() {
<div className="med-name"> <div className="med-name">
<div <div
className={med?.imageUrl ? "med-avatar clickable" : ""} className={med?.imageUrl ? "med-avatar clickable" : ""}
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, med.name)} onClick={() => med?.imageUrl && openLightbox(med.imageUrl, getMedDisplayName(med))}
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
if (med?.imageUrl) openLightbox(med.imageUrl, med.name); if (med?.imageUrl) openLightbox(med.imageUrl, getMedDisplayName(med));
} }
}} }}
> >
@@ -1199,9 +1348,13 @@ export function SharedSchedule() {
</div> </div>
</div> </div>
<div className="tag-row"> <div className="tag-row">
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span> <span className="tag subtle">
{status && ( {formatTotalUsageLabel(med, item.total, item.doses)}
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span> </span>
{visibleStatus && (
<span className={`status-chip small ${visibleStatus.className}`}>
{t(visibleStatus.label)}
</span>
)} )}
</div> </div>
</div> </div>
@@ -1212,9 +1365,7 @@ export function SharedSchedule() {
<div key={dose.id} className={`dose-item future ${isTaken ? "all-taken" : ""}`}> <div key={dose.id} className={`dose-item future ${isTaken ? "all-taken" : ""}`}>
<span className="dose-time">{dose.timeStr}</span> <span className="dose-time">{dose.timeStr}</span>
<span className="dose-usage"> <span className="dose-usage">
<span className="dose-usage-main"> <span className="dose-usage-main">{renderDoseUsage(med, dose)}</span>
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
</span>
{med?.pillWeightMg && ( {med?.pillWeightMg && (
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span> <span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
)} )}
+7 -7
View File
@@ -6,7 +6,7 @@ import { useTranslation } from "react-i18next";
import { MedicationAvatar } from "../components"; import { MedicationAvatar } from "../components";
import { useEscapeKey } from "../hooks/useEscapeKey"; import { useEscapeKey } from "../hooks/useEscapeKey";
import type { Coverage, Medication, StockThresholds } from "../types"; import type { Coverage, Medication, StockThresholds } from "../types";
import { getMedTotal, getPackageSize } from "../types"; import { getMedDisplayName, getMedTotal, getPackageSize } from "../types";
import { formatNumber } from "../utils"; import { formatNumber } from "../utils";
import { getSystemLocale } from "../utils/formatters"; import { getSystemLocale } from "../utils/formatters";
import { getStockStatus } from "../utils/schedule"; import { getStockStatus } from "../utils/schedule";
@@ -64,11 +64,11 @@ export function UserFilterModal({
<div className="user-meds-list"> <div className="user-meds-list">
{userMeds.map((med) => { {userMeds.map((med) => {
const medCoverage = coverage.all.find((c) => c.name === med.name); const medCoverage = coverage.all.find((c) => c.name === getMedDisplayName(med));
// Fallback: if no coverage data (e.g. obsolete med), compute basic status from total pills // Fallback: if no coverage data (e.g. obsolete med), compute basic status from total pills
const status = medCoverage const status = medCoverage
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings) ? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings, med.packageType)
: getStockStatus(null, getMedTotal(med), settings); : getStockStatus(null, getMedTotal(med), settings, med.packageType);
const packageSize = getPackageSize(med); const packageSize = getPackageSize(med);
const currentStock = medCoverage ? formatNumber(medCoverage.medsLeft) : formatNumber(getMedTotal(med)); const currentStock = medCoverage ? formatNumber(medCoverage.medsLeft) : formatNumber(getMedTotal(med));
@@ -97,10 +97,10 @@ export function UserFilterModal({
} }
}} }}
> >
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="sm" /> <MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="sm" />
<div className="user-med-info"> <div className="user-med-info">
<span className="user-med-name">{med.name}</span> <span className="user-med-name">{getMedDisplayName(med)}</span>
{med.genericName && <span className="user-med-generic">{med.genericName}</span>} {med.name && med.genericName && <span className="user-med-generic">{med.genericName}</span>}
{personIntakes.length > 0 && ( {personIntakes.length > 0 && (
<div className="user-med-intakes"> <div className="user-med-intakes">
{personIntakes.map((intake) => { {personIntakes.map((intake) => {
+25 -16
View File
@@ -6,7 +6,7 @@ import { useCollapsedDays, useDoses, useMedications, useRefill, useSettings, use
import type { Coverage, FormState, Medication, ScheduleEvent, StockThresholds } from "../types"; import type { Coverage, FormState, Medication, ScheduleEvent, StockThresholds } from "../types";
import { getSystemLocale } from "../utils/formatters"; import { getSystemLocale } from "../utils/formatters";
import { log } from "../utils/logger"; import { log } from "../utils/logger";
import { buildSchedulePreview, calculateCoverage, computeMissedPastDoseIds } from "../utils/schedule"; import { buildSchedulePreview, calculateCoverage, computeMissedPastDoseIds, getStockStatus } from "../utils/schedule";
// ============================================================================= // =============================================================================
// Types // Types
@@ -17,6 +17,7 @@ export type DoseInfo = {
timeStr: string; timeStr: string;
when: number; when: number;
usage: number; usage: number;
intakeUnit?: "ml" | "tsp" | "tbsp" | null;
takenBy: string[]; takenBy: string[];
intakeRemindersEnabled: boolean; intakeRemindersEnabled: boolean;
}; };
@@ -384,6 +385,7 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
(dayMeds: { medName: string; lastWhen: number }[]): "success" | "warning" | "danger" => { (dayMeds: { medName: string; lastWhen: number }[]): "success" | "warning" | "danger" => {
const statuses = dayMeds.map((item) => { const statuses = dayMeds.map((item) => {
const cov = coverageByMed[item.medName]; const cov = coverageByMed[item.medName];
const med = activeMeds.find((m) => m.name === item.medName || m.genericName === item.medName);
const depletionTime = depletionByMed[item.medName]; const depletionTime = depletionByMed[item.medName];
// Will be out of stock by this day? // Will be out of stock by this day?
@@ -392,21 +394,15 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
} }
if (!cov) return "success"; if (!cov) return "success";
const { daysLeft, medsLeft } = cov; const status = getStockStatus(cov.daysLeft, cov.medsLeft, stockThresholds, med?.packageType);
if (status.className === "danger") return "danger";
// Currently out of stock if (status.className === "warning") return "warning";
if (medsLeft <= 0 || daysLeft === 0) return "danger";
// No schedule (can't calculate)
if (daysLeft === null) return "success";
// Low stock: < lowStockDays (warning)
if (daysLeft < settingsHook.settings.lowStockDays) return "warning";
// Normal/High stock
return "success"; return "success";
}); });
const fallbackStatus = statuses.includes("warning") ? "warning" : "success"; const fallbackStatus = statuses.includes("warning") ? "warning" : "success";
return statuses.includes("danger") ? "danger" : fallbackStatus; return statuses.includes("danger") ? "danger" : fallbackStatus;
}, },
[coverageByMed, depletionByMed, settingsHook.settings.lowStockDays] [coverageByMed, depletionByMed, activeMeds, stockThresholds]
); );
const groupedSchedule = useMemo(() => { const groupedSchedule = useMemo(() => {
@@ -439,6 +435,7 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
timeStr: event.timeStr, timeStr: event.timeStr,
when: event.when, when: event.when,
usage: event.usage, usage: event.usage,
intakeUnit: event.intakeUnit ?? null,
takenBy: event.takenBy ? [event.takenBy] : [], takenBy: event.takenBy ? [event.takenBy] : [],
intakeRemindersEnabled: event.intakeRemindersEnabled, intakeRemindersEnabled: event.intakeRemindersEnabled,
}); });
@@ -665,7 +662,18 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
// Get the response text first to handle non-JSON responses // Get the response text first to handle non-JSON responses
const text = await res.text(); const text = await res.text();
let data: { error?: string; message?: string; imported?: number } = {}; let data: {
error?: string;
message?: string;
imported?:
| {
medications?: number;
doseHistory?: number;
refillHistory?: number;
shareLinks?: number;
}
| number;
} = {};
try { try {
data = text ? JSON.parse(text) : {}; data = text ? JSON.parse(text) : {};
} catch { } catch {
@@ -680,11 +688,12 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
} }
// Show success message in UI instead of browser alert // Show success message in UI instead of browser alert
const importedCounts = typeof data.imported === "object" && data.imported !== null ? data.imported : null;
setImportResult({ setImportResult({
medications: data.imported?.medications || 0, medications: importedCounts?.medications || 0,
doses: data.imported?.doseHistory || 0, doses: importedCounts?.doseHistory || 0,
refills: data.imported?.refillHistory || 0, refills: importedCounts?.refillHistory || 0,
shares: data.imported?.shareLinks || 0, shares: importedCounts?.shareLinks || 0,
}); });
// Reload all data // Reload all data
+6
View File
@@ -27,6 +27,12 @@ export function useEscapeKey(active: boolean, onClose: () => void, options?: { c
if (!active) return; if (!active) return;
const handleKeyDown = (e: KeyboardEvent) => { const handleKeyDown = (e: KeyboardEvent) => {
if (e.key === "Escape" && activeRef.current) { if (e.key === "Escape" && activeRef.current) {
if (capture) {
// In nested modals, consume Escape so parent/global handlers
// do not process the same key press again.
e.preventDefault();
e.stopPropagation();
}
onCloseRef.current(); onCloseRef.current();
} }
}; };
+144 -10
View File
@@ -1,7 +1,13 @@
import { useCallback, useEffect, useMemo, useState } from "react"; import { useCallback, useEffect, useMemo, useState } from "react";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import type { FieldErrors, FormBlister, FormIntake, FormState, Medication } from "../types"; import type { FieldErrors, FormBlister, FormIntake, FormState, Medication } from "../types";
import { FIELD_LIMITS } from "../types"; import {
FIELD_LIMITS,
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
normalizePackageType,
} from "../types";
import { toDateValue, toTimeValue } from "../utils/formatters"; import { toDateValue, toTimeValue } from "../utils/formatters";
export const defaultBlister = (): FormBlister => { export const defaultBlister = (): FormBlister => {
@@ -24,6 +30,7 @@ export const defaultIntake = (takenBy: string = ""): FormIntake => {
every: "1", every: "1",
startDate: toDateValue(now), startDate: toDateValue(now),
startTime: toTimeValue(now), startTime: toTimeValue(now),
intakeUnit: "ml",
takenBy, // Per-intake user assignment (empty string = null/everyone) takenBy, // Per-intake user assignment (empty string = null/everyone)
intakeRemindersEnabled: false, intakeRemindersEnabled: false,
}; };
@@ -33,15 +40,22 @@ export const defaultForm = (): FormState => ({
name: "", name: "",
genericName: "", genericName: "",
takenBy: [], takenBy: [],
medicationForm: "tablet",
pillForm: "tablet",
lifecycleCategory: "refill_when_empty",
packageType: "blister", packageType: "blister",
packCount: "1", packCount: "1",
blistersPerPack: "1", blistersPerPack: "1",
pillsPerBlister: "1", pillsPerBlister: "1",
packageAmountValue: "0",
packageAmountUnit: "ml",
totalPills: "", totalPills: "",
looseTablets: "0", looseTablets: "0",
pillWeightMg: "", pillWeightMg: "",
doseUnit: "mg", doseUnit: "mg",
medicationStartDate: "", medicationStartDate: "",
medicationEndDate: "",
autoMarkObsoleteAfterEndDate: true,
expiryDate: "", expiryDate: "",
notes: "", notes: "",
prescriptionEnabled: false, prescriptionEnabled: false,
@@ -115,9 +129,6 @@ export function useMedicationForm(): UseMedicationFormReturn {
// Skip validation for takenBy array (individual items validated on add) // Skip validation for takenBy array (individual items validated on add)
if (field === "takenBy") return undefined; if (field === "takenBy") return undefined;
const strValue = typeof value === "string" ? value : ""; const strValue = typeof value === "string" ? value : "";
if (field === "name" && (!strValue || strValue.trim().length === 0)) {
return t("common.validation.required");
}
if ("max" in limits && strValue.length > limits.max) { if ("max" in limits && strValue.length > limits.max) {
return t("common.validation.maxLength", { max: limits.max, current: strValue.length }); return t("common.validation.maxLength", { max: limits.max, current: strValue.length });
} }
@@ -150,8 +161,16 @@ export function useMedicationForm(): UseMedicationFormReturn {
const error = validateField(f, form[f]); const error = validateField(f, form[f]);
if (error) errors[f] = error; if (error) errors[f] = error;
}); });
// Cross-field validation: at least one of name or genericName is required
const hasName = form.name && form.name.trim().length > 0;
const hasGenericName = form.genericName && form.genericName.trim().length > 0;
if (!hasName && !hasGenericName) {
const msg = t("common.validation.nameOrGenericRequired");
errors.name = errors.name || msg;
errors.genericName = errors.genericName || msg;
}
setFieldErrors(errors); setFieldErrors(errors);
}, [form.name, form.genericName, form.notes, validateField, form]); }, [form.name, form.genericName, form.notes, validateField, form, t]);
const setBlisterValue = useCallback((idx: number, field: keyof FormBlister, value: string) => { const setBlisterValue = useCallback((idx: number, field: keyof FormBlister, value: string) => {
setForm((prev) => { setForm((prev) => {
@@ -200,6 +219,7 @@ export function useMedicationForm(): UseMedicationFormReturn {
every: String(i.every), every: String(i.every),
startDate: toDateValue(i.start), startDate: toDateValue(i.start),
startTime: toTimeValue(i.start), startTime: toTimeValue(i.start),
intakeUnit: (i.intakeUnit ?? "ml") as FormIntake["intakeUnit"],
takenBy: i.takenBy ?? "", // Convert null to empty string for form takenBy: i.takenBy ?? "", // Convert null to empty string for form
intakeRemindersEnabled: i.intakeRemindersEnabled, intakeRemindersEnabled: i.intakeRemindersEnabled,
})) }))
@@ -208,6 +228,7 @@ export function useMedicationForm(): UseMedicationFormReturn {
every: String(s.every), every: String(s.every),
startDate: toDateValue(s.start), startDate: toDateValue(s.start),
startTime: toTimeValue(s.start), startTime: toTimeValue(s.start),
intakeUnit: "ml" as const,
takenBy: "", // Legacy blisters have no per-intake takenBy takenBy: "", // Legacy blisters have no per-intake takenBy
intakeRemindersEnabled: med.intakeRemindersEnabled ?? false, intakeRemindersEnabled: med.intakeRemindersEnabled ?? false,
})); }));
@@ -215,21 +236,77 @@ export function useMedicationForm(): UseMedicationFormReturn {
const authorizedRefills = Math.max(0, med.prescriptionAuthorizedRefills ?? 0); const authorizedRefills = Math.max(0, med.prescriptionAuthorizedRefills ?? 0);
const remainingRefills = Math.min(Math.max(0, med.prescriptionRemainingRefills ?? 0), authorizedRefills); const remainingRefills = Math.min(Math.max(0, med.prescriptionRemainingRefills ?? 0), authorizedRefills);
const lowRefillThreshold = Math.min(Math.max(0, med.prescriptionLowRefillThreshold ?? 1), authorizedRefills); const lowRefillThreshold = Math.min(Math.max(0, med.prescriptionLowRefillThreshold ?? 1), authorizedRefills);
const packageType = normalizePackageType(med.packageType);
const isTubeOrLiquidPackage = isTubePackageType(packageType) || isLiquidContainerPackageType(packageType);
let normalizedPackCount = String(med.packCount);
let normalizedPackageAmountValue = String(med.packageAmountValue ?? 0);
const bottleTotalPills = med.packageType === "bottle" && med.looseTablets ? String(med.looseTablets) : ""; if (isTubeOrLiquidPackage) {
const safePackCount = isTubePackageType(packageType) ? 1 : Math.max(1, med.packCount || 1);
normalizedPackCount = String(safePackCount);
const rawPackageAmount = Number(med.packageAmountValue ?? 0);
const legacyKnownAmount = Math.max(0, Number(med.totalPills ?? 0), Number(med.looseTablets ?? 0));
if (isTubePackageType(packageType)) {
normalizedPackageAmountValue = String(
legacyKnownAmount > 0 ? legacyKnownAmount : Math.max(1, rawPackageAmount)
);
} else if (rawPackageAmount > 0) {
normalizedPackageAmountValue = String(rawPackageAmount);
} else {
normalizedPackageAmountValue = String(legacyKnownAmount);
}
}
const normalizedDerivedTotal = isTubeOrLiquidPackage
? Math.max(0, (Number(normalizedPackCount) || 0) * (Number(normalizedPackageAmountValue) || 0))
: null;
const bottleTotalPills = isAmountBasedPackageType(packageType) && med.looseTablets ? String(med.looseTablets) : "";
let resolvedForm = med.medicationForm;
if (!resolvedForm) {
if (isTubePackageType(packageType)) {
resolvedForm = "topical";
} else if (isLiquidContainerPackageType(packageType)) {
resolvedForm = "liquid";
} else {
resolvedForm = med.pillForm ?? "tablet";
}
}
const resolvedPillForm = med.pillForm ?? (resolvedForm === "capsule" ? "capsule" : "tablet");
let normalizedPackageAmountUnit = med.packageAmountUnit ?? "ml";
if (isTubePackageType(packageType)) {
normalizedPackageAmountUnit = "g";
} else if (isLiquidContainerPackageType(packageType)) {
normalizedPackageAmountUnit = "ml";
}
let resolvedTotalPills = bottleTotalPills;
if (normalizedDerivedTotal != null) {
resolvedTotalPills = String(normalizedDerivedTotal);
} else if (med.totalPills) {
resolvedTotalPills = String(med.totalPills);
}
const editForm: FormState = { const editForm: FormState = {
name: med.name, name: med.name,
genericName: med.genericName ?? "", genericName: med.genericName ?? "",
takenBy: med.takenBy || [], // Already an array from API takenBy: med.takenBy || [], // Already an array from API
packageType: med.packageType ?? "blister", medicationForm: resolvedForm,
packCount: String(med.packCount), pillForm: resolvedPillForm,
lifecycleCategory: med.lifecycleCategory ?? "refill_when_empty",
packageType,
packCount: normalizedPackCount,
blistersPerPack: String(med.blistersPerPack), blistersPerPack: String(med.blistersPerPack),
pillsPerBlister: String(med.pillsPerBlister), pillsPerBlister: String(med.pillsPerBlister),
totalPills: med.totalPills ? String(med.totalPills) : bottleTotalPills, packageAmountValue: normalizedPackageAmountValue,
looseTablets: String(med.looseTablets), packageAmountUnit: normalizedPackageAmountUnit,
totalPills: resolvedTotalPills,
looseTablets: normalizedDerivedTotal != null ? String(normalizedDerivedTotal) : String(med.looseTablets),
pillWeightMg: med.pillWeightMg ? String(med.pillWeightMg) : "", pillWeightMg: med.pillWeightMg ? String(med.pillWeightMg) : "",
doseUnit: med.doseUnit ?? "mg", doseUnit: med.doseUnit ?? "mg",
medicationStartDate: med.medicationStartDate ?? "", medicationStartDate: med.medicationStartDate ?? "",
medicationEndDate: med.medicationEndDate ?? "",
autoMarkObsoleteAfterEndDate: med.autoMarkObsoleteAfterEndDate ?? true,
expiryDate: med.expiryDate ? med.expiryDate.slice(0, 10) : "", expiryDate: med.expiryDate ? med.expiryDate.slice(0, 10) : "",
notes: med.notes ?? "", notes: med.notes ?? "",
prescriptionEnabled: med.prescriptionEnabled ?? false, prescriptionEnabled: med.prescriptionEnabled ?? false,
@@ -272,6 +349,63 @@ export function useMedicationForm(): UseMedicationFormReturn {
setForm((prev) => { setForm((prev) => {
const next = { ...prev, [key]: value } as FormState; const next = { ...prev, [key]: value } as FormState;
if (key === "packageType") {
if (isTubePackageType(value)) {
next.packCount = "1";
next.packageAmountValue = String(Math.max(1, Number(next.packageAmountValue) || 0));
next.medicationForm = "topical";
next.lifecycleCategory = "treatment_period";
next.doseUnit = "units";
next.packageAmountUnit = "g";
} else if (isLiquidContainerPackageType(value)) {
next.packCount = String(Math.max(1, Number(next.packCount) || 1));
next.packageAmountValue = String(Math.max(1, Number(next.packageAmountValue) || 0));
next.medicationForm = "liquid";
next.lifecycleCategory = "refill_when_empty";
next.doseUnit = "ml";
next.packageAmountUnit = "ml";
next.intakes = next.intakes.map((intake) => ({ ...intake, intakeUnit: intake.intakeUnit || "ml" }));
} else {
next.medicationForm = next.pillForm;
next.lifecycleCategory = "refill_when_empty";
}
}
if (key === "medicationForm") {
if (isTubePackageType(next.packageType)) {
next.medicationForm = "topical";
next.lifecycleCategory = "treatment_period";
next.doseUnit = "units";
next.packageAmountUnit = "g";
} else if (isLiquidContainerPackageType(next.packageType)) {
next.medicationForm = "liquid";
next.lifecycleCategory = "refill_when_empty";
next.doseUnit = "ml";
next.packageAmountUnit = "ml";
next.intakes = next.intakes.map((intake) => ({ ...intake, intakeUnit: intake.intakeUnit || "ml" }));
}
}
if (isTubePackageType(next.packageType)) {
next.packCount = "1";
next.packageAmountUnit = "g";
} else if (isLiquidContainerPackageType(next.packageType)) {
next.packageAmountUnit = "ml";
}
if (key === "pillForm" && value === "capsule") {
next.medicationForm = "capsule";
next.intakes = next.intakes.map((intake) => {
const parsedUsage = Number.parseFloat(intake.usage);
const rounded = Number.isFinite(parsedUsage) ? Math.max(0, Math.round(parsedUsage)) : 1;
return { ...intake, usage: String(rounded || 1) };
});
}
if (key === "pillForm" && value === "tablet") {
next.medicationForm = "tablet";
}
if (key === "prescriptionAuthorizedRefills") { if (key === "prescriptionAuthorizedRefills") {
const raw = String(value); const raw = String(value);
next.prescriptionAuthorizedRefills = raw === "" ? "" : String(parseNonNegativeInt(raw)); next.prescriptionAuthorizedRefills = raw === "" ? "" : String(parseNonNegativeInt(raw));
+83 -26
View File
@@ -1,6 +1,12 @@
import { useCallback, useEffect, useState } from "react"; import { useCallback, useEffect, useState } from "react";
import type { Coverage, FormState, Medication, RefillEntry } from "../types"; import type { Coverage, FormState, Medication, RefillEntry } from "../types";
import { getMedTotal, getPackageSize } from "../types"; import {
getMedTotal,
getPackageSize,
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
} from "../types";
export interface UseRefillReturn { export interface UseRefillReturn {
// Refill state // Refill state
@@ -137,51 +143,96 @@ export function useRefill(): UseRefillReturn {
if (!selectedMed) return; if (!selectedMed) return;
setEditStockSaving(true); setEditStockSaving(true);
try { try {
const isTubePackage = isTubePackageType(selectedMed.packageType);
const isLiquidPackage = isLiquidContainerPackageType(selectedMed.packageType);
const isAmountPackage = isAmountBasedPackageType(selectedMed.packageType);
const liquidAmountPerBottle = Math.max(
1,
Number.isFinite(Number(selectedMed.packageAmountValue)) && Number(selectedMed.packageAmountValue) > 0
? Number(selectedMed.packageAmountValue)
: Math.max(
1,
Math.round(Number(getPackageSize(selectedMed) || 0) / Math.max(1, Number(selectedMed.packCount || 1)))
)
);
// Clamp all fields to non-negative values. // Clamp all fields to non-negative values.
let finalFullBlisters = Math.max(0, editStockFullBlisters); let finalFullBlisters = Math.max(0, editStockFullBlisters);
let finalPartialPills = let finalPartialPills = isAmountPackage
selectedMed.packageType === "bottle" ? Math.max(0, editStockPartialBlisterPills)
? Math.max(0, editStockPartialBlisterPills) : Math.max(0, editStockPartialBlisterPills);
: Math.max(0, editStockPartialBlisterPills);
const finalLoosePills = Math.max(0, editStockLoosePills); const finalLoosePills = Math.max(0, editStockLoosePills);
// Canonicalize blister values: partial overflow becomes additional full blisters. // Canonicalize blister values: partial overflow becomes additional full blisters.
if (selectedMed.packageType !== "bottle" && selectedMed.pillsPerBlister > 0) { if (!isAmountPackage && selectedMed.pillsPerBlister > 0) {
finalFullBlisters += Math.floor(finalPartialPills / selectedMed.pillsPerBlister); finalFullBlisters += Math.floor(finalPartialPills / selectedMed.pillsPerBlister);
finalPartialPills %= selectedMed.pillsPerBlister; finalPartialPills %= selectedMed.pillsPerBlister;
} }
// Structural max = sealed package capacity only (no looseTablets offset). // Structural max = sealed package capacity only (no looseTablets offset).
const structuralMax = const structuralMax = isAmountPackage
selectedMed.packageType === "bottle" ? (selectedMed.totalPills ?? getPackageSize(selectedMed))
? (selectedMed.totalPills ?? getPackageSize(selectedMed)) : selectedMed.packCount * selectedMed.blistersPerPack * selectedMed.pillsPerBlister;
: selectedMed.packCount * selectedMed.blistersPerPack * selectedMed.pillsPerBlister; const correctedLiquidBottleCount = isLiquidPackage
? Math.max(1, finalFullBlisters)
: Math.max(1, selectedMed.packCount);
const liquidStructuralMax = isLiquidPackage
? correctedLiquidBottleCount * liquidAmountPerBottle
: structuralMax;
// For blister meds, only sealed pills are capped to package size. // For blister meds, only sealed pills are capped to package size.
// Loose pills are extra and can be above package size. // Loose pills are extra and can be above package size.
const desiredTotal = let desiredTotal: number;
selectedMed.packageType === "bottle" if (isTubePackage) {
? Math.min(structuralMax, Math.max(0, finalPartialPills)) desiredTotal = Math.max(0, finalPartialPills);
: Math.min(structuralMax, finalFullBlisters * selectedMed.pillsPerBlister + finalPartialPills) + } else if (isAmountPackage) {
finalLoosePills; desiredTotal = Math.min(liquidStructuralMax, Math.max(0, finalPartialPills));
} else {
desiredTotal =
Math.min(structuralMax, finalFullBlisters * selectedMed.pillsPerBlister + finalPartialPills) +
finalLoosePills;
}
// The "base" from DB structure used to compute stockAdjustment differs by type: // The "base" from DB structure used to compute stockAdjustment differs by type:
// - Bottle: looseTablets is the base (not changed during correction) // - Bottle: looseTablets is the base (not changed during correction)
// - Blister: use structuralMax + finalLoosePills as the new base so that // - Blister: use structuralMax + finalLoosePills as the new base so that
// updating looseTablets in the DB doesn't cause a stale-split display bug. // updating looseTablets in the DB doesn't cause a stale-split display bug.
const baseTotal = let baseTotal: number;
selectedMed.packageType === "bottle" if (isLiquidPackage) {
? getPackageSize(selectedMed) // bottle: stockAdjustment relative to fixed looseTablets base baseTotal = liquidStructuralMax;
: structuralMax + finalLoosePills; // blister: base = sealed capacity + NEW loose pills } else if (isAmountPackage) {
baseTotal = getPackageSize(selectedMed); // bottle: stockAdjustment relative to fixed looseTablets base
} else {
baseTotal = structuralMax + finalLoosePills; // blister: base = sealed capacity + NEW loose pills
}
// stockAdjustment = what we need to make getMedTotal() return desiredTotal // stockAdjustment = what we need to make getMedTotal() return desiredTotal
const newStockAdjustment = desiredTotal - baseTotal; const newStockAdjustment = desiredTotal - baseTotal;
// For blister corrections also send the new looseTablets value so the DB // For blister corrections also send the new looseTablets value so the DB
// reflects the actual loose count (avoids stale-split display on reload). // reflects the actual loose count (avoids stale-split display on reload).
const patchBody: { stockAdjustment: number; looseTablets?: number } = { const patchBody: {
stockAdjustment: number;
looseTablets?: number;
totalPills?: number;
packageAmountValue?: number;
packCount?: number;
} = {
stockAdjustment: newStockAdjustment, stockAdjustment: newStockAdjustment,
}; };
if (selectedMed.packageType !== "bottle") { if (isTubePackage) {
// Tube has fixed count=1 and no automatic depletion.
// Correction must update the base amount fields directly.
patchBody.stockAdjustment = 0;
patchBody.packCount = 1;
patchBody.totalPills = desiredTotal;
patchBody.looseTablets = desiredTotal;
patchBody.packageAmountValue = desiredTotal;
} else if (isLiquidPackage) {
// Liquid correction supports bottle-count updates.
// Keep packageAmountValue (ml per bottle) and update capacity base by bottle count.
patchBody.packCount = correctedLiquidBottleCount;
patchBody.totalPills = liquidStructuralMax;
} else if (!isAmountPackage) {
patchBody.looseTablets = finalLoosePills; patchBody.looseTablets = finalLoosePills;
} }
@@ -222,6 +273,7 @@ export function useRefill(): UseRefillReturn {
const openEditStockModal = useCallback((selectedMed: Medication, coverage: { all: Coverage[] }) => { const openEditStockModal = useCallback((selectedMed: Medication, coverage: { all: Coverage[] }) => {
if (!selectedMed) return; if (!selectedMed) return;
setEditStockMedication(selectedMed); setEditStockMedication(selectedMed);
const isAmountPackage = isAmountBasedPackageType(selectedMed.packageType);
// Get current stock from coverage (after consumption) // Get current stock from coverage (after consumption)
const medCoverage = coverage.all.find((c) => c.name === selectedMed.name); const medCoverage = coverage.all.find((c) => c.name === selectedMed.name);
const dbTotal = getMedTotal(selectedMed); const dbTotal = getMedTotal(selectedMed);
@@ -231,15 +283,20 @@ export function useRefill(): UseRefillReturn {
// For blister, keep loose pills separated from sealed blister/partial counts. // For blister, keep loose pills separated from sealed blister/partial counts.
const knownLoose = Math.min(currentStock, Math.max(0, selectedMed.looseTablets)); const knownLoose = Math.min(currentStock, Math.max(0, selectedMed.looseTablets));
const sealedPills = Math.max(0, currentStock - knownLoose); const sealedPills = Math.max(0, currentStock - knownLoose);
const fullBlisters = let fullBlisters: number;
selectedMed.packageType === "bottle" ? 0 : Math.floor(sealedPills / selectedMed.pillsPerBlister); if (isLiquidContainerPackageType(selectedMed.packageType)) {
const partialPills = fullBlisters = Math.max(1, selectedMed.packCount);
selectedMed.packageType === "bottle" ? Math.max(0, currentStock) : sealedPills % selectedMed.pillsPerBlister; } else if (isAmountPackage) {
fullBlisters = 0;
} else {
fullBlisters = Math.floor(sealedPills / selectedMed.pillsPerBlister);
}
const partialPills = isAmountPackage ? Math.max(0, currentStock) : sealedPills % selectedMed.pillsPerBlister;
// Pre-fill with current values // Pre-fill with current values
setEditStockFullBlisters(fullBlisters); setEditStockFullBlisters(fullBlisters);
setEditStockPartialBlisterPills(partialPills); setEditStockPartialBlisterPills(partialPills);
setEditStockLoosePills(selectedMed.packageType === "bottle" ? 0 : knownLoose); setEditStockLoosePills(isAmountPackage ? 0 : knownLoose);
setShowEditStockModal(true); setShowEditStockModal(true);
window.history.pushState({ modal: "editStock" }, ""); window.history.pushState({ modal: "editStock" }, "");
}, []); }, []);
+4
View File
@@ -49,6 +49,8 @@ export interface Settings {
upcomingTodayOnly: boolean; upcomingTodayOnly: boolean;
shareScheduleTodayOnly: boolean; shareScheduleTodayOnly: boolean;
swapDashboardMainSections: boolean; swapDashboardMainSections: boolean;
reminderHour: number;
reminderMinutesBefore: number;
expiryWarningDays: number; expiryWarningDays: number;
} }
@@ -96,6 +98,8 @@ const defaultSettings: Settings = {
upcomingTodayOnly: false, upcomingTodayOnly: false,
shareScheduleTodayOnly: false, shareScheduleTodayOnly: false,
swapDashboardMainSections: false, swapDashboardMainSections: false,
reminderHour: 6,
reminderMinutesBefore: 15,
expiryWarningDays: 30, expiryWarningDays: 30,
}; };
+68 -11
View File
@@ -110,6 +110,7 @@
"fullBlisters": "Volle Blister", "fullBlisters": "Volle Blister",
"openBlister": "Offener Blister", "openBlister": "Offener Blister",
"stock": "Bestand", "stock": "Bestand",
"dailyConsumption": "Taeglicher Verbrauch",
"stockDetails": "Details", "stockDetails": "Details",
"daysLeft": "Tage übrig", "daysLeft": "Tage übrig",
"status": "Status", "status": "Status",
@@ -118,7 +119,8 @@
"expiry": "Ablaufdatum", "expiry": "Ablaufdatum",
"pillsCount": "{{count}} Tabletten", "pillsCount": "{{count}} Tabletten",
"pillsCount_one": "{{count}} Tablette", "pillsCount_one": "{{count}} Tablette",
"pillsCount_other": "{{count}} Tabletten" "pillsCount_other": "{{count}} Tabletten",
"perDayWithUnit": "{{value}} {{unit}}"
}, },
"medications": { "medications": {
"list": { "list": {
@@ -130,7 +132,8 @@
"reactivate": "Reaktivieren", "reactivate": "Reaktivieren",
"obsoleteTitle": "Obsolet ({{count}})", "obsoleteTitle": "Obsolet ({{count}})",
"obsoleteSince": "Beendet", "obsoleteSince": "Beendet",
"started": "Gestartet" "started": "Gestartet",
"emptyState": "Noch keine Medikamente. Fuege dein erstes Medikament hinzu."
}, },
"details": { "details": {
"packs": "Packungen", "packs": "Packungen",
@@ -139,6 +142,7 @@
"loose": "Lose", "loose": "Lose",
"total": "Gesamt", "total": "Gesamt",
"stock": "Bestand", "stock": "Bestand",
"capacityPerPackage": "Kapazitaet pro Packung",
"totalCapacity": "Kapazität", "totalCapacity": "Kapazität",
"type": "Typ" "type": "Typ"
}, },
@@ -167,18 +171,41 @@
"commercialName": "Handelsname", "commercialName": "Handelsname",
"genericName": "Wirkstoff", "genericName": "Wirkstoff",
"takenBy": "Eingenommen von", "takenBy": "Eingenommen von",
"medicationForm": "Medikationsform",
"medicationFormCapsule": "Kapsel",
"medicationFormTablet": "Tablette",
"medicationFormLiquid": "Fluessigkeit",
"medicationFormTopical": "Topisch",
"pillForm": "Pillenform",
"lifecycleCategory": "Lebenszyklus",
"lifecycleRefillWhenEmpty": "Nachfuellen wenn leer",
"lifecycleTreatmentPeriod": "Behandlungszeitraum",
"packageType": "Verpackungsart", "packageType": "Verpackungsart",
"packageTypeBlister": "Blisterpackung", "packageTypeBlister": "Blisterpackung",
"packageTypeBottle": "Pillendose", "packageTypeBottle": "Pillendose",
"packageTypeTube": "Tube",
"packageTypeLiquidContainer": "Fluessigbehaeltnis",
"packs": "Packungen", "packs": "Packungen",
"bottles": "Flaschen",
"tubes": "Tuben",
"blistersPerPack": "Blister pro Packung", "blistersPerPack": "Blister pro Packung",
"pillsPerBlister": "Tabletten pro Blister", "pillsPerBlister": "Tabletten pro Blister",
"totalCapacity": "Gesamtkapazität", "totalCapacity": "Gesamtkapazität",
"currentPills": "Aktuelle Tabletten", "currentPills": "Aktuelle Tabletten",
"totalAmount": "Gesamtmenge",
"currentAmount": "Aktuelle Menge",
"totalAmountLabel": "Gesamt (Menge)",
"packageAmount": "Packungsinhalt",
"packageAmountPerBottle": "Inhalt pro Flasche",
"packageAmountPerTube": "Inhalt pro Tube",
"packageAmountUnitMl": "ml",
"packageAmountUnitG": "g",
"loosePills": "Lose Tabletten", "loosePills": "Lose Tabletten",
"pillWeight": "Dosis pro Tablette", "pillWeight": "Dosis pro Tablette",
"total": "Gesamt (Tabletten)", "total": "Gesamt (Tabletten)",
"medicationStartDate": "Startdatum der Medikation", "medicationStartDate": "Startdatum der Medikation",
"medicationEndDate": "Enddatum der Medikation",
"autoMarkObsoleteAfterEndDate": "Nach Enddatum automatisch als obsolet markieren",
"expiryDate": "Ablaufdatum", "expiryDate": "Ablaufdatum",
"notes": "Notizen", "notes": "Notizen",
"medicationImage": "Medikamentenbild", "medicationImage": "Medikamentenbild",
@@ -192,21 +219,44 @@
}, },
"placeholders": { "placeholders": {
"commercial": "z.B. Ozempic", "commercial": "z.B. Ozempic",
"generic": "z.B. Semaglutid (optional)", "generic": "z.B. Semaglutid",
"takenBy": "Name eingeben und Enter drücken", "takenBy": "Name eingeben und Enter drücken",
"addPerson": "Weitere Person hinzufügen...", "addPerson": "Weitere Person hinzufügen...",
"weight": "z.B. 240", "weight": "z.B. 240",
"notes": "z.B. Mit Essen einnehmen, Alkohol vermeiden... (optional)" "notes": "z.B. Mit Essen einnehmen, Alkohol vermeiden... (optional)"
}, },
"validation": { "validation": {
"startDateAfterIntake": "Das Medikations-Startdatum ({{medicationStartDate}}) darf nicht nach dem Einnahmedatum ({{intakeDate}}) liegen." "startDateAfterIntake": "Das Medikations-Startdatum ({{medicationStartDate}}) darf nicht nach dem Einnahmedatum ({{intakeDate}}) liegen.",
}, "endDateBeforeStart": "Das Medikations-Enddatum ({{medicationEndDate}}) darf nicht vor dem Startdatum ({{medicationStartDate}}) liegen."
},
"blisters": { "blisters": {
"title": "Einnahmeplan", "title": "Einnahmeplan",
"remind": "Erinnern", "remind": "Erinnern",
"remindTooltip": "Erhalte eine Benachrichtigung 15 Minuten vor jeder geplanten Einnahme", "remindTooltip": "Erhalte eine Benachrichtigung 15 Minuten vor jeder geplanten Einnahme",
"addIntake": "Einnahme", "addIntake": "Einnahme",
"usage": "Dosis (Tabletten)", "usage": "Dosis",
"usageTablets": "Dosis (Tabletten)",
"usageCapsules": "Dosis (Kapseln)",
"usageMl": "Dosis (ml)",
"usageTsp": "Dosis (tsp)",
"usageTbsp": "Dosis (tbsp)",
"usageApplication": "Dosis (Anwendungen)",
"intakeUnit": "Einnahmeeinheit",
"intakeUnitMl": "Milliliter (ml)",
"intakeUnitTsp": "Teeloeffel (5 ml)",
"intakeUnitTbsp": "Essloeffel (15 ml)",
"intakes": "Einnahmen",
"intakes_one": "Einnahme",
"intakes_other": "Einnahmen",
"teaspoons": "Teeloeffel",
"teaspoons_one": "Teeloeffel",
"teaspoons_other": "Teeloeffel",
"tablespoons": "Essloeffel",
"tablespoons_one": "Essloeffel",
"tablespoons_other": "Essloeffel",
"applications": "Anwendungen",
"applications_one": "Anwendung",
"applications_other": "Anwendungen",
"everyDays": "Alle (Tage)", "everyDays": "Alle (Tage)",
"every": "alle", "every": "alle",
"from": "ab", "from": "ab",
@@ -273,9 +323,9 @@
"schedule": { "schedule": {
"title": "Erinnerungsplan", "title": "Erinnerungsplan",
"stockCheck": "Bestands- & Rezeptprüfung", "stockCheck": "Bestands- & Rezeptprüfung",
"dailyAt6": "Täglich um 6:00 Uhr", "dailyAtHour": "Täglich um {{hour}}:00 Uhr",
"intakeCheck": "Einnahmeprüfung", "intakeCheck": "Einnahmeprüfung",
"15minBefore": "15 Min. vor geplanter Zeit", "minutesBefore": "{{minutes}} Min. vor geplanter Zeit",
"nextCheck": "Nächste Bestandsprüfung", "nextCheck": "Nächste Bestandsprüfung",
"lastSent": "Letzte Benachrichtigung", "lastSent": "Letzte Benachrichtigung",
"lastStockSent": "Letzte Bestands-Erinnerung", "lastStockSent": "Letzte Bestands-Erinnerung",
@@ -299,7 +349,8 @@
"highStockTooltip": "Bestand über diesem Wert bedeutet, dass du gut versorgt bist", "highStockTooltip": "Bestand über diesem Wert bedeutet, dass du gut versorgt bist",
"thresholdValidation": "Werte müssen wie folgt sein: Kritisch < Niedrig < Hoch", "thresholdValidation": "Werte müssen wie folgt sein: Kritisch < Niedrig < Hoch",
"shareStockStatus": "Bestand auf geteilten Links anzeigen", "shareStockStatus": "Bestand auf geteilten Links anzeigen",
"shareStockStatusDesc": "Bestandsstatus (Normal/Niedrig/Kritisch) und farbige Rahmen auf geteilten Zeitplan-Links für Einnahme-Nutzer anzeigen" "shareStockStatusDesc": "Bestandsstatus (Normal/Niedrig/Kritisch) und farbige Rahmen auf geteilten Zeitplan-Links für Einnahme-Nutzer anzeigen",
"packageTypesNote": "Hinweis: Tubenmedikamente sind von Bestands-Erinnerungen ausgeschlossen. Flüssigbehälter-Medikamente verwenden einen einzelnen Reminder-Basiswert (Niedrig und Kritisch werden automatisch von diesem Wert abgeleitet)."
}, },
"timeline": { "timeline": {
"title": "Allgemeine UI", "title": "Allgemeine UI",
@@ -316,7 +367,7 @@
"stockReminder": { "stockReminder": {
"title": "Bestands-Erinnerung", "title": "Bestands-Erinnerung",
"description": "Bestands-Erinnerungen aktivieren", "description": "Bestands-Erinnerungen aktivieren",
"infoTooltip": "Benachrichtigungen umfassen alle Medikamente mit Niedrig- oder Kritisch-Status. Niedrig: Bestand wird knapp. Kritisch: Bestand ist kritisch niedrig — bald nachbestellen.", "infoTooltip": "Benachrichtigungen umfassen alle Medikamente mit Niedrig- oder Kritisch-Status. Hinweis: Tubenmedikamente sind ausgeschlossen; Flüssigbehälter verwenden einen einzelnen Basiswert (Niedrig und Kritisch werden abgeleitet).",
"repeatDaily": "Täglich wiederholen", "repeatDaily": "Täglich wiederholen",
"repeatTooltip": "Wenn aktiviert, wird täglich eine Erinnerung gesendet solange der Bestand kritisch ist. Andernfalls nur einmal pro Medikament bis zum Auffüllen." "repeatTooltip": "Wenn aktiviert, wird täglich eine Erinnerung gesendet solange der Bestand kritisch ist. Andernfalls nur einmal pro Medikament bis zum Auffüllen."
}, },
@@ -327,6 +378,8 @@
"at": "um", "at": "um",
"stockInfo": "Aktueller Bestand", "stockInfo": "Aktueller Bestand",
"packageDetails": "Packungsdetails", "packageDetails": "Packungsdetails",
"packageTypeTubeHint": "Tubenmedikamente enthalten feste Mengen (z. B. Cremes, Gele). Der Bestand wird nicht verfolgt und Erinnerungen werden nicht gesendet.",
"packageTypeLiquidHint": "Flüssigbehälter verwenden ein vereinfachtes Erinnerungsmodell. Niedrig- und Kritisch-Stufen werden automatisch von einem einzelnen Basiswert abgeleitet.",
"currentStock": "Tabletten", "currentStock": "Tabletten",
"packs": "Packungen", "packs": "Packungen",
"blistersPerPack": "Blister/Packung", "blistersPerPack": "Blister/Packung",
@@ -436,6 +489,7 @@
}, },
"validation": { "validation": {
"required": "Dieses Feld ist erforderlich", "required": "Dieses Feld ist erforderlich",
"nameOrGenericRequired": "Handelsname oder Wirkstoff ist erforderlich",
"maxLength": "Maximal {{max}} Zeichen ({{current}}/{{max}})", "maxLength": "Maximal {{max}} Zeichen ({{current}}/{{max}})",
"tooLong": "{{current}}/{{max}} Zeichen" "tooLong": "{{current}}/{{max}} Zeichen"
}, },
@@ -576,9 +630,11 @@
"loosePills": "Lose Tabletten", "loosePills": "Lose Tabletten",
"pillsPerBlister": "(je {{count}} Tabletten)", "pillsPerBlister": "(je {{count}} Tabletten)",
"packageSize": "Packungsgröße: {{count}} Tabletten", "packageSize": "Packungsgröße: {{count}} Tabletten",
"packageSizeAmount": "Packungsgroesse: {{count}} {{unit}}",
"packageSizeBreakdown": "{{packCount}} x {{sizePerPack}} Tabletten Packung = {{total}} Tabletten", "packageSizeBreakdown": "{{packCount}} x {{sizePerPack}} Tabletten Packung = {{total}} Tabletten",
"currentComposition": "Aktueller Bestand: {{fullBlisters}} volle Blister + {{partialPills}} angebrochen + {{loosePills}} lose = {{total}} Tabletten", "currentComposition": "Aktueller Bestand: {{fullBlisters}} volle Blister + {{partialPills}} angebrochen + {{loosePills}} lose = {{total}} Tabletten",
"maxExceeded": "Die maximale Packungsgröße beträgt {{count}} Tabletten. Werte wurden begrenzt.", "maxExceeded": "Die maximale Packungsgröße beträgt {{count}} Tabletten. Werte wurden begrenzt.",
"maxExceededAmount": "Die maximale Packungsgroesse betraegt {{count}} {{unit}}. Werte wurden begrenzt.",
"decreaseValue": "Wert verringern", "decreaseValue": "Wert verringern",
"increaseValue": "Wert erhöhen", "increaseValue": "Wert erhöhen",
"currentTotal": "Aktueller Bestand", "currentTotal": "Aktueller Bestand",
@@ -638,6 +694,7 @@
"docPackageType": "Verpackungsart", "docPackageType": "Verpackungsart",
"docBlister": "Blisterpackung", "docBlister": "Blisterpackung",
"docBottle": "Pillendose", "docBottle": "Pillendose",
"docTube": "Tube",
"docPacks": "Packungen", "docPacks": "Packungen",
"docBlistersPerPack": "Blister pro Packung", "docBlistersPerPack": "Blister pro Packung",
"docPillsPerBlister": "Tabletten pro Blister", "docPillsPerBlister": "Tabletten pro Blister",
+66 -9
View File
@@ -110,6 +110,7 @@
"fullBlisters": "Full blisters", "fullBlisters": "Full blisters",
"openBlister": "Open blister", "openBlister": "Open blister",
"stock": "Stock", "stock": "Stock",
"dailyConsumption": "Daily consumption",
"stockDetails": "Details", "stockDetails": "Details",
"daysLeft": "Days left", "daysLeft": "Days left",
"status": "Status", "status": "Status",
@@ -118,7 +119,8 @@
"expiry": "Expiry", "expiry": "Expiry",
"pillsCount": "{{count}} pills", "pillsCount": "{{count}} pills",
"pillsCount_one": "{{count}} pill", "pillsCount_one": "{{count}} pill",
"pillsCount_other": "{{count}} pills" "pillsCount_other": "{{count}} pills",
"perDayWithUnit": "{{value}} {{unit}}"
}, },
"medications": { "medications": {
"list": { "list": {
@@ -130,7 +132,8 @@
"reactivate": "Reactivate", "reactivate": "Reactivate",
"obsoleteTitle": "Obsolete ({{count}})", "obsoleteTitle": "Obsolete ({{count}})",
"obsoleteSince": "Stopped", "obsoleteSince": "Stopped",
"started": "Started" "started": "Started",
"emptyState": "No medications yet. Add your first medication to get started."
}, },
"details": { "details": {
"packs": "Packs", "packs": "Packs",
@@ -139,6 +142,7 @@
"loose": "Loose", "loose": "Loose",
"total": "Total", "total": "Total",
"stock": "Stock", "stock": "Stock",
"capacityPerPackage": "Capacity per package",
"totalCapacity": "Capacity", "totalCapacity": "Capacity",
"type": "Type" "type": "Type"
}, },
@@ -167,18 +171,41 @@
"commercialName": "Commercial Name", "commercialName": "Commercial Name",
"genericName": "Generic Name", "genericName": "Generic Name",
"takenBy": "Taken by", "takenBy": "Taken by",
"medicationForm": "Medication Form",
"medicationFormCapsule": "Capsule",
"medicationFormTablet": "Tablet",
"medicationFormLiquid": "Liquid",
"medicationFormTopical": "Topical",
"pillForm": "Pill Form",
"lifecycleCategory": "Lifecycle",
"lifecycleRefillWhenEmpty": "Refill when empty",
"lifecycleTreatmentPeriod": "Treatment period",
"packageType": "Package Type", "packageType": "Package Type",
"packageTypeBlister": "Blister Pack", "packageTypeBlister": "Blister Pack",
"packageTypeBottle": "Pill Bottle", "packageTypeBottle": "Pill Bottle",
"packageTypeTube": "Tube",
"packageTypeLiquidContainer": "Liquid Container",
"packs": "Packs", "packs": "Packs",
"bottles": "Bottles",
"tubes": "Tubes",
"blistersPerPack": "Blisters per pack", "blistersPerPack": "Blisters per pack",
"pillsPerBlister": "Pills per blister", "pillsPerBlister": "Pills per blister",
"totalCapacity": "Total Capacity", "totalCapacity": "Total Capacity",
"currentPills": "Current Pills", "currentPills": "Current Pills",
"totalAmount": "Total Amount",
"currentAmount": "Current Amount",
"totalAmountLabel": "Total (amount)",
"packageAmount": "Package amount",
"packageAmountPerBottle": "Amount per bottle",
"packageAmountPerTube": "Amount per tube",
"packageAmountUnitMl": "ml",
"packageAmountUnitG": "g",
"loosePills": "Loose pills", "loosePills": "Loose pills",
"pillWeight": "Dose per pill", "pillWeight": "Dose per pill",
"total": "Total (pills)", "total": "Total (pills)",
"medicationStartDate": "Medication Start Date", "medicationStartDate": "Medication Start Date",
"medicationEndDate": "Medication End Date",
"autoMarkObsoleteAfterEndDate": "Automatically mark obsolete after end date",
"expiryDate": "Expiry Date", "expiryDate": "Expiry Date",
"notes": "Notes", "notes": "Notes",
"medicationImage": "Medication Image", "medicationImage": "Medication Image",
@@ -192,21 +219,44 @@
}, },
"placeholders": { "placeholders": {
"commercial": "e.g. Ozempic", "commercial": "e.g. Ozempic",
"generic": "e.g. Semaglutide (optional)", "generic": "e.g. Semaglutide",
"takenBy": "Type name and press Enter", "takenBy": "Type name and press Enter",
"addPerson": "Add another person...", "addPerson": "Add another person...",
"weight": "e.g. 240", "weight": "e.g. 240",
"notes": "e.g. Take with food, avoid alcohol... (optional)" "notes": "e.g. Take with food, avoid alcohol... (optional)"
}, },
"validation": { "validation": {
"startDateAfterIntake": "Medication start date ({{medicationStartDate}}) cannot be after intake date ({{intakeDate}})." "startDateAfterIntake": "Medication start date ({{medicationStartDate}}) cannot be after intake date ({{intakeDate}}).",
"endDateBeforeStart": "Medication end date ({{medicationEndDate}}) cannot be before medication start date ({{medicationStartDate}})."
}, },
"blisters": { "blisters": {
"title": "Intake schedule", "title": "Intake schedule",
"remind": "Remind", "remind": "Remind",
"remindTooltip": "Receive a notification 15 minutes before each scheduled intake", "remindTooltip": "Receive a notification 15 minutes before each scheduled intake",
"addIntake": "Intake", "addIntake": "Intake",
"usage": "Usage (pills)", "usage": "Usage",
"usageTablets": "Usage (tablets)",
"usageCapsules": "Usage (capsules)",
"usageMl": "Usage (ml)",
"usageTsp": "Usage (tsp)",
"usageTbsp": "Usage (tbsp)",
"usageApplication": "Usage (applications)",
"intakeUnit": "Intake unit",
"intakeUnitMl": "Milliliters (ml)",
"intakeUnitTsp": "Teaspoon (5 ml)",
"intakeUnitTbsp": "Tablespoon (15 ml)",
"intakes": "intakes",
"intakes_one": "intake",
"intakes_other": "intakes",
"teaspoons": "teaspoons",
"teaspoons_one": "teaspoon",
"teaspoons_other": "teaspoons",
"tablespoons": "tablespoons",
"tablespoons_one": "tablespoon",
"tablespoons_other": "tablespoons",
"applications": "applications",
"applications_one": "application",
"applications_other": "applications",
"everyDays": "Every (days)", "everyDays": "Every (days)",
"every": "every", "every": "every",
"from": "from", "from": "from",
@@ -273,9 +323,9 @@
"schedule": { "schedule": {
"title": "Reminder Schedule", "title": "Reminder Schedule",
"stockCheck": "Stock & prescription check", "stockCheck": "Stock & prescription check",
"dailyAt6": "Daily at 6:00 AM", "dailyAtHour": "Daily at {{hour}}:00",
"intakeCheck": "Intake check", "intakeCheck": "Intake check",
"15minBefore": "15 min before scheduled time", "minutesBefore": "{{minutes}} min before scheduled time",
"nextCheck": "Next stock check", "nextCheck": "Next stock check",
"lastSent": "Last notification sent", "lastSent": "Last notification sent",
"lastStockSent": "Last stock reminder", "lastStockSent": "Last stock reminder",
@@ -299,7 +349,8 @@
"highStockTooltip": "Stock above this value means you are well supplied", "highStockTooltip": "Stock above this value means you are well supplied",
"thresholdValidation": "Values must be: Critical < Low < High", "thresholdValidation": "Values must be: Critical < Low < High",
"shareStockStatus": "Show Stock on Shared Links", "shareStockStatus": "Show Stock on Shared Links",
"shareStockStatusDesc": "Show stock status (Normal/Low/Critical) and colored borders on shared schedule links for intake users" "shareStockStatusDesc": "Show stock status (Normal/Low/Critical) and colored borders on shared schedule links for intake users",
"packageTypesNote": "Note: Tube medications are excluded from stock reminders. Liquid container medications use a single reminder baseline (Low and Critical are automatically derived from this value)."
}, },
"timeline": { "timeline": {
"title": "General UI", "title": "General UI",
@@ -316,7 +367,7 @@
"stockReminder": { "stockReminder": {
"title": "Stock Reminder", "title": "Stock Reminder",
"description": "Enable stock reminders", "description": "Enable stock reminders",
"infoTooltip": "Notifications include all medications with Low or Critical stock status. Low: stock is running low. Critical: stock is critically low — reorder soon.", "infoTooltip": "Notifications include all medications with Low or Critical stock status. Note: Tube medications are excluded; Liquid containers use a single baseline threshold (Low and Critical are derived).",
"repeatDaily": "Repeat daily", "repeatDaily": "Repeat daily",
"repeatTooltip": "When enabled, sends reminders every day while stock is critical. Otherwise, only notifies once per medication until restocked." "repeatTooltip": "When enabled, sends reminders every day while stock is critical. Otherwise, only notifies once per medication until restocked."
}, },
@@ -327,6 +378,8 @@
"at": "at", "at": "at",
"stockInfo": "Current Stock", "stockInfo": "Current Stock",
"packageDetails": "Package Details", "packageDetails": "Package Details",
"packageTypeTubeHint": "Tube medications contain fixed amounts (e.g., creams, gels). Stock is not tracked and reminders are not sent.",
"packageTypeLiquidHint": "Liquid containers use a simplified reminder model. Low and Critical levels are automatically derived from a single baseline threshold for simplicity.",
"currentStock": "Pills", "currentStock": "Pills",
"packs": "Packs", "packs": "Packs",
"blistersPerPack": "Blisters/Pack", "blistersPerPack": "Blisters/Pack",
@@ -436,6 +489,7 @@
}, },
"validation": { "validation": {
"required": "This field is required", "required": "This field is required",
"nameOrGenericRequired": "Either commercial name or generic name is required",
"maxLength": "Maximum {{max}} characters ({{current}}/{{max}})", "maxLength": "Maximum {{max}} characters ({{current}}/{{max}})",
"tooLong": "{{current}}/{{max}} characters" "tooLong": "{{current}}/{{max}} characters"
}, },
@@ -576,9 +630,11 @@
"loosePills": "Loose pills", "loosePills": "Loose pills",
"pillsPerBlister": "({{count}} pills each)", "pillsPerBlister": "({{count}} pills each)",
"packageSize": "Package size: {{count}} pills", "packageSize": "Package size: {{count}} pills",
"packageSizeAmount": "Package size: {{count}} {{unit}}",
"packageSizeBreakdown": "{{packCount}} x {{sizePerPack}} pills Pack = {{total}} pills", "packageSizeBreakdown": "{{packCount}} x {{sizePerPack}} pills Pack = {{total}} pills",
"currentComposition": "Current stock: {{fullBlisters}} full blisters + {{partialPills}} partial + {{loosePills}} loose = {{total}} pills", "currentComposition": "Current stock: {{fullBlisters}} full blisters + {{partialPills}} partial + {{loosePills}} loose = {{total}} pills",
"maxExceeded": "Maximum package size is {{count}} pills. Values were capped.", "maxExceeded": "Maximum package size is {{count}} pills. Values were capped.",
"maxExceededAmount": "Maximum package size is {{count}} {{unit}}. Values were capped.",
"decreaseValue": "Decrease value", "decreaseValue": "Decrease value",
"increaseValue": "Increase value", "increaseValue": "Increase value",
"currentTotal": "Current total", "currentTotal": "Current total",
@@ -638,6 +694,7 @@
"docPackageType": "Package Type", "docPackageType": "Package Type",
"docBlister": "Blister Pack", "docBlister": "Blister Pack",
"docBottle": "Pill Bottle", "docBottle": "Pill Bottle",
"docTube": "Tube",
"docPacks": "Packs", "docPacks": "Packs",
"docBlistersPerPack": "Blisters per pack", "docBlistersPerPack": "Blisters per pack",
"docPillsPerBlister": "Pills per blister", "docPillsPerBlister": "Pills per blister",
+255 -45
View File
@@ -6,6 +6,14 @@ import { ConfirmModal, MedicationAvatar } from "../components";
import { useAuth } from "../components/Auth"; import { useAuth } from "../components/Auth";
import { useAppContext } from "../context"; import { useAppContext } from "../context";
import { useModalHistory } from "../hooks"; import { useModalHistory } from "../hooks";
import {
allowsPillFormSelection,
type Coverage,
getMedDisplayName,
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
} from "../types";
import { formatNumber, getExpiryClass, getSystemLocale } from "../utils/formatters"; import { formatNumber, getExpiryClass, getSystemLocale } from "../utils/formatters";
import { expandDoseIds, getStockStatus, isDoseDismissed } from "../utils/schedule"; import { expandDoseIds, getStockStatus, isDoseDismissed } from "../utils/schedule";
import { import {
@@ -86,6 +94,7 @@ export function DashboardPage() {
settings.lowStockDays, settings.lowStockDays,
coverage.low, coverage.low,
coverage.all, coverage.all,
meds,
settings.lastAutoEmailSent, settings.lastAutoEmailSent,
settings.lastNotificationType, settings.lastNotificationType,
settings.lastNotificationChannel, settings.lastNotificationChannel,
@@ -118,7 +127,7 @@ export function DashboardPage() {
}) })
.map((med) => ({ .map((med) => ({
id: med.id, id: med.id,
name: med.name, name: getMedDisplayName(med),
remainingRefills: med.prescriptionRemainingRefills ?? 0, remainingRefills: med.prescriptionRemainingRefills ?? 0,
threshold: med.prescriptionLowRefillThreshold ?? 1, threshold: med.prescriptionLowRefillThreshold ?? 1,
})) }))
@@ -128,6 +137,158 @@ export function DashboardPage() {
const showOnlyToday = settings.upcomingTodayOnly; const showOnlyToday = settings.upcomingTodayOnly;
const prescriptionEmptyCount = prescriptionLowMeds.filter((med) => med.remainingRefills <= 0).length; const prescriptionEmptyCount = prescriptionLowMeds.filter((med) => med.remainingRefills <= 0).length;
const getTubeUnitLabel = (med: (typeof meds)[number] | undefined, value: number) =>
isLiquidContainerPackageType(med?.packageType) || med?.medicationForm === "liquid"
? t("form.packageAmountUnitMl")
: t("form.blisters.applications", { count: Math.abs(value) });
const formatStockLabel = (med: (typeof meds)[number] | undefined, medsLeft: number) => {
if (isLiquidContainerPackageType(med?.packageType)) {
return `${formatNumber(medsLeft)} ${t("form.packageAmountUnitMl")}`;
}
if (isTubePackageType(med?.packageType)) {
return `${formatNumber(medsLeft)} ${getTubeUnitLabel(med, medsLeft)}`;
}
return t("table.pillsCount", { count: Math.round(medsLeft) });
};
const convertLiquidUsageToMl = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): number => {
if (unit === "tsp") return usage * 5;
if (unit === "tbsp") return usage * 15;
return usage;
};
const getLiquidCountUnitLabel = (unit: "ml" | "tsp" | "tbsp" | null | undefined, usage: number): string => {
if (unit === "tsp") return t("form.blisters.teaspoons", { count: Math.abs(usage) });
if (unit === "tbsp") return t("form.blisters.tablespoons", { count: Math.abs(usage) });
return t("form.packageAmountUnitMl");
};
const formatLiquidUsageLabel = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): string => {
const normalizedUsage = Number(usage);
if (!Number.isFinite(normalizedUsage) || normalizedUsage <= 0) {
return `0 ${t("form.packageAmountUnitMl")}`;
}
if (unit === "ml" || unit == null) {
return `${formatNumber(normalizedUsage)} ${t("form.packageAmountUnitMl")}`;
}
const mlTotal = convertLiquidUsageToMl(normalizedUsage, unit);
return `${formatNumber(normalizedUsage)} ${getLiquidCountUnitLabel(unit, normalizedUsage)} ${formatNumber(mlTotal)} ${t("form.packageAmountUnitMl")}`;
};
const formatDoseUsageLabel = (
med: (typeof meds)[number] | undefined,
usage: number,
intakeUnit?: "ml" | "tsp" | "tbsp" | null
) => {
if (isLiquidContainerPackageType(med?.packageType)) {
return formatLiquidUsageLabel(usage, intakeUnit);
}
if (isTubePackageType(med?.packageType)) {
return `${usage} ${getTubeUnitLabel(med, usage)}`;
}
return `${usage} ${usage !== 1 ? t("common.pills") : t("common.pill")}`;
};
const formatTotalUsageLabel = (
med: (typeof meds)[number] | undefined,
total: number,
intakeUnit?: "ml" | "tsp" | "tbsp" | null,
doses?: Array<{ usage: number; intakeUnit?: "ml" | "tsp" | "tbsp" | null }>
) => {
if (isLiquidContainerPackageType(med?.packageType)) {
if (doses && doses.length > 0) {
const normalizedDoses = doses.filter((dose) => Number.isFinite(Number(dose.usage)) && Number(dose.usage) > 0);
if (normalizedDoses.length > 0) {
const allUnits = new Set(normalizedDoses.map((dose) => dose.intakeUnit ?? "ml"));
const totalMl = normalizedDoses.reduce(
(sum, dose) => sum + convertLiquidUsageToMl(Number(dose.usage), dose.intakeUnit ?? "ml"),
0
);
if (allUnits.size === 1) {
const onlyUnit = normalizedDoses[0]?.intakeUnit ?? "ml";
const totalUsageInUnit = normalizedDoses.reduce((sum, dose) => sum + Number(dose.usage), 0);
return formatLiquidUsageLabel(totalUsageInUnit, onlyUnit);
}
return `${formatNumber(totalMl)} ${t("form.packageAmountUnitMl")}`;
}
}
return formatLiquidUsageLabel(total, intakeUnit);
}
if (isTubePackageType(med?.packageType)) {
return `${total} ${getTubeUnitLabel(med, total)}`;
}
return t("common.pillsTotal", { count: total });
};
const formatDailyConsumption = (med: (typeof meds)[number] | undefined) => {
if (!med) return "-";
const intakes =
med.intakes && med.intakes.length > 0
? med.intakes
: med.blisters.map((blister) => ({
usage: blister.usage,
every: blister.every,
intakeUnit: null as "ml" | "tsp" | "tbsp" | null,
takenBy: null as string | null,
}));
if (intakes.length === 0) return "-";
let dailyTotal = 0;
for (const intake of intakes) {
const usage = Number(intake.usage);
const every = Math.max(1, Number(intake.every) || 1);
if (!Number.isFinite(usage) || usage <= 0) continue;
const hasPerIntakeTakenBy = typeof intake.takenBy === "string" && intake.takenBy.trim().length > 0;
const personMultiplier = hasPerIntakeTakenBy ? 1 : Math.max(1, med.takenBy?.length ?? 0);
const normalizedUsage = (usage * personMultiplier) / every;
if (isLiquidContainerPackageType(med.packageType)) {
dailyTotal += convertLiquidUsageToMl(normalizedUsage, intake.intakeUnit ?? "ml");
} else {
dailyTotal += normalizedUsage;
}
}
if (dailyTotal <= 0) return "-";
if (isLiquidContainerPackageType(med.packageType)) {
return t("table.perDayWithUnit", { value: formatNumber(dailyTotal), unit: t("form.packageAmountUnitMl") });
}
if (isTubePackageType(med.packageType)) {
const tubeUnit =
med.medicationForm === "liquid"
? t("form.packageAmountUnitMl")
: t("form.blisters.applications", { count: Math.abs(dailyTotal) });
return t("table.perDayWithUnit", { value: formatNumber(dailyTotal), unit: tubeUnit });
}
const pillUnit = dailyTotal === 1 ? t("common.pill") : t("common.pills");
return t("table.perDayWithUnit", { value: formatNumber(dailyTotal), unit: pillUnit });
};
const shouldHideNoScheduleStatusForTube = (
med: (typeof meds)[number] | undefined,
status: { className: string; label: string } | null
) => isTubePackageType(med?.packageType) && status?.label === "status.noSchedule";
const getVisibleStockStatus = (
med: (typeof meds)[number] | undefined,
status: { className: string; label: string } | null
) => (shouldHideNoScheduleStatusForTube(med, status) ? null : status);
const getMedByName = (name: string) => meds.find((m) => getMedDisplayName(m) === name);
const prescriptionStatus = const prescriptionStatus =
prescriptionRemindersEnabled && prescriptionLowMeds.length > 0 prescriptionRemindersEnabled && prescriptionLowMeds.length > 0
? { ? {
@@ -250,9 +411,11 @@ export function DashboardPage() {
<span className="reminder-status-label">{t("dashboard.reminders.needsRefill")}:</span> <span className="reminder-status-label">{t("dashboard.reminders.needsRefill")}:</span>
<span className="reminder-status-value"> <span className="reminder-status-value">
{reminderData.lowStockMeds.map((med, idx) => { {reminderData.lowStockMeds.map((med, idx) => {
const medication = meds.find((m) => m.name === med.name); const medication = meds.find((m) => getMedDisplayName(m) === med.name);
const cov = coverage.all.find((c) => c.name === med.name); const cov = coverage.all.find((c) => c.name === med.name);
const status = cov ? getStockStatus(cov.daysLeft, cov.medsLeft, stockThresholds) : null; const status = cov
? getStockStatus(cov.daysLeft, cov.medsLeft, stockThresholds, medication?.packageType)
: null;
const textClass = const textClass =
status?.className === "danger" status?.className === "danger"
? "danger-text" ? "danger-text"
@@ -322,7 +485,7 @@ export function DashboardPage() {
(() => { (() => {
const names = reminderData.lastStockSent!.medNames!.split(", "); const names = reminderData.lastStockSent!.medNames!.split(", ");
return names.map((name, idx) => { return names.map((name, idx) => {
const medication = meds.find((m) => m.name === name); const medication = meds.find((m) => getMedDisplayName(m) === name);
return ( return (
<span key={name}> <span key={name}>
{idx > 0 && ", "} {idx > 0 && ", "}
@@ -353,7 +516,9 @@ export function DashboardPage() {
<span className="reminder-status-value"> <span className="reminder-status-value">
{reminderData.lastIntakeSent.medName && {reminderData.lastIntakeSent.medName &&
(() => { (() => {
const medication = meds.find((m) => m.name === reminderData.lastIntakeSent!.medName); const medication = meds.find(
(m) => getMedDisplayName(m) === reminderData.lastIntakeSent!.medName
);
return medication ? ( return medication ? (
<span <span
className="med-link clickable" className="med-link clickable"
@@ -408,7 +573,9 @@ export function DashboardPage() {
const lowStockMap = new Map<string, Coverage>(); const lowStockMap = new Map<string, Coverage>();
for (const c of coverage.all) { for (const c of coverage.all) {
if (c.daysLeft === null && c.medsLeft > 0) continue; // no schedule, has stock if (c.daysLeft === null && c.medsLeft > 0) continue; // no schedule, has stock
if (c.medsLeft <= 0 || c.daysLeft === null || c.daysLeft < settings.lowStockDays) { const med = getMedByName(c.name);
const status = getStockStatus(c.daysLeft, c.medsLeft, stockThresholds, med?.packageType);
if (status.className === "danger" || status.className === "warning") {
const existing = lowStockMap.get(c.name); const existing = lowStockMap.get(c.name);
if (!existing || (c.daysLeft ?? 0) < (existing.daysLeft ?? 0)) { if (!existing || (c.daysLeft ?? 0) < (existing.daysLeft ?? 0)) {
lowStockMap.set(c.name, c); lowStockMap.set(c.name, c);
@@ -428,8 +595,8 @@ export function DashboardPage() {
<p> <p>
{t("dashboard.reorder.lowWarningPrefix")}{" "} {t("dashboard.reorder.lowWarningPrefix")}{" "}
{lowStockMeds.map((c, idx) => { {lowStockMeds.map((c, idx) => {
const med = meds.find((m) => m.name === c.name); const med = meds.find((m) => getMedDisplayName(m) === c.name);
const status = getStockStatus(c.daysLeft, c.medsLeft, stockThresholds); const status = getStockStatus(c.daysLeft, c.medsLeft, stockThresholds, med?.packageType);
const textClass = const textClass =
status.className === "danger" status.className === "danger"
? "danger-text" ? "danger-text"
@@ -473,10 +640,11 @@ export function DashboardPage() {
<div className="card-head"> <div className="card-head">
<h2>{t("dashboard.overview.title")}</h2> <h2>{t("dashboard.overview.title")}</h2>
</div> </div>
<div className="table table-7"> <div className="table table-8">
<div className="table-head"> <div className="table-head">
<span>{t("table.name")}</span> <span>{t("table.name")}</span>
<span>{t("table.stock")}</span> <span>{t("table.stock")}</span>
<span>{t("table.dailyConsumption")}</span>
<span>{t("table.stockDetails")}</span> <span>{t("table.stockDetails")}</span>
<span>{t("table.daysLeft")}</span> <span>{t("table.daysLeft")}</span>
<span>{t("table.runsOut")}</span> <span>{t("table.runsOut")}</span>
@@ -484,13 +652,14 @@ export function DashboardPage() {
<span>{t("table.status")}</span> <span>{t("table.status")}</span>
</div> </div>
{coverage.all.map((row) => { {coverage.all.map((row) => {
const status = getStockStatus(row.daysLeft, row.medsLeft, stockThresholds); const med = meds.find((m) => getMedDisplayName(m) === row.name);
const med = meds.find((m) => m.name === row.name); const rawStatus = getStockStatus(row.daysLeft, row.medsLeft, stockThresholds, med?.packageType);
const status = getVisibleStockStatus(med, rawStatus);
const expiryClass = getExpiryClass(med?.expiryDate, settings.expiryWarningDays); const expiryClass = getExpiryClass(med?.expiryDate, settings.expiryWarningDays);
const textClass = const textClass =
status.className === "danger" rawStatus.className === "danger"
? "danger-text" ? "danger-text"
: status.className === "warning" : rawStatus.className === "warning"
? "warning-text" ? "warning-text"
: "success-text"; : "success-text";
const stock = getBlisterStock( const stock = getBlisterStock(
@@ -584,15 +753,18 @@ export function DashboardPage() {
</span> </span>
</span> </span>
<span data-label={t("table.stock")} className={textClass}> <span data-label={t("table.stock")} className={textClass}>
{med?.packageType === "bottle" {isAmountBasedPackageType(med?.packageType)
? t("table.pillsCount", { count: Math.round(row.medsLeft) }) ? formatStockLabel(med, row.medsLeft)
: formatFullBlisters(stock.fullBlisters, t)} : formatFullBlisters(stock.fullBlisters, t)}
</span> </span>
<span data-label={t("table.dailyConsumption")} className={textClass}>
{formatDailyConsumption(med)}
</span>
<span <span
data-label={t("table.stockDetails")} data-label={t("table.stockDetails")}
className={`${textClass}${med?.packageType === "bottle" ? " hide-on-card" : ""}`} className={`${textClass}${isAmountBasedPackageType(med?.packageType) ? " hide-on-card" : ""}`}
> >
{med?.packageType === "bottle" {isAmountBasedPackageType(med?.packageType)
? "—" ? "—"
: formatOpenBlisterAndLoose( : formatOpenBlisterAndLoose(
stock.openBlisterPills, stock.openBlisterPills,
@@ -614,8 +786,8 @@ export function DashboardPage() {
}) })
: "-"} : "-"}
</span> </span>
<span data-label={t("table.status")} className={`status-chip ${status.className}`}> <span data-label={t("table.status")} className={status ? `status-chip ${status.className}` : ""}>
{t(status.label)} {status ? t(status.label) : "-"}
</span> </span>
</div> </div>
); );
@@ -673,7 +845,7 @@ export function DashboardPage() {
// Count missed doses that are NOT dismissed (for warning icon) // Count missed doses that are NOT dismissed (for warning icon)
const missedNotDismissedCount = day.meds.reduce((count, item) => { const missedNotDismissedCount = day.meds.reduce((count, item) => {
const med = meds.find((m) => m.name === item.medName); const med = meds.find((m) => getMedDisplayName(m) === item.medName);
const dismissedUntilDate = med?.dismissedUntil ?? undefined; const dismissedUntilDate = med?.dismissedUntil ?? undefined;
return ( return (
count + count +
@@ -729,12 +901,13 @@ export function DashboardPage() {
</div> </div>
{!isCollapsed && {!isCollapsed &&
day.meds.map((item) => { day.meds.map((item) => {
const med = meds.find((m) => m.name === item.medName); const med = meds.find((m) => getMedDisplayName(m) === item.medName);
const medCov = coverageByMed[item.medName]; const medCov = coverageByMed[item.medName];
const isEmpty = medCov ? medCov.medsLeft <= 0 : false; const isEmpty = medCov ? medCov.medsLeft <= 0 : false;
const status = medCov const rawStatus = medCov
? getStockStatus(medCov.daysLeft, medCov.medsLeft, stockThresholds) ? getStockStatus(medCov.daysLeft, medCov.medsLeft, stockThresholds, med?.packageType)
: null; : null;
const status = getVisibleStockStatus(med, rawStatus);
const itemDoseIds = expandDoseIds(item.doses); const itemDoseIds = expandDoseIds(item.doses);
const allTaken = itemDoseIds.every((id) => takenDoses.has(id)); const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
return ( return (
@@ -769,7 +942,9 @@ export function DashboardPage() {
</div> </div>
</div> </div>
<div className="tag-row"> <div className="tag-row">
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span> <span className="tag subtle">
{formatTotalUsageLabel(med, item.total, item.doses[0]?.intakeUnit, item.doses)}
</span>
{status && ( {status && (
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span> <span className={`status-chip small ${status.className}`}>{t(status.label)}</span>
)} )}
@@ -784,9 +959,9 @@ export function DashboardPage() {
<span className="dose-time">{dose.timeStr}</span> <span className="dose-time">{dose.timeStr}</span>
<span className="dose-usage"> <span className="dose-usage">
<span className="dose-usage-main"> <span className="dose-usage-main">
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")} {formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
</span> </span>
{med?.pillWeightMg && ( {allowsPillFormSelection(med?.packageType) && med?.pillWeightMg && (
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span> <span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
)} )}
</span> </span>
@@ -831,7 +1006,8 @@ export function DashboardPage() {
🤖 🤖
</span> </span>
)} )}
<span className="dose-btn-label">{t("common.undo")}</span>
<span aria-hidden="true"></span>
</button> </button>
) : ( ) : (
<button <button
@@ -943,7 +1119,13 @@ export function DashboardPage() {
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime; const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
if (willBeOutOfStock) return "danger"; if (willBeOutOfStock) return "danger";
if (!medCoverage) return "success"; if (!medCoverage) return "success";
const status = getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds); const med = getMedByName(item.medName);
const status = getStockStatus(
medCoverage.daysLeft,
medCoverage.medsLeft,
stockThresholds,
med?.packageType
);
return status.className; return status.className;
}); });
const worstStatus = dayStockStatuses.includes("danger") const worstStatus = dayStockStatuses.includes("danger")
@@ -986,15 +1168,21 @@ export function DashboardPage() {
{!isCollapsed && {!isCollapsed &&
day.meds.map((item) => { day.meds.map((item) => {
const medCoverage = coverageByMed[item.medName]; const medCoverage = coverageByMed[item.medName];
const med = meds.find((m) => m.name === item.medName); const med = meds.find((m) => getMedDisplayName(m) === item.medName);
const depletionTime = depletionByMed[item.medName]; const depletionTime = depletionByMed[item.medName];
const isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false; const isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime; const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
const status = willBeOutOfStock const status = willBeOutOfStock
? { className: "danger", label: "status.outOfStock" } ? { className: "danger", label: "status.outOfStock" }
: medCoverage : medCoverage
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds) ? getStockStatus(
medCoverage.daysLeft,
medCoverage.medsLeft,
stockThresholds,
med?.packageType
)
: null; : null;
const visibleStatus = getVisibleStockStatus(med, status);
const itemDoseIds = expandDoseIds(item.doses); const itemDoseIds = expandDoseIds(item.doses);
const allTaken = itemDoseIds.every((id) => takenDoses.has(id)); const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
return ( return (
@@ -1029,9 +1217,13 @@ export function DashboardPage() {
</div> </div>
</div> </div>
<div className="tag-row"> <div className="tag-row">
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span> <span className="tag subtle">
{status && ( {formatTotalUsageLabel(med, item.total, item.doses[0]?.intakeUnit, item.doses)}
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span> </span>
{visibleStatus && (
<span className={`status-chip small ${visibleStatus.className}`}>
{t(visibleStatus.label)}
</span>
)} )}
</div> </div>
</div> </div>
@@ -1048,9 +1240,9 @@ export function DashboardPage() {
<span className="dose-time">{dose.timeStr}</span> <span className="dose-time">{dose.timeStr}</span>
<span className="dose-usage"> <span className="dose-usage">
<span className="dose-usage-main"> <span className="dose-usage-main">
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")} {formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
</span> </span>
{med?.pillWeightMg && ( {allowsPillFormSelection(med?.packageType) && med?.pillWeightMg && (
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span> <span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
)} )}
</span> </span>
@@ -1095,7 +1287,8 @@ export function DashboardPage() {
🤖 🤖
</span> </span>
)} )}
<span className="dose-btn-label">{t("common.undo")}</span>
<span aria-hidden="true"></span>
</button> </button>
) : ( ) : (
<button <button
@@ -1175,7 +1368,13 @@ export function DashboardPage() {
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime; const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
if (willBeOutOfStock) return "danger"; if (willBeOutOfStock) return "danger";
if (!medCoverage) return "success"; if (!medCoverage) return "success";
const status = getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds); const med = getMedByName(item.medName);
const status = getStockStatus(
medCoverage.daysLeft,
medCoverage.medsLeft,
stockThresholds,
med?.packageType
);
return status.className; return status.className;
}); });
const worstStatus = dayStockStatuses.includes("danger") const worstStatus = dayStockStatuses.includes("danger")
@@ -1217,15 +1416,21 @@ export function DashboardPage() {
{!isCollapsed && {!isCollapsed &&
day.meds.map((item) => { day.meds.map((item) => {
const medCoverage = coverageByMed[item.medName]; const medCoverage = coverageByMed[item.medName];
const med = meds.find((m) => m.name === item.medName); const med = meds.find((m) => getMedDisplayName(m) === item.medName);
const depletionTime = depletionByMed[item.medName]; const depletionTime = depletionByMed[item.medName];
const _isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false; const _isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime; const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
const status = willBeOutOfStock const status = willBeOutOfStock
? { className: "danger", label: "status.outOfStock" } ? { className: "danger", label: "status.outOfStock" }
: medCoverage : medCoverage
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds) ? getStockStatus(
medCoverage.daysLeft,
medCoverage.medsLeft,
stockThresholds,
med?.packageType
)
: null; : null;
const visibleStatus = getVisibleStockStatus(med, status);
const itemDoseIds = expandDoseIds(item.doses); const itemDoseIds = expandDoseIds(item.doses);
const allTaken = itemDoseIds.every((id) => takenDoses.has(id)); const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
return ( return (
@@ -1260,9 +1465,13 @@ export function DashboardPage() {
</div> </div>
</div> </div>
<div className="tag-row"> <div className="tag-row">
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span> <span className="tag subtle">
{status && ( {formatTotalUsageLabel(med, item.total, item.doses[0]?.intakeUnit, item.doses)}
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span> </span>
{visibleStatus && (
<span className={`status-chip small ${visibleStatus.className}`}>
{t(visibleStatus.label)}
</span>
)} )}
</div> </div>
</div> </div>
@@ -1275,9 +1484,9 @@ export function DashboardPage() {
<span className="dose-time">{dose.timeStr}</span> <span className="dose-time">{dose.timeStr}</span>
<span className="dose-usage"> <span className="dose-usage">
<span className="dose-usage-main"> <span className="dose-usage-main">
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")} {formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
</span> </span>
{med?.pillWeightMg && ( {allowsPillFormSelection(med?.packageType) && med?.pillWeightMg && (
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span> <span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
)} )}
</span> </span>
@@ -1322,7 +1531,8 @@ export function DashboardPage() {
🤖 🤖
</span> </span>
)} )}
<span className="dose-btn-label">{t("common.undo")}</span>
<span aria-hidden="true"></span>
</button> </button>
) : ( ) : (
<button <button
+397 -128
View File
@@ -17,8 +17,20 @@ import {
import { useAuth } from "../components/Auth"; import { useAuth } from "../components/Auth";
import { useAppContext, useUnsavedChanges } from "../context"; import { useAppContext, useUnsavedChanges } from "../context";
import { useMedicationForm, useModalHistory, useUnsavedChangesWarning } from "../hooks"; import { useMedicationForm, useModalHistory, useUnsavedChangesWarning } from "../hooks";
import type { DoseUnit, Medication } from "../types"; import type { DoseUnit, FormState, Medication, PackageType } from "../types";
import { DOSE_UNITS, FIELD_LIMITS, getPackageSize } from "../types"; import {
allowsPillFormSelection,
DOSE_UNITS,
FIELD_LIMITS,
getMedDisplayName,
getPackageProfile,
getPackageSize,
isAmountBasedPackageType,
isLiquidContainerPackageType,
isTubePackageType,
normalizePackageType,
PACKAGE_PROFILES,
} from "../types";
import { combineDateAndTime, formatDate, formatDateTime, formatNumber } from "../utils/formatters"; import { combineDateAndTime, formatDate, formatDateTime, formatNumber } from "../utils/formatters";
import { MAX_IMAGE_UPLOAD_BYTES, resolveImageUploadError } from "../utils/image-upload"; import { MAX_IMAGE_UPLOAD_BYTES, resolveImageUploadError } from "../utils/image-upload";
import { log } from "../utils/logger"; import { log } from "../utils/logger";
@@ -239,7 +251,7 @@ export function MedicationsPage() {
// Calculate total tablets // Calculate total tablets
const totalTablets = useMemo(() => { const totalTablets = useMemo(() => {
if (form.packageType === "bottle") { if (isAmountBasedPackageType(form.packageType)) {
// For bottle type, looseTablets is the current stock // For bottle type, looseTablets is the current stock
return Number(form.looseTablets) || 0; return Number(form.looseTablets) || 0;
} }
@@ -254,6 +266,14 @@ export function MedicationsPage() {
const dateConsistencyError = useMemo(() => { const dateConsistencyError = useMemo(() => {
const medicationStartDate = form.medicationStartDate; const medicationStartDate = form.medicationStartDate;
const medicationEndDate = form.medicationEndDate;
if (medicationStartDate && medicationEndDate && medicationEndDate < medicationStartDate) {
return t("form.validation.endDateBeforeStart", {
medicationStartDate,
medicationEndDate,
});
}
if (!medicationStartDate) return null; if (!medicationStartDate) return null;
const conflictingIntake = form.intakes.find((intake) => intake.startDate && intake.startDate < medicationStartDate); const conflictingIntake = form.intakes.find((intake) => intake.startDate && intake.startDate < medicationStartDate);
@@ -263,7 +283,62 @@ export function MedicationsPage() {
medicationStartDate, medicationStartDate,
intakeDate: conflictingIntake.startDate, intakeDate: conflictingIntake.startDate,
}); });
}, [form.medicationStartDate, form.intakes, t]); }, [form.medicationStartDate, form.medicationEndDate, form.intakes, t]);
const allowFractionalIntake = useMemo(() => {
if (isLiquidContainerPackageType(form.packageType)) return true;
if (isTubePackageType(form.packageType)) return form.medicationForm === "liquid";
return form.pillForm === "tablet";
}, [form.packageType, form.medicationForm, form.pillForm]);
const getUsageLabel = useCallback(
(intakeUnit: "ml" | "tsp" | "tbsp") => {
if (isLiquidContainerPackageType(form.packageType)) {
if (intakeUnit === "tsp") return t("form.blisters.usageTsp");
if (intakeUnit === "tbsp") return t("form.blisters.usageTbsp");
return t("form.blisters.usageMl");
}
if (isTubePackageType(form.packageType)) {
return form.medicationForm === "liquid" ? t("form.blisters.usageMl") : t("form.blisters.usageApplication");
}
if (form.pillForm === "capsule") return t("form.blisters.usageCapsules");
return t("form.blisters.usageTablets");
},
[form.packageType, form.medicationForm, form.pillForm, t]
);
const usesAmountLabels = isTubePackageType(form.packageType) || isLiquidContainerPackageType(form.packageType);
const totalCapacityLabel = usesAmountLabels ? t("form.totalAmount") : t("form.totalCapacity");
const currentStockLabel = usesAmountLabels ? t("form.currentAmount") : t("form.currentPills");
const totalLabel = usesAmountLabels ? t("form.totalAmountLabel") : t("form.total");
const getMedicationPackageTypeLabel = useCallback(
(med: Medication) => {
return t(getPackageProfile(med.packageType).labelKey);
},
[t]
);
const getMedicationStockSuffix = useCallback(
(med: Medication) => {
if (isTubePackageType(med.packageType)) return "";
if (isLiquidContainerPackageType(med.packageType)) return " ml";
return ` ${getPackageSize(med) === 1 ? t("common.pill") : t("common.pills")}`;
},
[t]
);
const getMedicationUsageUnitLabel = useCallback(
(med: Medication, usage: number) => {
if (isTubePackageType(med.packageType)) {
return med.medicationForm === "liquid" ? "ml" : t("form.blisters.usageApplication");
}
if (isLiquidContainerPackageType(med.packageType)) return "ml";
if (usage === 1) return t("common.pill");
return t("common.pills");
},
[t]
);
const clearEditMedIdParam = useCallback(() => { const clearEditMedIdParam = useCallback(() => {
setSearchParams( setSearchParams(
@@ -450,6 +525,10 @@ export function MedicationsPage() {
return; return;
} }
if (saving) return; if (saving) return;
if (form.pillForm === "capsule" && form.intakes.some((i) => !Number.isInteger(Number(i.usage)))) {
setShowNameValidation(true);
return;
}
setSaving(true); setSaving(true);
// Prepare intakes data with per-intake takenBy // Prepare intakes data with per-intake takenBy
@@ -457,6 +536,7 @@ export function MedicationsPage() {
usage: Number(intake.usage) || 1, usage: Number(intake.usage) || 1,
every: Number(intake.every) || 1, every: Number(intake.every) || 1,
start: combineDateAndTime(intake.startDate, intake.startTime), start: combineDateAndTime(intake.startDate, intake.startTime),
intakeUnit: isLiquidContainerPackageType(form.packageType) ? intake.intakeUnit : null,
takenBy: intake.takenBy.trim() || null, // Empty string becomes null takenBy: intake.takenBy.trim() || null, // Empty string becomes null
intakeRemindersEnabled: intake.intakeRemindersEnabled, intakeRemindersEnabled: intake.intakeRemindersEnabled,
})); }));
@@ -472,19 +552,50 @@ export function MedicationsPage() {
const remainingRefills = Math.min(Number(form.prescriptionRemainingRefills || 0), authorizedRefills); const remainingRefills = Math.min(Number(form.prescriptionRemainingRefills || 0), authorizedRefills);
const lowRefillThreshold = Math.min(Number(form.prescriptionLowRefillThreshold || 1), authorizedRefills); const lowRefillThreshold = Math.min(Number(form.prescriptionLowRefillThreshold || 1), authorizedRefills);
let derivedMedicationForm: string;
if (isTubePackageType(form.packageType)) {
derivedMedicationForm =
form.medicationForm === "liquid" || form.medicationForm === "topical" ? form.medicationForm : "topical";
} else if (isLiquidContainerPackageType(form.packageType)) {
derivedMedicationForm = "liquid";
} else {
derivedMedicationForm = form.pillForm;
}
const tubeTotalAmount = isTubePackageType(form.packageType)
? (Number(form.packCount) || 0) * (Number(form.packageAmountValue ?? 0) || 0)
: null;
let packageAmountUnit = form.packageAmountUnit ?? "ml";
if (isTubePackageType(form.packageType)) {
packageAmountUnit = "g";
} else if (isLiquidContainerPackageType(form.packageType)) {
packageAmountUnit = "ml";
}
const body = { const body = {
name: form.name.trim(), name: form.name.trim(),
genericName: form.genericName.trim() || null, genericName: form.genericName.trim() || null,
takenBy: form.takenBy.length > 0 ? form.takenBy : [], takenBy: form.takenBy.length > 0 ? form.takenBy : [],
packageType: form.packageType, medicationForm: derivedMedicationForm,
packCount: Number(form.packCount) || 0, pillForm:
blistersPerPack: Number(form.blistersPerPack) || 1, isTubePackageType(form.packageType) || isLiquidContainerPackageType(form.packageType) ? null : form.pillForm,
pillsPerBlister: Number(form.pillsPerBlister) || 1, lifecycleCategory: form.lifecycleCategory,
totalPills: Number(form.totalPills) || null, packageType: normalizePackageType(form.packageType),
looseTablets: Number(form.looseTablets) || 0, packCount: isTubePackageType(form.packageType)
? Math.max(1, Number(form.packCount) || 1)
: Number(form.packCount) || 0,
blistersPerPack: isTubePackageType(form.packageType) ? 1 : Number(form.blistersPerPack) || 1,
pillsPerBlister: isTubePackageType(form.packageType) ? 1 : Number(form.pillsPerBlister) || 1,
packageAmountValue: Number(form.packageAmountValue ?? 0) || 0,
packageAmountUnit,
totalPills: isTubePackageType(form.packageType) ? tubeTotalAmount : Number(form.totalPills) || null,
looseTablets: isTubePackageType(form.packageType) ? tubeTotalAmount || 0 : Number(form.looseTablets) || 0,
pillWeightMg: Number(form.pillWeightMg) || null, pillWeightMg: Number(form.pillWeightMg) || null,
doseUnit: form.doseUnit, doseUnit: form.doseUnit,
medicationStartDate: form.medicationStartDate || null, medicationStartDate: form.medicationStartDate || null,
medicationEndDate: form.medicationEndDate || null,
autoMarkObsoleteAfterEndDate: form.autoMarkObsoleteAfterEndDate,
expiryDate: form.expiryDate || null, expiryDate: form.expiryDate || null,
notes: form.notes.trim() || null, notes: form.notes.trim() || null,
intakeRemindersEnabled: form.intakeRemindersEnabled, intakeRemindersEnabled: form.intakeRemindersEnabled,
@@ -719,6 +830,7 @@ export function MedicationsPage() {
setReadOnlyView(true); setReadOnlyView(true);
startEdit(med, openEditModal); startEdit(med, openEditModal);
setViewMode("form"); setViewMode("form");
scrollToTopForDesktopEdit();
}; };
setUnsavedConfirmSource(showEditModal ? "mobile-edit" : "desktop-form"); setUnsavedConfirmSource(showEditModal ? "mobile-edit" : "desktop-form");
setShowUnsavedConfirm(true); setShowUnsavedConfirm(true);
@@ -729,6 +841,7 @@ export function MedicationsPage() {
setActiveTab("general"); setActiveTab("general");
startEdit(med, openEditModal); startEdit(med, openEditModal);
setViewMode("form"); setViewMode("form");
scrollToTopForDesktopEdit();
} }
function handleNewEntryClick() { function handleNewEntryClick() {
@@ -792,6 +905,7 @@ export function MedicationsPage() {
setActiveTab("general"); setActiveTab("general");
startEdit(medicationToEdit, openEditModal); startEdit(medicationToEdit, openEditModal);
setViewMode("form"); setViewMode("form");
scrollToTopForDesktopEdit();
setPendingEditTransition(false); setPendingEditTransition(false);
window.dispatchEvent(new Event("medassist:edit-transition-ready")); window.dispatchEvent(new Event("medassist:edit-transition-ready"));
@@ -836,19 +950,21 @@ export function MedicationsPage() {
<span <span
className={med.imageUrl ? "med-avatar-clickable" : undefined} className={med.imageUrl ? "med-avatar-clickable" : undefined}
onClick={() => onClick={() =>
med.imageUrl && setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: med.name }) med.imageUrl &&
setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: getMedDisplayName(med) })
} }
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
if (med.imageUrl) setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: med.name }); if (med.imageUrl)
setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: getMedDisplayName(med) });
} }
}} }}
> >
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="lg" /> <MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="lg" />
</span> </span>
<div className="med-name-block"> <div className="med-name-block">
<div className="med-name">{med.name}</div> <div className="med-name">{getMedDisplayName(med)}</div>
{med.genericName && <div className="med-generic-name">{med.genericName}</div>} {med.name && med.genericName && <div className="med-generic-name">{med.genericName}</div>}
</div> </div>
</div> </div>
<div className="med-actions"> <div className="med-actions">
@@ -876,12 +992,9 @@ export function MedicationsPage() {
</div> </div>
<div className="med-details"> <div className="med-details">
<span> <span>
{t("medications.details.type")}:{" "} {t("medications.details.type")}: <strong>{getMedicationPackageTypeLabel(med)}</strong>
<strong>
{med.packageType === "bottle" ? t("form.packageTypeBottle") : t("form.packageTypeBlister")}
</strong>
</span> </span>
{med.packageType === "blister" ? ( {!isAmountBasedPackageType(med.packageType) ? (
<> <>
<span> <span>
{t("medications.details.packs")}: <strong>{med.packCount}</strong> {t("medications.details.packs")}: <strong>{med.packCount}</strong>
@@ -910,10 +1023,13 @@ export function MedicationsPage() {
)} )}
<div className="med-total"> <div className="med-total">
{t("medications.details.stock")}:{" "} {t("medications.details.stock")}:{" "}
{coverageByMed[med.name] ? Math.round(coverageByMed[med.name].medsLeft) : getPackageSize(med)} /{" "} {coverageByMed[getMedDisplayName(med)]
{getPackageSize(med)} {getPackageSize(med) === 1 ? t("common.pill") : t("common.pills")} ? Math.round(coverageByMed[getMedDisplayName(med)].medsLeft)
{(coverageByMed[med.name] : getPackageSize(med)}{" "}
? Math.round(coverageByMed[med.name].medsLeft) / {getPackageSize(med)}
{getMedicationStockSuffix(med)}
{(coverageByMed[getMedDisplayName(med)]
? Math.round(coverageByMed[getMedDisplayName(med)].medsLeft)
: getPackageSize(med)) > getPackageSize(med) && ( : getPackageSize(med)) > getPackageSize(med) && (
<span <span
className="info-tooltip tooltip-align-left warning-text" className="info-tooltip tooltip-align-left warning-text"
@@ -929,7 +1045,7 @@ export function MedicationsPage() {
<div className="blister-list"> <div className="blister-list">
{(med.intakes ?? med.blisters).map((s, idx) => ( {(med.intakes ?? med.blisters).map((s, idx) => (
<div key={`${med.id}-${idx}`} className="blister-row-simple"> <div key={`${med.id}-${idx}`} className="blister-row-simple">
{s.usage} {s.usage === 1 ? t("common.pill") : t("common.pills")} ·{" "} {s.usage} {getMedicationUsageUnitLabel(med, s.usage)} ·{" "}
{s.every === 1 ? t("common.daily") : t("common.everyNDays", { count: s.every })} ·{" "} {s.every === 1 ? t("common.daily") : t("common.everyNDays", { count: s.every })} ·{" "}
{t("form.blisters.from")} {formatDateTime(s.start)} {t("form.blisters.from")} {formatDateTime(s.start)}
{"takenBy" in s && (s as import("../types").Intake).takenBy && ( {"takenBy" in s && (s as import("../types").Intake).takenBy && (
@@ -970,20 +1086,24 @@ export function MedicationsPage() {
<span <span
className={med.imageUrl ? "med-avatar-clickable" : undefined} className={med.imageUrl ? "med-avatar-clickable" : undefined}
onClick={() => onClick={() =>
med.imageUrl && setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: med.name }) med.imageUrl &&
setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: getMedDisplayName(med) })
} }
onKeyDown={(e) => { onKeyDown={(e) => {
if (e.key === "Enter" || e.key === " ") { if (e.key === "Enter" || e.key === " ") {
if (med.imageUrl) if (med.imageUrl)
setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: med.name }); setLightboxImage({
src: `/api/images/${med.imageUrl}`,
alt: getMedDisplayName(med),
});
} }
}} }}
> >
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="lg" /> <MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="lg" />
</span> </span>
<div className="med-name-block"> <div className="med-name-block">
<div className="med-name">{med.name}</div> <div className="med-name">{getMedDisplayName(med)}</div>
{med.genericName && <div className="med-generic-name">{med.genericName}</div>} {med.name && med.genericName && <div className="med-generic-name">{med.genericName}</div>}
</div> </div>
</div> </div>
<div className="med-actions"> <div className="med-actions">
@@ -1106,27 +1226,33 @@ export function MedicationsPage() {
onBlur={() => setShowNameValidation(true)} onBlur={() => setShowNameValidation(true)}
placeholder={t("form.placeholders.commercial")} placeholder={t("form.placeholders.commercial")}
maxLength={FIELD_LIMITS.name.max} maxLength={FIELD_LIMITS.name.max}
required={!readOnlyView}
/> />
{!readOnlyView && showNameValidation && fieldErrors.name && ( {!readOnlyView && showNameValidation && fieldErrors.name && (
<span className="field-error">{fieldErrors.name}</span> <span className="field-error">{fieldErrors.name}</span>
)} )}
</label> </label>
<label className={fieldErrors.genericName ? "has-error" : ""}> <label className={!readOnlyView && showNameValidation && fieldErrors.genericName ? "has-error" : ""}>
{t("form.genericName")} {t("form.genericName")}
<input <input
value={form.genericName} value={form.genericName}
onChange={(e) => setForm({ ...form, genericName: e.target.value })} onChange={(e) => {
setShowNameValidation(true);
setForm({ ...form, genericName: e.target.value });
}}
onBlur={() => setShowNameValidation(true)}
placeholder={t("form.placeholders.generic")} placeholder={t("form.placeholders.generic")}
maxLength={FIELD_LIMITS.genericName.max} maxLength={FIELD_LIMITS.genericName.max}
/> />
{fieldErrors.genericName && <span className="field-error">{fieldErrors.genericName}</span>} {!readOnlyView && showNameValidation && fieldErrors.genericName && (
<span className="field-error">{fieldErrors.genericName}</span>
)}
</label> </label>
<label> <label>
{t("form.medicationStartDate")} {t("form.medicationStartDate")}
<DateInput <DateInput
value={form.medicationStartDate} value={form.medicationStartDate}
onChange={(e) => handleValueChange("medicationStartDate", e.target.value)} onChange={(e) => handleValueChange("medicationStartDate", e.target.value)}
placeholder={t("common.optional")}
/> />
{!readOnlyView && dateConsistencyError && ( {!readOnlyView && dateConsistencyError && (
<span className="field-error">{dateConsistencyError}</span> <span className="field-error">{dateConsistencyError}</span>
@@ -1137,14 +1263,64 @@ export function MedicationsPage() {
<select <select
className="package-type-select" className="package-type-select"
value={form.packageType} value={form.packageType}
onChange={(e) => onChange={(e) => handleValueChange("packageType", e.target.value as PackageType)}
handleValueChange("packageType", e.target.value as import("../types").PackageType)
}
> >
<option value="blister">{t("form.packageTypeBlister")}</option> {PACKAGE_PROFILES.map((profile) => (
<option value="bottle">{t("form.packageTypeBottle")}</option> <option key={profile.value} value={profile.value}>
{t(profile.labelKey)}
</option>
))}
</select> </select>
</label> </label>
<label>
{t("form.medicationEndDate")}
<DateInput
value={form.medicationEndDate}
onChange={(e) => handleValueChange("medicationEndDate", e.target.value)}
placeholder={t("common.optional")}
/>
</label>
{allowsPillFormSelection(form.packageType) && (
<label>
{t("form.pillForm")}
<select
value={form.pillForm}
onChange={(e) => handleValueChange("pillForm", e.target.value as FormState["pillForm"])}
>
<option value="tablet">{t("form.medicationFormTablet")}</option>
<option value="capsule">{t("form.medicationFormCapsule")}</option>
</select>
</label>
)}
{isTubePackageType(form.packageType) && (
<label>
{t("form.medicationForm")}
<select value={"topical"} onChange={() => handleValueChange("medicationForm", "topical")}>
<option value="topical">{t("form.medicationFormTopical")}</option>
</select>
</label>
)}
{isLiquidContainerPackageType(form.packageType) && (
<label>
{t("form.medicationForm")}
<select value={"liquid"} onChange={() => handleValueChange("medicationForm", "liquid")}>
<option value="liquid">{t("form.medicationFormLiquid")}</option>
</select>
</label>
)}
{form.medicationEndDate && (
<label className="full">
{t("form.autoMarkObsoleteAfterEndDate")}
<label className="toggle-switch small">
<input
type="checkbox"
checked={form.autoMarkObsoleteAfterEndDate}
onChange={(e) => handleValueChange("autoMarkObsoleteAfterEndDate", e.target.checked)}
/>
<span className="toggle-slider"></span>
</label>
</label>
)}
<label className={`full ${fieldErrors.takenBy ? "has-error" : ""}`}> <label className={`full ${fieldErrors.takenBy ? "has-error" : ""}`}>
{t("form.takenBy")} {t("form.takenBy")}
<div className="tag-input-container"> <div className="tag-input-container">
@@ -1258,99 +1434,177 @@ export function MedicationsPage() {
<div className={`form-tab-panel${activeTab === "stock" ? " active" : ""}`}> <div className={`form-tab-panel${activeTab === "stock" ? " active" : ""}`}>
<div className="full form-category"> <div className="full form-category">
<h4 className="form-category-title">{t("form.sections.stock")}</h4> <h4 className="form-category-title">{t("form.sections.stock")}</h4>
{form.packageType === "blister" ? ( {(() => {
<> if (!isAmountBasedPackageType(form.packageType)) {
<label> return (
{t("form.packs")} <>
<FormNumberStepper <label>
value={form.packCount} {t("form.packs")}
onChange={(nextValue) => handleValueChange("packCount", nextValue)} <FormNumberStepper
min={0} value={form.packCount}
decrementLabel={decrementValueLabel} onChange={(nextValue) => handleValueChange("packCount", nextValue)}
incrementLabel={incrementValueLabel} min={0}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label>
{t("form.blistersPerPack")}
<FormNumberStepper
value={form.blistersPerPack}
onChange={(nextValue) => handleValueChange("blistersPerPack", nextValue)}
min={1}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label>
{t("form.pillsPerBlister")}
<FormNumberStepper
value={form.pillsPerBlister}
onChange={(nextValue) => handleValueChange("pillsPerBlister", nextValue)}
min={1}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label>
{t("form.total")}
<div className="static-value">{formatNumber(totalTablets)}</div>
</label>
</>
);
}
if (isTubePackageType(form.packageType)) {
return (
<>
<label>
{t("form.tubes")}
<FormNumberStepper
value={form.packCount}
onChange={(nextValue) => handleValueChange("packCount", nextValue)}
min={1}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label className="full">
{t("form.packageAmountPerTube")}
<div className="dose-input-group">
<input
type="text"
inputMode="decimal"
pattern="[0-9]*\.?[0-9]*"
value={form.packageAmountValue ?? "0"}
onChange={(e) => handleValueChange("packageAmountValue", e.target.value)}
placeholder="0"
/>
<select
value="g"
disabled
className="dose-unit-select"
aria-label={t("form.packageAmountUnitG")}
>
<option value="g">{t("form.packageAmountUnitG")}</option>
</select>
</div>
</label>
<label>
{t("form.totalAmount")}
<div className="static-value">
{formatNumber(
(Number(form.packCount) || 0) * (Number(form.packageAmountValue ?? 0) || 0)
)}
{t("form.packageAmountUnitG")}
</div>
</label>
</>
);
}
return (
<>
<label>
{totalCapacityLabel}
<FormNumberStepper
value={form.totalPills}
onChange={(nextValue) => handleValueChange("totalPills", nextValue)}
min={0}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label>
{currentStockLabel}
<FormNumberStepper
value={form.looseTablets}
onChange={(nextValue) => handleValueChange("looseTablets", nextValue)}
min={0}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
</>
);
})()}
{allowsPillFormSelection(form.packageType) && (
<label className="full">
{t("form.pillWeight")} ({form.doseUnit})
<div className="dose-input-group">
<input
type="text"
inputMode="decimal"
pattern="[0-9]*\.?[0-9]*"
value={form.pillWeightMg}
onChange={(e) => handleValueChange("pillWeightMg", e.target.value)}
placeholder={t("form.placeholders.weight")}
/> />
</label> <select
<label> value={form.doseUnit}
{t("form.blistersPerPack")} onChange={(e) => handleValueChange("doseUnit", e.target.value as DoseUnit)}
<FormNumberStepper className="dose-unit-select"
value={form.blistersPerPack} >
onChange={(nextValue) => handleValueChange("blistersPerPack", nextValue)} {DOSE_UNITS.map((unit) => (
min={1} <option key={unit.value} value={unit.value}>
decrementLabel={decrementValueLabel} {unit.label}
incrementLabel={incrementValueLabel} </option>
/> ))}
</label> </select>
<label> </div>
{t("form.pillsPerBlister")} </label>
<FormNumberStepper
value={form.pillsPerBlister}
onChange={(nextValue) => handleValueChange("pillsPerBlister", nextValue)}
min={1}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label>
{t("form.total")}
<div className="static-value">{formatNumber(totalTablets)}</div>
</label>
</>
) : (
<>
<label>
{t("form.totalCapacity")}
<FormNumberStepper
value={form.totalPills}
onChange={(nextValue) => handleValueChange("totalPills", nextValue)}
min={0}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
<label>
{t("form.currentPills")}
<FormNumberStepper
value={form.looseTablets}
onChange={(nextValue) => handleValueChange("looseTablets", nextValue)}
min={0}
decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel}
/>
</label>
</>
)} )}
<label className="full"> {isAmountBasedPackageType(form.packageType) && !isTubePackageType(form.packageType) && (
{t("form.pillWeight")} ({form.doseUnit})
<div className="dose-input-group">
<input
type="text"
inputMode="decimal"
pattern="[0-9]*\.?[0-9]*"
value={form.pillWeightMg}
onChange={(e) => handleValueChange("pillWeightMg", e.target.value)}
placeholder={t("form.placeholders.weight")}
/>
<select
value={form.doseUnit}
onChange={(e) => handleValueChange("doseUnit", e.target.value as DoseUnit)}
className="dose-unit-select"
>
{DOSE_UNITS.map((unit) => (
<option key={unit.value} value={unit.value}>
{unit.label}
</option>
))}
</select>
</div>
</label>
{form.packageType === "bottle" && (
<div className="full stock-total-row"> <div className="full stock-total-row">
<label className="stock-total-field"> <label className="stock-total-field">
{t("form.total")} {totalLabel}
<div className="static-value">{formatNumber(totalTablets)}</div> <div className="static-value">{formatNumber(totalTablets)}</div>
</label> </label>
</div> </div>
)} )}
{isLiquidContainerPackageType(form.packageType) && (
<label className="full">
{t("form.packageAmount")}
<div className="dose-input-group">
<input
type="text"
inputMode="decimal"
pattern="[0-9]*\.?[0-9]*"
value={form.packageAmountValue ?? "0"}
onChange={(e) => handleValueChange("packageAmountValue", e.target.value)}
placeholder="0"
/>
<select
value="ml"
disabled
className="dose-unit-select"
aria-label={t("form.packageAmountUnitMl")}
>
<option value="ml">{t("form.packageAmountUnitMl")}</option>
</select>
</div>
</label>
)}
<label> <label>
{t("form.expiryDate")} {t("form.expiryDate")}
<DateInput <DateInput
@@ -1466,13 +1720,13 @@ export function MedicationsPage() {
<div key={idx} className="blister-row"> <div key={idx} className="blister-row">
<div className="blister-inputs"> <div className="blister-inputs">
<label> <label>
{t("form.blisters.usage")} {getUsageLabel(intake.intakeUnit ?? "ml")}
<FormNumberStepper <FormNumberStepper
value={intake.usage} value={intake.usage}
onChange={(nextValue) => setIntakeValue(idx, "usage", nextValue)} onChange={(nextValue) => setIntakeValue(idx, "usage", nextValue)}
min={0.5} min={allowFractionalIntake ? 0.5 : 1}
step={0.5} step={allowFractionalIntake ? 0.5 : 1}
allowDecimal={true} allowDecimal={allowFractionalIntake}
decrementLabel={decrementValueLabel} decrementLabel={decrementValueLabel}
incrementLabel={incrementValueLabel} incrementLabel={incrementValueLabel}
/> />
@@ -1502,6 +1756,21 @@ export function MedicationsPage() {
onChange={(e) => setIntakeValue(idx, "startTime", e.target.value)} onChange={(e) => setIntakeValue(idx, "startTime", e.target.value)}
/> />
</label> </label>
{isLiquidContainerPackageType(form.packageType) && (
<label>
{t("form.blisters.intakeUnit")}
<select
value={intake.intakeUnit}
onChange={(e) =>
setIntakeValue(idx, "intakeUnit", e.target.value as "ml" | "tsp" | "tbsp")
}
>
<option value="ml">{t("form.blisters.intakeUnitMl")}</option>
<option value="tsp">{t("form.blisters.intakeUnitTsp")}</option>
<option value="tbsp">{t("form.blisters.intakeUnitTbsp")}</option>
</select>
</label>
)}
{form.takenBy.length === 0 ? null : ( {form.takenBy.length === 0 ? null : (
<label className="taken-by-field" title={t("form.blisters.takenByTooltip")}> <label className="taken-by-field" title={t("form.blisters.takenByTooltip")}>
{t("form.blisters.takenByIntake")} {t("form.blisters.takenByIntake")}
+31 -5
View File
@@ -5,6 +5,7 @@ import { DateTimeInput, MedicationAvatar } from "../components";
import { useAuth } from "../components/Auth"; import { useAuth } from "../components/Auth";
import { useAppContext } from "../context"; import { useAppContext } from "../context";
import type { PlannerRow } from "../types"; import type { PlannerRow } from "../types";
import { getMedDisplayName, isAmountBasedPackageType, isLiquidContainerPackageType, isTubePackageType } from "../types";
import { toInputValue } from "../utils/formatters"; import { toInputValue } from "../utils/formatters";
// Date helpers // Date helpers
@@ -121,6 +122,30 @@ export function PlannerPage() {
const canSendNotification = const canSendNotification =
(settings.emailEnabled && settings.notificationEmail) || (settings.shoutrrrEnabled && settings.shoutrrrUrl); (settings.emailEnabled && settings.notificationEmail) || (settings.shoutrrrEnabled && settings.shoutrrrUrl);
const getUsageUnitLabel = (medicationId: number, count: number): string => {
const med = meds.find((m) => m.id === medicationId);
if (isLiquidContainerPackageType(med?.packageType)) {
return t("form.ml");
}
if (isTubePackageType(med?.packageType)) {
return med.medicationForm === "liquid" ? t("form.ml") : t("blisters.applications");
}
return count === 1 ? t("common.pill") : t("common.pills");
};
const getAvailableLabel = (medicationId: number, loosePills: number): string => {
const med = meds.find((m) => m.id === medicationId);
const roundedLoose = Math.round(loosePills * 10) / 10;
if (isLiquidContainerPackageType(med?.packageType)) {
return `${roundedLoose} ${t("form.ml")}`;
}
if (isTubePackageType(med?.packageType)) {
const unit = med.medicationForm === "liquid" ? t("form.ml") : t("blisters.applications");
return `${roundedLoose} ${unit}`;
}
return `${roundedLoose} ${roundedLoose === 1 ? t("common.pill") : t("common.pills")}`;
};
async function sendPlannerNotification() { async function sendPlannerNotification() {
if (!canSendNotification || plannerRows.length === 0) return; if (!canSendNotification || plannerRows.length === 0) return;
setSendingPlannerEmail(true); setSendingPlannerEmail(true);
@@ -204,7 +229,8 @@ export function PlannerPage() {
</div> </div>
{plannerRows.map((row) => { {plannerRows.map((row) => {
const med = const med =
meds.find((m) => m.id === row.medicationId) || meds.find((m) => m.name === row.medicationName); meds.find((m) => m.id === row.medicationId) ||
meds.find((m) => getMedDisplayName(m) === row.medicationName);
const remainingRefills = med?.prescriptionEnabled ? (med.prescriptionRemainingRefills ?? 0) : null; const remainingRefills = med?.prescriptionEnabled ? (med.prescriptionRemainingRefills ?? 0) : null;
return ( return (
<div <div
@@ -224,16 +250,16 @@ export function PlannerPage() {
<span data-label={t("planner.table.usage")}> <span data-label={t("planner.table.usage")}>
<span> <span>
<strong>{row.plannerUsage}</strong>&nbsp; <strong>{row.plannerUsage}</strong>&nbsp;
{row.plannerUsage === 1 ? t("common.pill") : t("common.pills")} {getUsageUnitLabel(row.medicationId, row.plannerUsage)}
</span> </span>
</span> </span>
<span data-label={t("planner.table.blisters")}> <span data-label={t("planner.table.blisters")}>
{row.packageType === "bottle" ? "" : `${row.blistersNeeded} × ${row.blisterSize}`} {isAmountBasedPackageType(row.packageType) ? "" : `${row.blistersNeeded} × ${row.blisterSize}`}
</span> </span>
<span data-label={t("planner.table.prescriptionRefills")}>{remainingRefills ?? ""}</span> <span data-label={t("planner.table.prescriptionRefills")}>{remainingRefills ?? ""}</span>
<span data-label={t("planner.table.available")}> <span data-label={t("planner.table.available")}>
{row.packageType === "bottle" ? ( {isAmountBasedPackageType(row.packageType) ? (
`${Math.round(row.loosePills * 10) / 10} ${Math.round(row.loosePills * 10) / 10 === 1 ? t("common.pill") : t("common.pills")}` getAvailableLabel(row.medicationId, row.loosePills)
) : ( ) : (
<> <>
{row.fullBlisters} {t("common.blisters")} {row.fullBlisters} {t("common.blisters")}
+110 -13
View File
@@ -5,6 +5,8 @@ import { MedicationAvatar } from "../components";
import { useAuth } from "../components/Auth"; import { useAuth } from "../components/Auth";
import { useAppContext } from "../context"; import { useAppContext } from "../context";
import type { Coverage } from "../types"; import type { Coverage } from "../types";
import { getMedDisplayName, isLiquidContainerPackageType, isTubePackageType } from "../types";
import { formatNumber } from "../utils/formatters";
import { expandDoseIds, isDoseDismissed } from "../utils/schedule"; import { expandDoseIds, isDoseDismissed } from "../utils/schedule";
// Helper for user-specific localStorage keys // Helper for user-specific localStorage keys
@@ -16,12 +18,21 @@ function userStorageKey(userId: number | undefined, key: string): string {
function getStockStatus( function getStockStatus(
daysLeft: number | null, daysLeft: number | null,
medsLeft: number, medsLeft: number,
settings: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number } settings: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number },
packageType?: string
) { ) {
if (isTubePackageType(packageType)) return { className: "success", label: "status.noSchedule" };
// Out of stock or completely depleted = danger (red) // Out of stock or completely depleted = danger (red)
if (medsLeft <= 0 || daysLeft === 0) return { className: "danger", label: "status.outOfStock" }; if (medsLeft <= 0 || daysLeft === 0) return { className: "danger", label: "status.outOfStock" };
// No schedule, but has stock = normal // No schedule, but has stock = normal
if (daysLeft === null) return { className: "success", label: "status.noSchedule" }; if (daysLeft === null) return { className: "success", label: "status.noSchedule" };
if (isLiquidContainerPackageType(packageType)) {
const lowDays = Math.max(1, Math.floor(settings.reminderDaysBefore));
const criticalDays = Math.max(1, Math.ceil(lowDays / 2));
if (daysLeft <= criticalDays) return { className: "danger", label: "status.criticalStock" };
if (daysLeft <= lowDays) return { className: "warning", label: "status.lowStock" };
return { className: "success", label: "status.normal" };
}
// Critical: at or below reminder threshold = danger (red) // Critical: at or below reminder threshold = danger (red)
if (daysLeft <= settings.reminderDaysBefore) return { className: "danger", label: "status.criticalStock" }; if (daysLeft <= settings.reminderDaysBefore) return { className: "danger", label: "status.criticalStock" };
// Low: below low stock threshold = warning (yellow) // Low: below low stock threshold = warning (yellow)
@@ -36,13 +47,15 @@ function getStockStatus(
function getDayStockStatus( function getDayStockStatus(
dayMeds: Array<{ medName: string }>, dayMeds: Array<{ medName: string }>,
coverageByMed: Record<string, Coverage>, coverageByMed: Record<string, Coverage>,
settings: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number } settings: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number },
meds: Array<{ name: string; genericName?: string | null; packageType?: string }>
): string { ): string {
let worstLevel = 3; // 3=success, 2=warning, 1=danger let worstLevel = 3; // 3=success, 2=warning, 1=danger
for (const item of dayMeds) { for (const item of dayMeds) {
const cov = coverageByMed[item.medName]; const cov = coverageByMed[item.medName];
if (!cov) continue; if (!cov) continue;
const status = getStockStatus(cov.daysLeft, cov.medsLeft, settings); const med = meds.find((m) => getMedDisplayName(m) === item.medName);
const status = getStockStatus(cov.daysLeft, cov.medsLeft, settings, med?.packageType);
if (status.className === "danger") worstLevel = Math.min(worstLevel, 1); if (status.className === "danger") worstLevel = Math.min(worstLevel, 1);
else if (status.className === "warning") worstLevel = Math.min(worstLevel, 2); else if (status.className === "warning") worstLevel = Math.min(worstLevel, 2);
} }
@@ -79,6 +92,87 @@ export function SchedulePage() {
missedPastDoseIds, missedPastDoseIds,
} = useAppContext(); } = useAppContext();
const shouldHideNoScheduleStatusForTube = (
med: (typeof meds)[number] | undefined,
status: { className: string; label: string } | null
) => isTubePackageType(med?.packageType) && status?.label === "status.noSchedule";
const getTubeUnitLabel = (med: (typeof meds)[number] | undefined, value: number) =>
isLiquidContainerPackageType(med?.packageType) || med?.medicationForm === "liquid"
? t("form.packageAmountUnitMl")
: t("form.blisters.applications", { count: Math.abs(value) });
const convertLiquidUsageToMl = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): number => {
if (unit === "tsp") return usage * 5;
if (unit === "tbsp") return usage * 15;
return usage;
};
const getLiquidCountUnitLabel = (unit: "ml" | "tsp" | "tbsp" | null | undefined, usage: number): string => {
if (unit === "tsp") return t("form.blisters.teaspoons", { count: Math.abs(usage) });
if (unit === "tbsp") return t("form.blisters.tablespoons", { count: Math.abs(usage) });
return t("form.packageAmountUnitMl");
};
const formatLiquidUsageLabel = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): string => {
const normalizedUsage = Number(usage);
if (!Number.isFinite(normalizedUsage) || normalizedUsage <= 0) {
return `0 ${t("form.packageAmountUnitMl")}`;
}
if (unit === "ml" || unit == null) {
return `${formatNumber(normalizedUsage)} ${t("form.packageAmountUnitMl")}`;
}
const mlTotal = convertLiquidUsageToMl(normalizedUsage, unit);
return `${formatNumber(normalizedUsage)} ${getLiquidCountUnitLabel(unit, normalizedUsage)} ${formatNumber(mlTotal)} ${t("form.packageAmountUnitMl")}`;
};
const formatDoseUsageLabel = (
med: (typeof meds)[number] | undefined,
usage: number,
intakeUnit?: "ml" | "tsp" | "tbsp" | null
) => {
if (isLiquidContainerPackageType(med?.packageType)) {
return formatLiquidUsageLabel(usage, intakeUnit);
}
if (isTubePackageType(med?.packageType)) {
return `${usage} ${getTubeUnitLabel(med, usage)}`;
}
return `${usage} ${usage !== 1 ? t("common.pills") : t("common.pill")}`;
};
const formatTotalUsageLabel = (
med: (typeof meds)[number] | undefined,
total: number,
doses?: Array<{ usage: number; intakeUnit?: "ml" | "tsp" | "tbsp" | null }>
) => {
if (isLiquidContainerPackageType(med?.packageType)) {
if (doses && doses.length > 0) {
const normalizedDoses = doses.filter((dose) => Number.isFinite(Number(dose.usage)) && Number(dose.usage) > 0);
if (normalizedDoses.length > 0) {
const allUnits = new Set(normalizedDoses.map((dose) => dose.intakeUnit ?? "ml"));
if (allUnits.size === 1) {
const onlyUnit = normalizedDoses[0]?.intakeUnit ?? "ml";
const totalUsageInUnit = normalizedDoses.reduce((sum, dose) => sum + Number(dose.usage), 0);
return formatLiquidUsageLabel(totalUsageInUnit, onlyUnit);
}
const totalMl = normalizedDoses.reduce(
(sum, dose) => sum + convertLiquidUsageToMl(Number(dose.usage), dose.intakeUnit ?? "ml"),
0
);
return `${formatNumber(totalMl)} ${t("form.packageAmountUnitMl")}`;
}
}
return `${formatNumber(total)} ${t("form.packageAmountUnitMl")}`;
}
if (isTubePackageType(med?.packageType)) {
return `${total} ${getTubeUnitLabel(med, total)}`;
}
return t("common.pillsTotal", { count: total });
};
return ( return (
<section className="grid"> <section className="grid">
<article className="card schedule-full"> <article className="card schedule-full">
@@ -116,7 +210,7 @@ export function SchedulePage() {
// Count missed doses that are NOT dismissed (for warning icon) // Count missed doses that are NOT dismissed (for warning icon)
const missedNotDismissedCount = day.meds.reduce((count, item) => { const missedNotDismissedCount = day.meds.reduce((count, item) => {
const med = meds.find((m) => m.name === item.medName); const med = meds.find((m) => getMedDisplayName(m) === item.medName);
const dismissedUntilDate = med?.dismissedUntil ?? undefined; const dismissedUntilDate = med?.dismissedUntil ?? undefined;
return ( return (
count + count +
@@ -132,7 +226,7 @@ export function SchedulePage() {
const isManuallyExpanded = manuallyExpandedDays.has(day.dateStr); const isManuallyExpanded = manuallyExpandedDays.has(day.dateStr);
const isCollapsed = !isManuallyExpanded; const isCollapsed = !isManuallyExpanded;
const _worstStatus = getDayStockStatus(day.meds, coverageByMed, settings); const _worstStatus = getDayStockStatus(day.meds, coverageByMed, settings, meds);
return ( return (
<div <div
@@ -171,7 +265,7 @@ export function SchedulePage() {
</div> </div>
{!isCollapsed && {!isCollapsed &&
day.meds.map((item) => { day.meds.map((item) => {
const med = meds.find((m) => m.name === item.medName); const med = meds.find((m) => getMedDisplayName(m) === item.medName);
const medCov = coverageByMed[item.medName]; const medCov = coverageByMed[item.medName];
const isEmpty = medCov ? medCov.medsLeft <= 0 : false; const isEmpty = medCov ? medCov.medsLeft <= 0 : false;
const itemDoseIds = expandDoseIds(item.doses); const itemDoseIds = expandDoseIds(item.doses);
@@ -184,7 +278,7 @@ export function SchedulePage() {
<span className="med-name-text">{item.medName}</span> <span className="med-name-text">{item.medName}</span>
</div> </div>
<div className="tag-row"> <div className="tag-row">
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span> <span className="tag subtle">{formatTotalUsageLabel(med, item.total, item.doses)}</span>
</div> </div>
</div> </div>
<div className="doses-col"> <div className="doses-col">
@@ -196,7 +290,7 @@ export function SchedulePage() {
<span className="dose-time">{dose.timeStr}</span> <span className="dose-time">{dose.timeStr}</span>
<span className="dose-usage"> <span className="dose-usage">
<span className="dose-usage-main"> <span className="dose-usage-main">
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")} {formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
</span> </span>
{med?.pillWeightMg && ( {med?.pillWeightMg && (
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span> <span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
@@ -333,15 +427,16 @@ export function SchedulePage() {
{day.meds.map((item) => { {day.meds.map((item) => {
const medCoverage = coverageByMed[item.medName]; const medCoverage = coverageByMed[item.medName];
const isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false; const isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
const med = meds.find((m) => m.name === item.medName); const med = meds.find((m) => getMedDisplayName(m) === item.medName);
const depletionTime = depletionByMed[item.medName]; const depletionTime = depletionByMed[item.medName];
// Check if this dose is scheduled after medication runs out // Check if this dose is scheduled after medication runs out
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime; const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
const status = willBeOutOfStock const status = willBeOutOfStock
? { className: "danger", label: "status.outOfStock" } ? { className: "danger", label: "status.outOfStock" }
: medCoverage : medCoverage
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings) ? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings, med?.packageType)
: null; : null;
const visibleStatus = shouldHideNoScheduleStatusForTube(med, status) ? null : status;
const itemDoseIds = expandDoseIds(item.doses); const itemDoseIds = expandDoseIds(item.doses);
const allTaken = itemDoseIds.every((id) => takenDoses.has(id)); const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
return ( return (
@@ -352,8 +447,10 @@ export function SchedulePage() {
<span className="med-name-text">{item.medName}</span> <span className="med-name-text">{item.medName}</span>
</div> </div>
<div className="tag-row"> <div className="tag-row">
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span> <span className="tag subtle">{formatTotalUsageLabel(med, item.total, item.doses)}</span>
{status && <span className={`tag ${status.className}`}>{t(status.label)}</span>} {visibleStatus && (
<span className={`tag ${visibleStatus.className}`}>{t(visibleStatus.label)}</span>
)}
</div> </div>
</div> </div>
<div className="doses-col"> <div className="doses-col">
@@ -367,7 +464,7 @@ export function SchedulePage() {
<span className="dose-time">{dose.timeStr}</span> <span className="dose-time">{dose.timeStr}</span>
<span className="dose-usage"> <span className="dose-usage">
<span className="dose-usage-main"> <span className="dose-usage-main">
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")} {formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
</span> </span>
{med?.pillWeightMg && ( {med?.pillWeightMg && (
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span> <span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
+9 -2
View File
@@ -479,11 +479,15 @@ export function SettingsPage() {
</div> </div>
<div className="schedule-row"> <div className="schedule-row">
<span className="schedule-label">{t("settings.schedule.stockCheck")}</span> <span className="schedule-label">{t("settings.schedule.stockCheck")}</span>
<span className="schedule-value">{t("settings.schedule.dailyAt6")}</span> <span className="schedule-value">
{t("settings.schedule.dailyAtHour", { hour: settings.reminderHour })}
</span>
</div> </div>
<div className="schedule-row"> <div className="schedule-row">
<span className="schedule-label">{t("settings.schedule.intakeCheck")}</span> <span className="schedule-label">{t("settings.schedule.intakeCheck")}</span>
<span className="schedule-value">{t("settings.schedule.15minBefore")}</span> <span className="schedule-value">
{t("settings.schedule.minutesBefore", { minutes: settings.reminderMinutesBefore })}
</span>
</div> </div>
{settings.nextScheduledCheck && ( {settings.nextScheduledCheck && (
<div className="schedule-row"> <div className="schedule-row">
@@ -663,6 +667,9 @@ export function SettingsPage() {
settings.lowStockDays >= settings.highStockDays) && ( settings.lowStockDays >= settings.highStockDays) && (
<p className="threshold-validation-error">{t("settings.stock.thresholdValidation")}</p> <p className="threshold-validation-error">{t("settings.stock.thresholdValidation")}</p>
)} )}
<p className="hint-text" style={{ marginTop: "12px" }}>
{t("settings.stock.packageTypesNote")}
</p>
</div> </div>
</article> </article>
+17 -6
View File
@@ -1,5 +1,5 @@
import type { Coverage } from "../types"; import type { Coverage, Medication, PackageType } from "../types";
import { getMedTotal as getMedTotalFromTypes } from "../types"; import { getMedTotal as getMedTotalFromTypes, isLiquidContainerPackageType, isTubePackageType } from "../types";
import { splitCurrentBlisterStock } from "../utils/stock"; import { splitCurrentBlisterStock } from "../utils/stock";
export function userStorageKey(userId: number | undefined, key: string): string { export function userStorageKey(userId: number | undefined, key: string): string {
@@ -43,9 +43,12 @@ export function getMedTotal(med: {
pillsPerBlister: number; pillsPerBlister: number;
looseTablets: number; looseTablets: number;
stockAdjustment?: number | null; stockAdjustment?: number | null;
packageType?: string; packageType?: PackageType;
}): number { }): number {
return getMedTotalFromTypes(med); return getMedTotalFromTypes({
...med,
stockAdjustment: med.stockAdjustment ?? undefined,
});
} }
export function getReminderStatusData( export function getReminderStatusData(
@@ -53,6 +56,7 @@ export function getReminderStatusData(
lowStockDays: number, lowStockDays: number,
_allLowCoverage: Coverage[], _allLowCoverage: Coverage[],
allCoverage: Coverage[], allCoverage: Coverage[],
meds: Medication[],
lastAutoEmailSent: string | null, lastAutoEmailSent: string | null,
_lastNotificationType: string | null, _lastNotificationType: string | null,
_lastNotificationChannel: string | null, _lastNotificationChannel: string | null,
@@ -70,8 +74,12 @@ export function getReminderStatusData(
lastIntakeSent: { date: string; medName: string | null; takenBy: string | null } | null; lastIntakeSent: { date: string; medName: string | null; takenBy: string | null } | null;
} { } {
const lowStockMap = new Map<string, { name: string; daysLeft: number; isCritical: boolean }>(); const lowStockMap = new Map<string, { name: string; daysLeft: number; isCritical: boolean }>();
const medByName = new Map(meds.map((med) => [med.name || med.genericName || "", med] as const));
for (const c of allCoverage) { for (const c of allCoverage) {
const med = medByName.get(c.name);
if (isTubePackageType(med?.packageType)) continue;
if (c.medsLeft <= 0) { if (c.medsLeft <= 0) {
lowStockMap.set(c.name, { name: c.name, daysLeft: 0, isCritical: true }); lowStockMap.set(c.name, { name: c.name, daysLeft: 0, isCritical: true });
continue; continue;
@@ -80,8 +88,11 @@ export function getReminderStatusData(
if (c.daysLeft === null) continue; if (c.daysLeft === null) continue;
const roundedDaysLeft = Math.round(c.daysLeft); const roundedDaysLeft = Math.round(c.daysLeft);
const isCritical = c.daysLeft <= reminderDaysBefore; const isLiquid = isLiquidContainerPackageType(med?.packageType);
const isLow = c.daysLeft < lowStockDays; const liquidLowDays = Math.max(1, Math.floor(reminderDaysBefore));
const liquidCriticalDays = Math.max(1, Math.ceil(liquidLowDays / 2));
const isCritical = isLiquid ? c.daysLeft <= liquidCriticalDays : c.daysLeft <= reminderDaysBefore;
const isLow = isLiquid ? c.daysLeft <= liquidLowDays : c.daysLeft < lowStockDays;
if (!isCritical && !isLow) continue; if (!isCritical && !isLow) continue;
const existing = lowStockMap.get(c.name); const existing = lowStockMap.get(c.name);
+60 -22
View File
@@ -104,7 +104,7 @@ body.modal-open {
.page { .page {
max-width: 1200px; max-width: 1200px;
margin: 0 auto; margin: 0 auto;
padding: 2.5rem 1.5rem 3rem; padding: 2.5rem 1.5rem 1.5rem;
overflow-x: hidden; overflow-x: hidden;
} }
@@ -121,7 +121,7 @@ body.modal-open {
.route-transition-mask.active { .route-transition-mask.active {
transition: none; transition: none;
opacity: 1; opacity: 1;
pointer-events: auto; pointer-events: none;
} }
.hero { .hero {
@@ -669,6 +669,16 @@ body.modal-open {
background: color-mix(in srgb, var(--bg-secondary) 75%, var(--bg-tertiary)); background: color-mix(in srgb, var(--bg-secondary) 75%, var(--bg-tertiary));
} }
.med-grid-wrapper.is-empty .med-group-active {
padding: 0.7rem 0.85rem;
}
.med-empty-state {
color: var(--text-secondary);
font-size: 0.92rem;
padding: 0.35rem 0.1rem;
}
.med-group-head { .med-group-head {
display: flex; display: flex;
align-items: center; align-items: center;
@@ -2660,6 +2670,11 @@ button.has-validation-error {
grid-template-columns: minmax(140px, 1.5fr) 90px 70px 100px 100px 90px 90px; grid-template-columns: minmax(140px, 1.5fr) 90px 70px 100px 100px 90px 90px;
} }
.table-8 .table-head,
.table-8 .table-row {
grid-template-columns: minmax(130px, 1.4fr) 90px 130px 70px 95px 95px 90px 95px;
}
.email-sent-status { .email-sent-status {
font-size: 0.8rem; font-size: 0.8rem;
color: var(--success); color: var(--success);
@@ -2842,7 +2857,7 @@ button.has-validation-error {
@media (max-width: 600px) { @media (max-width: 600px) {
.page { .page {
padding: 0.75rem 0.4rem 2rem; padding: 0.75rem 0.4rem 1rem;
} }
.grid { .grid {
@@ -4674,55 +4689,78 @@ button.has-validation-error {
} }
.med-detail-schedules { .med-detail-schedules {
display: flex; display: grid;
flex-direction: column; grid-template-columns: auto auto 1fr auto auto auto;
gap: 0.5rem; gap: 0.45rem 0;
} }
.med-schedule-item { .med-schedule-row {
display: flex; display: grid;
grid-column: 1 / -1;
grid-template-columns: subgrid;
align-items: center; align-items: center;
flex-wrap: wrap; column-gap: 0.75rem;
gap: 0.35rem 0.75rem;
background: var(--bg-secondary);
padding: 0.75rem 1rem;
border-radius: 8px;
font-size: 0.9rem;
} }
.med-schedule-usage { .med-schedule-usage {
font-weight: 600; font-weight: 600;
color: var(--accent); color: var(--accent);
white-space: nowrap;
grid-column: 1;
} }
.med-schedule-freq { .med-schedule-freq {
color: var(--text-secondary); color: var(--text-secondary);
white-space: nowrap; white-space: nowrap;
} grid-column: 2;
.med-schedule-time {
font-weight: 500;
margin-left: auto;
white-space: nowrap;
} }
.med-schedule-person { .med-schedule-person {
color: var(--text-secondary); color: var(--text-secondary);
font-size: 0.85rem;
white-space: nowrap; white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
grid-column: 4;
}
.med-schedule-time {
font-weight: 700;
white-space: nowrap;
color: var(--text-primary);
text-align: right;
grid-column: 5;
} }
.med-schedule-bell { .med-schedule-bell {
color: var(--warning); color: var(--warning);
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
margin-left: 0.35rem; grid-column: 6;
} }
[data-theme="light"] .med-schedule-bell { [data-theme="light"] .med-schedule-bell {
color: #b45309; color: #b45309;
} }
@media (max-width: 700px) {
.med-detail-schedules {
grid-template-columns: 1fr;
}
.med-schedule-row {
grid-template-columns: 1fr;
row-gap: 0.25rem;
}
.med-schedule-usage,
.med-schedule-freq,
.med-schedule-person,
.med-schedule-time,
.med-schedule-bell {
grid-column: auto;
}
}
.med-detail-footer { .med-detail-footer {
padding: 1rem 2rem 1.5rem; padding: 1rem 2rem 1.5rem;
display: flex; display: flex;
@@ -500,6 +500,8 @@
color: var(--text-primary); color: var(--text-primary);
font-size: 0.95rem; font-size: 0.95rem;
font-family: inherit; font-family: inherit;
text-transform: none;
letter-spacing: normal;
white-space: nowrap; white-space: nowrap;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
@@ -36,7 +36,7 @@ describe("AppHeader", () => {
json: () => json: () =>
Promise.resolve({ Promise.resolve({
authEnabled: false, authEnabled: false,
localAuthEnabled: true, formLoginEnabled: true,
hasUsers: false, hasUsers: false,
needsSetup: false, needsSetup: false,
}), }),
@@ -171,7 +171,7 @@ describe("AppHeader", () => {
json: () => json: () =>
Promise.resolve({ Promise.resolve({
authEnabled: false, authEnabled: false,
localAuthEnabled: true, formLoginEnabled: true,
hasUsers: false, hasUsers: false,
needsSetup: false, needsSetup: false,
}), }),
@@ -205,7 +205,7 @@ describe("AppHeader", () => {
json: () => json: () =>
Promise.resolve({ Promise.resolve({
authEnabled: false, authEnabled: false,
localAuthEnabled: true, formLoginEnabled: true,
hasUsers: false, hasUsers: false,
needsSetup: false, needsSetup: false,
}), }),
@@ -239,7 +239,7 @@ describe("AppHeader", () => {
json: () => json: () =>
Promise.resolve({ Promise.resolve({
authEnabled: false, authEnabled: false,
localAuthEnabled: true, formLoginEnabled: true,
hasUsers: false, hasUsers: false,
needsSetup: false, needsSetup: false,
}), }),
@@ -322,7 +322,7 @@ describe("AppHeader", () => {
Promise.resolve({ Promise.resolve({
authEnabled: true, authEnabled: true,
registrationEnabled: true, registrationEnabled: true,
localAuthEnabled: true, formLoginEnabled: true,
oidcEnabled: false, oidcEnabled: false,
oidcProviderName: "", oidcProviderName: "",
hasUsers: true, hasUsers: true,
+11 -11
View File
@@ -11,7 +11,7 @@ describe("AuthProvider", () => {
vi.resetAllMocks(); vi.resetAllMocks();
(global.fetch as ReturnType<typeof vi.fn>).mockResolvedValue({ (global.fetch as ReturnType<typeof vi.fn>).mockResolvedValue({
ok: true, ok: true,
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }), json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
}); });
}); });
@@ -79,7 +79,7 @@ describe("AuthProvider", () => {
(global.fetch as ReturnType<typeof vi.fn>) (global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({ .mockResolvedValueOnce({
ok: true, ok: true,
json: () => Promise.resolve({ authEnabled: false, localAuthEnabled: true }), json: () => Promise.resolve({ authEnabled: false, formLoginEnabled: true }),
}) })
.mockResolvedValueOnce({ ok: false, status: 401 }) .mockResolvedValueOnce({ ok: false, status: 401 })
.mockResolvedValueOnce({ ok: true, status: 200 }) .mockResolvedValueOnce({ ok: true, status: 200 })
@@ -116,7 +116,7 @@ describe("AuthProvider", () => {
(global.fetch as ReturnType<typeof vi.fn>) (global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({ .mockResolvedValueOnce({
ok: true, ok: true,
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }), json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
}) })
.mockResolvedValueOnce({ ok: true, status: 200, json: () => Promise.resolve({ id: 1, username: "tester" }) }) .mockResolvedValueOnce({ ok: true, status: 200, json: () => Promise.resolve({ id: 1, username: "tester" }) })
.mockResolvedValueOnce({ ok: false, status: 401 }) .mockResolvedValueOnce({ ok: false, status: 401 })
@@ -141,7 +141,7 @@ describe("AuthProvider", () => {
(global.fetch as ReturnType<typeof vi.fn>) (global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({ .mockResolvedValueOnce({
ok: true, ok: true,
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }), json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
}) })
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "timer-user" }) }) .mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "timer-user" }) })
.mockResolvedValueOnce({ ok: true, status: 200 }); .mockResolvedValueOnce({ ok: true, status: 200 });
@@ -167,7 +167,7 @@ describe("AuthProvider", () => {
describe("LoginForm", () => { describe("LoginForm", () => {
const mockAuthState = { const mockAuthState = {
authEnabled: true, authEnabled: true,
localAuthEnabled: true, formLoginEnabled: true,
oidcEnabled: false, oidcEnabled: false,
registrationEnabled: true, registrationEnabled: true,
hasUsers: true, hasUsers: true,
@@ -281,7 +281,7 @@ describe("LoginForm", () => {
json: () => json: () =>
Promise.resolve({ Promise.resolve({
authEnabled: false, authEnabled: false,
localAuthEnabled: true, formLoginEnabled: true,
oidcEnabled: false, oidcEnabled: false,
registrationEnabled: true, registrationEnabled: true,
hasUsers: true, hasUsers: true,
@@ -317,7 +317,7 @@ describe("LoginForm", () => {
describe("RegisterForm", () => { describe("RegisterForm", () => {
const mockAuthState = { const mockAuthState = {
authEnabled: true, authEnabled: true,
localAuthEnabled: true, formLoginEnabled: true,
oidcEnabled: false, oidcEnabled: false,
registrationEnabled: true, registrationEnabled: true,
hasUsers: false, hasUsers: false,
@@ -404,7 +404,7 @@ describe("RegisterForm", () => {
json: () => json: () =>
Promise.resolve({ Promise.resolve({
authEnabled: true, authEnabled: true,
localAuthEnabled: true, formLoginEnabled: true,
oidcEnabled: false, oidcEnabled: false,
registrationEnabled: true, registrationEnabled: true,
hasUsers: false, hasUsers: false,
@@ -439,7 +439,7 @@ describe("RegisterForm", () => {
describe("AuthPage", () => { describe("AuthPage", () => {
const mockAuthState = { const mockAuthState = {
authEnabled: true, authEnabled: true,
localAuthEnabled: true, formLoginEnabled: true,
oidcEnabled: false, oidcEnabled: false,
registrationEnabled: true, registrationEnabled: true,
hasUsers: true, hasUsers: true,
@@ -504,7 +504,7 @@ describe("UserProfile", () => {
(global.fetch as ReturnType<typeof vi.fn>) (global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({ .mockResolvedValueOnce({
ok: true, ok: true,
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }), json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
}) })
.mockResolvedValueOnce({ .mockResolvedValueOnce({
ok: true, ok: true,
@@ -724,7 +724,7 @@ describe("AuthProvider methods", () => {
it("refreshUser retries after token refresh on 401", async () => { it("refreshUser retries after token refresh on 401", async () => {
vi.clearAllMocks(); vi.clearAllMocks();
(global.fetch as ReturnType<typeof vi.fn>) (global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ authEnabled: false, localAuthEnabled: true }) }) .mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ authEnabled: false, formLoginEnabled: true }) })
.mockResolvedValueOnce({ ok: false, status: 401 }) .mockResolvedValueOnce({ ok: false, status: 401 })
.mockResolvedValueOnce({ ok: true }) .mockResolvedValueOnce({ ok: true })
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "refreshed-user" }) }); .mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "refreshed-user" }) });
@@ -1,5 +1,5 @@
import { fireEvent, render, screen } from "@testing-library/react"; import { fireEvent, render, screen } from "@testing-library/react";
import { describe, expect, it, vi } from "vitest"; import { beforeEach, describe, expect, it, vi } from "vitest";
import { Lightbox } from "../../components/Lightbox"; import { Lightbox } from "../../components/Lightbox";
describe("Lightbox", () => { describe("Lightbox", () => {
@@ -55,6 +55,8 @@ const defaultProps = {
onRefillPacksChange: vi.fn(), onRefillPacksChange: vi.fn(),
refillLoose: 0, refillLoose: 0,
onRefillLooseChange: vi.fn(), onRefillLooseChange: vi.fn(),
usePrescriptionRefill: false,
onUsePrescriptionRefillChange: vi.fn(),
refillSaving: false, refillSaving: false,
refillHistory: [] as RefillEntry[], refillHistory: [] as RefillEntry[],
refillHistoryExpanded: false, refillHistoryExpanded: false,
@@ -324,7 +326,7 @@ describe("MedDetailModal with refill modal", () => {
const submitBtn = document.querySelector(".refill-modal .modal-footer .success") as HTMLButtonElement; const submitBtn = document.querySelector(".refill-modal .modal-footer .success") as HTMLButtonElement;
fireEvent.click(submitBtn); fireEvent.click(submitBtn);
expect(onSubmitRefill).toHaveBeenCalledWith(mockMedication.id, undefined); expect(onSubmitRefill).toHaveBeenCalledWith(mockMedication.id, false);
}); });
it("disables refill submit button when no pills are entered", () => { it("disables refill submit button when no pills are entered", () => {
@@ -589,7 +591,7 @@ describe("MedDetailModal with refill history", () => {
it("shows refill history when expanded", () => { it("shows refill history when expanded", () => {
const refillHistory: RefillEntry[] = [ const refillHistory: RefillEntry[] = [
{ id: 1, medicationId: 1, timestamp: new Date().toISOString(), packsAdded: 1, looseAdded: 0 }, { id: 1, refillDate: new Date().toISOString(), packsAdded: 1, loosePillsAdded: 0 },
]; ];
render(<MedDetailModal {...defaultProps} refillHistory={refillHistory} refillHistoryExpanded={true} />); render(<MedDetailModal {...defaultProps} refillHistory={refillHistory} refillHistoryExpanded={true} />);
@@ -602,7 +604,7 @@ describe("MedDetailModal with refill history", () => {
it("calls onRefillHistoryExpandedChange when toggle clicked", () => { it("calls onRefillHistoryExpandedChange when toggle clicked", () => {
const onRefillHistoryExpandedChange = vi.fn(); const onRefillHistoryExpandedChange = vi.fn();
const refillHistory: RefillEntry[] = [ const refillHistory: RefillEntry[] = [
{ id: 1, medicationId: 1, timestamp: new Date().toISOString(), packsAdded: 1, looseAdded: 0 }, { id: 1, refillDate: new Date().toISOString(), packsAdded: 1, loosePillsAdded: 0 },
]; ];
render( render(
@@ -642,9 +644,9 @@ describe("MedDetailModal intake schedule usage display", () => {
}; };
render(<MedDetailModal {...defaultProps} selectedMed={med} />); render(<MedDetailModal {...defaultProps} selectedMed={med} />);
const usageElements = document.querySelectorAll(".med-schedule-usage"); const rows = document.querySelectorAll(".med-schedule-row .med-schedule-usage");
// Each intake should show "1 pill" (not "2 pills") // Each intake should show "1" in usage (not "2")
usageElements.forEach((el) => { rows.forEach((el) => {
expect(el.textContent).toContain("1"); expect(el.textContent).toContain("1");
expect(el.textContent).not.toMatch(/^2\b/); expect(el.textContent).not.toMatch(/^2\b/);
}); });
@@ -660,10 +662,10 @@ describe("MedDetailModal intake schedule usage display", () => {
}; };
render(<MedDetailModal {...defaultProps} selectedMed={med} />); render(<MedDetailModal {...defaultProps} selectedMed={med} />);
const usageElements = document.querySelectorAll(".med-schedule-usage"); const rows = document.querySelectorAll(".med-schedule-row .med-schedule-usage");
// Legacy: 1 pill * 2 people = "2 pills" // Legacy: 1 pill * 2 people = "2 pills"
expect(usageElements.length).toBe(1); expect(rows.length).toBe(1);
expect(usageElements[0].textContent).toContain("2"); expect(rows[0].textContent).toContain("2");
}); });
it("shows correct usage for single person with per-intake takenBy", () => { it("shows correct usage for single person with per-intake takenBy", () => {
@@ -676,11 +678,11 @@ describe("MedDetailModal intake schedule usage display", () => {
}; };
render(<MedDetailModal {...defaultProps} selectedMed={med} />); render(<MedDetailModal {...defaultProps} selectedMed={med} />);
const usageElements = document.querySelectorAll(".med-schedule-usage"); const rows = document.querySelectorAll(".med-schedule-row .med-schedule-usage");
expect(usageElements.length).toBe(1); expect(rows.length).toBe(1);
// Should show "2 pills (1000 mg)" - usage=2, not multiplied // Should show "2 pills (1000 mg)" - usage=2, not multiplied
expect(usageElements[0].textContent).toContain("2"); expect(rows[0].textContent).toContain("2");
expect(usageElements[0].textContent).toContain("1000"); expect(rows[0].textContent).toContain("1000");
}); });
}); });
@@ -1,4 +1,5 @@
import { fireEvent, render, screen } from "@testing-library/react"; import { fireEvent, render, screen } from "@testing-library/react";
import type { FormEvent } from "react";
import { beforeEach, describe, expect, it, vi } from "vitest"; import { beforeEach, describe, expect, it, vi } from "vitest";
import { MobileEditModal } from "../../components/MobileEditModal"; import { MobileEditModal } from "../../components/MobileEditModal";
import type { FormState } from "../../types"; import type { FormState } from "../../types";
@@ -7,15 +8,22 @@ const defaultForm: FormState = {
name: "", name: "",
genericName: "", genericName: "",
takenBy: [], takenBy: [],
medicationForm: "tablet",
pillForm: "tablet",
lifecycleCategory: "refill_when_empty",
packageType: "blister", packageType: "blister",
packCount: "1", packCount: "1",
blistersPerPack: "1", blistersPerPack: "1",
pillsPerBlister: "1", pillsPerBlister: "1",
packageAmountValue: "0",
packageAmountUnit: "ml",
looseTablets: "0", looseTablets: "0",
totalPills: "", totalPills: "",
pillWeightMg: "", pillWeightMg: "",
doseUnit: "mg", doseUnit: "mg",
medicationStartDate: "", medicationStartDate: "",
medicationEndDate: "",
autoMarkObsoleteAfterEndDate: true,
expiryDate: "", expiryDate: "",
notes: "", notes: "",
intakeRemindersEnabled: false, intakeRemindersEnabled: false,
@@ -78,6 +86,7 @@ const defaultProps = {
meds: [], meds: [],
onUploadMedImage: vi.fn(), onUploadMedImage: vi.fn(),
onDeleteMedImage: vi.fn(), onDeleteMedImage: vi.fn(),
imageUploadError: null,
onClose: vi.fn(), onClose: vi.fn(),
onResetForm: vi.fn(), onResetForm: vi.fn(),
onSaveMedication: vi.fn(), onSaveMedication: vi.fn(),
@@ -233,6 +242,54 @@ describe("MobileEditModal", () => {
const header = document.querySelector(".edit-modal-header"); const header = document.querySelector(".edit-modal-header");
expect(header).toBeInTheDocument(); expect(header).toBeInTheDocument();
}); });
it("uses plain numeric input for tube amount without stepper controls", () => {
render(
<MobileEditModal
{...defaultProps}
form={{
...defaultForm,
packageType: "tube",
medicationForm: "topical",
packageAmountValue: "150",
packageAmountUnit: "g",
}}
/>
);
const amountInput = screen.getByLabelText("form.packageAmountPerTube") as HTMLInputElement;
expect(amountInput).toBeInTheDocument();
expect(amountInput.tagName).toBe("INPUT");
expect(amountInput).toHaveAttribute("inputmode", "decimal");
const unitSelect = screen.getByLabelText("form.packageAmountUnitG") as HTMLSelectElement;
expect(unitSelect).toBeDisabled();
expect(unitSelect.value).toBe("g");
});
it("uses plain numeric input for liquid container package amount", () => {
render(
<MobileEditModal
{...defaultProps}
form={{
...defaultForm,
packageType: "liquid_container",
medicationForm: "liquid",
packageAmountValue: "250",
packageAmountUnit: "ml",
}}
/>
);
const amountInput = screen.getByLabelText("form.packageAmountPerBottle") as HTMLInputElement;
expect(amountInput).toBeInTheDocument();
expect(amountInput.tagName).toBe("INPUT");
expect(amountInput).toHaveAttribute("inputmode", "decimal");
const unitSelect = screen.getByLabelText("form.packageAmountUnitMl") as HTMLSelectElement;
expect(unitSelect).toBeDisabled();
expect(unitSelect.value).toBe("ml");
});
}); });
describe("MobileEditModal with existing people", () => { describe("MobileEditModal with existing people", () => {
@@ -383,7 +440,7 @@ describe("MobileEditModal form submission", () => {
}); });
it("calls onSaveMedication when form submitted", () => { it("calls onSaveMedication when form submitted", () => {
const onSaveMedication = vi.fn((e: Event) => e.preventDefault()); const onSaveMedication = vi.fn((e: FormEvent) => e.preventDefault());
const validForm = { ...defaultForm, name: "TestMed" }; const validForm = { ...defaultForm, name: "TestMed" };
render(<MobileEditModal {...defaultProps} form={validForm} onSaveMedication={onSaveMedication} />); render(<MobileEditModal {...defaultProps} form={validForm} onSaveMedication={onSaveMedication} />);
@@ -66,27 +66,7 @@ describe("ProfileModal", () => {
expect(onClose).not.toHaveBeenCalled(); expect(onClose).not.toHaveBeenCalled();
}); });
it("calls onClose when Escape is pressed on overlay", () => { // ESC key handling is tested at the App level — the global handler in
const onClose = vi.fn(); // App.tsx manages Escape for all modals, so per-component ESC tests are
render(<ProfileModal isOpen={true} onClose={onClose} />); // not applicable here.
const overlay = document.querySelector(".modal-overlay");
if (overlay) {
fireEvent.keyDown(overlay, { key: "Escape" });
}
expect(onClose).toHaveBeenCalledTimes(1);
});
it("does not close on non-escape keydown", () => {
const onClose = vi.fn();
render(<ProfileModal isOpen={true} onClose={onClose} />);
const overlay = document.querySelector(".modal-overlay");
if (overlay) {
fireEvent.keyDown(overlay, { key: "Enter" });
}
expect(onClose).not.toHaveBeenCalled();
});
}); });
@@ -15,6 +15,7 @@ const mockMedication: Medication = {
id: 1, id: 1,
name: "Test Med", name: "Test Med",
genericName: "Generic Name", genericName: "Generic Name",
packageType: "blister",
packCount: 1, packCount: 1,
blistersPerPack: 1, blistersPerPack: 1,
pillsPerBlister: 30, pillsPerBlister: 30,
+12 -7
View File
@@ -39,18 +39,23 @@ vi.mock("../../utils/formatters", () => ({
getSystemLocale: () => "en-US", getSystemLocale: () => "en-US",
})); }));
vi.mock("../../utils/schedule", () => ({ vi.mock("../../utils/schedule", async () => {
buildSchedulePreview: (...args: unknown[]) => mockBuildSchedulePreview(...args), const actual = await vi.importActual<typeof import("../../utils/schedule")>("../../utils/schedule");
calculateCoverage: (...args: unknown[]) => mockCalculateCoverage(...args), return {
computeMissedPastDoseIds: (...args: unknown[]) => mockComputeMissedPastDoseIds(...args), ...actual,
isDoseDismissed: vi.fn(() => false), buildSchedulePreview: (...args: unknown[]) => mockBuildSchedulePreview(...args),
})); calculateCoverage: (...args: unknown[]) => mockCalculateCoverage(...args),
computeMissedPastDoseIds: (...args: unknown[]) => mockComputeMissedPastDoseIds(...args),
isDoseDismissed: vi.fn(() => false),
};
});
const meds: Medication[] = [ const meds: Medication[] = [
{ {
id: 11, id: 11,
name: "Aspirin", name: "Aspirin",
takenBy: ["Max", "Anna"], takenBy: ["Max", "Anna"],
packageType: "blister",
packCount: 1, packCount: 1,
blistersPerPack: 1, blistersPerPack: 1,
pillsPerBlister: 10, pillsPerBlister: 10,
@@ -463,7 +468,7 @@ describe("useAppContext", () => {
all: [ all: [
{ {
name: "Aspirin", name: "Aspirin",
daysLeft: 2, daysLeft: 8,
medsLeft: 5, medsLeft: 5,
depletionTime: Date.now() + 100000, depletionTime: Date.now() + 100000,
}, },

Some files were not shown because too many files have changed in this diff Show More