New POST /api/profiles/{name}/feedback accepts explicit vibe text and
records feedback against a named profile. GET history endpoint added too.
Scoring now filters feedback by profile_name for profile-aware playlists.
Migration 005 adds profile_name column and makes playlist_id nullable.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Background poller monitors HA media_player state during playlist sessions.
When a track transition occurs and the previous track was played < 40% of
its duration, automatically records "skip" feedback. Also includes the
previously uncommitted delete_feedback endpoint.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
The vibe is the core concept of vibe-aware playlists but wasn't
visible anywhere on the dashboard. Now each recent playlist shows
its vibe text alongside track count and age.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Adds feedback API endpoints that record up/down/skip signals tied to
vibe context (CLAP embeddings + text). Dashboard now shows Feedback
Activity (recent events with signal counts) and Vibe Influence (how
the same track gets rated differently across vibes).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Add recent listens, profiles, taste profiles, and recent playlists
to the status page. Two-column responsive grid layout with progress
bar for embeddings and relative timestamps throughout.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Named profiles allow each household member to get personalized
recommendations without polluting each other's taste. Includes
profile CRUD API, speaker→profile auto-attribution, recent listen
history endpoint, and profile param on all existing endpoints.
All endpoints backward compatible (no profile param = "default").
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Blend taste profile with text-embedded mood descriptions (e.g. "chill
ambient lo-fi") using pre-blended vector search against the existing
HNSW index. New optional `vibe` and `alpha` params on playlist generate
and recommendations endpoints. Backward compatible — no vibe = pure
taste profile (alpha=1.0).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
The taste profile is just a weighted average of 512-dim vectors — trivially
cheap even with thousands of tracks. Rebuilding on every listen event keeps
recommendations always up to date.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Full documentation covering architecture, deployment, API endpoints,
speaker entity mapping, pipeline stages, and how recommendations
improve over time. Fixed stale speaker entity reference.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
When a track plays, multiple HA entities (Cast, WiFi, MA) all fire
the automation simultaneously, creating 3x duplicate listen events.
Now skips logging if the same track was recorded within the last 60s.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
The play_playlist_on_speaker function was sending text search queries
to raw Cast entities which can't resolve them. Now uses enqueue: "replace"
for the first track and "add" for subsequent tracks. Added 1s delay between
requests so MA can process each Apple Music search. Increased HTTP timeout
to 30s for search latency.
The caller must pass a Music Assistant entity (_2 suffix) for text-based
search to work.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
profile.embedding was being passed as str(numpy_array) which produces
scientific notation format. pgvector needs [n1,n2,...] format. Now
explicitly formats as comma-separated float list.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
In transformers 5.x, ClapModel.get_audio_features() returns a
BaseModelOutputWithPooling instead of a raw tensor. The 512-dim
embedding is in .pooler_output[0], not directly indexed. Added
backward-compatible extraction with hasattr check.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Two issues:
1. CLAP model output needed .flatten() to produce a 1-D vector for
pgvector. Without it, the nested array caused "expected ndim to be 1".
2. Worker now uses a fresh session per track instead of sharing one
across a batch, preventing PendingRollbackError cascading from one
failure to the next.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
The except clause wasn't binding the exception to a variable, so
str(Exception) stored the class name "<class 'Exception'>" instead of
the actual error message. Now properly captures `as e` and stores str(e).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
SQLAlchemy relationships require ForeignKey on the column definitions,
not just in the migration. Without them, mapper initialization fails.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
The pyproject.toml references the package source, so src/ must be
present when pip resolves metadata.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Full music recommendation pipeline: listening history capture via webhook,
Last.fm candidate discovery, iTunes preview download, CLAP audio embeddings
(512-dim), pgvector cosine similarity recommendations, playlist generation
with known/new track interleaving, and Music Assistant playback via HA.
Includes: FastAPI app, SQLAlchemy models, Alembic migrations, Docker Compose
with pgvector/pg17, status dashboard, and all API endpoints.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>