diff options
| author | yyamashita <yyamashita@mosquit.one> | 2026-05-15 00:16:27 +0900 |
|---|---|---|
| committer | yyamashita <yyamashita@mosquit.one> | 2026-05-15 00:16:27 +0900 |
| commit | b29876cf6f865182ef6958aaf63d91d43dd08d55 (patch) | |
| tree | 381808b8593febbde5abdcb320d7eba8915faa65 /app/routes/api-band-detail.tsx | |
| parent | 1a8a159a71fd018bab0e14d9df952c53ae5ea062 (diff) | |
Fix nav classes, add detail APIs, add update-band-info skill
- Fix nav: apply .nav class and .nav-brand to logo (styles were not rendering)
- Add GET+PATCH /api/bands/:uuid and /api/artists/:uuid endpoints
- PATCH supports append_links / append_members for non-destructive updates
- Add /update-band-info skill: fetch SNS profiles and update band description/members
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
Diffstat (limited to 'app/routes/api-band-detail.tsx')
| -rw-r--r-- | app/routes/api-band-detail.tsx | 69 |
1 files changed, 69 insertions, 0 deletions
diff --git a/app/routes/api-band-detail.tsx b/app/routes/api-band-detail.tsx new file mode 100644 index 0000000..aa3b5bb --- /dev/null +++ b/app/routes/api-band-detail.tsx @@ -0,0 +1,69 @@ +import type { ActionFunctionArgs, LoaderFunctionArgs } from "react-router"; +import { + getBandById, + getBandLinks, + getBandMembers, + getIpAddress, + toSlug, + updateBand, + type MemberInput, +} from "~/lib/db.server"; + +export function loader({ params }: LoaderFunctionArgs) { + const band = getBandById(params.uuid!); + if (!band) return Response.json({ error: "Not found" }, { status: 404 }); + const links = getBandLinks(band.id); + const members = getBandMembers(band.id); + return Response.json({ ...band, links, members }); +} + +export async function action({ request, params }: ActionFunctionArgs) { + if (request.method !== "PATCH") { + return Response.json({ error: "Method not allowed" }, { status: 405 }); + } + + const band = getBandById(params.uuid!); + if (!band) return Response.json({ error: "Not found" }, { status: 404 }); + + let body: Record<string, unknown>; + try { + body = await request.json(); + } catch { + return Response.json({ error: "Invalid JSON body" }, { status: 400 }); + } + + const currentLinks = getBandLinks(band.id); + const currentMembers = getBandMembers(band.id); + + const patchLinks = (body.links as { label: string; url: string }[] | undefined) ?? []; + const patchMembers = (body.members as MemberInput[] | undefined) ?? []; + const appendLinks = body.append_links !== false; + const appendMembers = body.append_members !== false; + + const existingUrls = new Set(currentLinks.map((l) => l.url)); + const newLinks = appendLinks + ? [...currentLinks.map((l) => ({ label: l.label, url: l.url })), ...patchLinks.filter((l) => !existingUrls.has(l.url))] + : patchLinks; + + const existingArtistIds = new Set(currentMembers.map((m) => m.artist_id)); + const newMembers: MemberInput[] = appendMembers + ? [ + ...currentMembers.map((m) => ({ artist_id: m.artist_id, role: m.role, since: m.since ?? "", until: m.until ?? "", note: m.note ?? "" })), + ...patchMembers.filter((m) => !existingArtistIds.has(m.artist_id)), + ] + : patchMembers; + + updateBand(band.id, { + slug: (body.slug as string | undefined) ?? band.slug, + name: (body.name as string | undefined) ?? band.name, + area: (body.area as string | undefined) ?? band.area, + description: "description" in body ? (body.description as string | null) : band.description, + status: (body.status as string | undefined) ?? band.status, + links: newLinks, + members: newMembers, + message: (body.message as string | undefined) || "API update", + ip_address: getIpAddress(request), + }); + + return Response.json(getBandById(band.id)); +} |
