improved questions
This commit is contained in:
parent
14ccaee48c
commit
7b8ec190a8
7 changed files with 226 additions and 96 deletions
|
|
@ -36,16 +36,16 @@ type BaseQuestion = {
|
|||
|
||||
export type Question =
|
||||
| (BaseQuestion & {
|
||||
type: "choice";
|
||||
options: string[];
|
||||
})
|
||||
type: "choice";
|
||||
options: string[];
|
||||
})
|
||||
| (BaseQuestion & {
|
||||
type: "numeric";
|
||||
range: {
|
||||
min: number;
|
||||
max: number;
|
||||
};
|
||||
});
|
||||
type: "numeric";
|
||||
range: {
|
||||
min: number;
|
||||
max: number;
|
||||
};
|
||||
});
|
||||
|
||||
export type QuizResponse = {
|
||||
playerId: string;
|
||||
|
|
|
|||
|
|
@ -1,10 +1,13 @@
|
|||
import type { Question } from "../party-types";
|
||||
import {
|
||||
buildOptionsWithCorrect,
|
||||
buildOrderedOptions,
|
||||
buildQuestionWindow,
|
||||
getTopArtistName,
|
||||
getTopGenreName,
|
||||
getTopTrackName,
|
||||
getMostSharedGenreNames,
|
||||
getTopClusterArtists,
|
||||
getTopClusterTracks,
|
||||
type PartyAnalytics,
|
||||
pickRandom,
|
||||
} from "./question-utils";
|
||||
|
||||
export function buildAudioMetadataQuestion(
|
||||
|
|
@ -12,34 +15,104 @@ export function buildAudioMetadataQuestion(
|
|||
index: number,
|
||||
): Question {
|
||||
type ChoiceQuestion = Extract<Question, { type: "choice" }>;
|
||||
const questions: Array<Omit<ChoiceQuestion, "startTimestamp" | "endTimestamp">> = [
|
||||
{
|
||||
const questions: Array<
|
||||
Omit<ChoiceQuestion, "startTimestamp" | "endTimestamp">
|
||||
> = [];
|
||||
|
||||
const genreOptions = buildOrderedOptions(
|
||||
getMostSharedGenreNames(analytics),
|
||||
4,
|
||||
);
|
||||
if (genreOptions) {
|
||||
questions.push({
|
||||
type: "choice",
|
||||
text: "Which genre appears most in the party analytics?",
|
||||
options: [getTopGenreName(analytics), "Electronic", "Rock", "Jazz"],
|
||||
options: genreOptions,
|
||||
correct: 0,
|
||||
points: 10,
|
||||
},
|
||||
{
|
||||
type: "choice",
|
||||
text: "Which artist shows up most often in the shared audio data?",
|
||||
options: [
|
||||
getTopArtistName(analytics),
|
||||
"Artist B",
|
||||
"Artist C",
|
||||
"Artist D",
|
||||
],
|
||||
correct: 0,
|
||||
points: 10,
|
||||
},
|
||||
{
|
||||
type: "choice",
|
||||
text: "Which track looks most shared across the party?",
|
||||
options: [getTopTrackName(analytics), "Track B", "Track C", "Track D"],
|
||||
correct: 0,
|
||||
points: 10,
|
||||
},
|
||||
];
|
||||
});
|
||||
}
|
||||
|
||||
const topArtists = getTopClusterArtists(analytics);
|
||||
const topArtist = topArtists[0];
|
||||
if (topArtist) {
|
||||
const artistOptions = buildOptionsWithCorrect(topArtist, topArtists, 4);
|
||||
if (artistOptions) {
|
||||
questions.push({
|
||||
type: "choice",
|
||||
text: "Which artist shows up most often in the shared audio data?",
|
||||
options: artistOptions,
|
||||
correct: 0,
|
||||
points: 10,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
const topTracks = getTopClusterTracks(analytics);
|
||||
if (topTracks.length > 0) {
|
||||
const trackNames = topTracks.map((track) => track.name);
|
||||
const topTrackName = trackNames[0];
|
||||
const trackOptions = topTrackName
|
||||
? buildOptionsWithCorrect(topTrackName, trackNames, 4)
|
||||
: null;
|
||||
if (trackOptions) {
|
||||
questions.push({
|
||||
type: "choice",
|
||||
text: "Which track looks most shared across the party?",
|
||||
options: trackOptions,
|
||||
correct: 0,
|
||||
points: 10,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
const randomTopTrack = pickRandom(topTracks);
|
||||
if (randomTopTrack) {
|
||||
const trackArtists =
|
||||
randomTopTrack.artists?.map((artist) => artist.name) ?? [];
|
||||
const allArtists = topArtists.length > 0 ? topArtists : trackArtists;
|
||||
const correctArtist = trackArtists[0] ?? allArtists[0];
|
||||
if (correctArtist) {
|
||||
const artistOptions = buildOptionsWithCorrect(
|
||||
correctArtist,
|
||||
allArtists,
|
||||
4,
|
||||
);
|
||||
if (artistOptions) {
|
||||
questions.push({
|
||||
type: "choice",
|
||||
text: `Who performs "${randomTopTrack.name}"?`,
|
||||
options: artistOptions,
|
||||
correct: 0,
|
||||
points: 10,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
if (randomTopTrack.albumName) {
|
||||
const albumNames = topTracks
|
||||
.map((track) => track.albumName)
|
||||
.filter((name): name is string => Boolean(name));
|
||||
const albumOptions = buildOptionsWithCorrect(
|
||||
randomTopTrack.albumName,
|
||||
albumNames,
|
||||
4,
|
||||
);
|
||||
if (albumOptions) {
|
||||
questions.push({
|
||||
type: "choice",
|
||||
text: `"${randomTopTrack.name}" appears on which album?`,
|
||||
options: albumOptions,
|
||||
correct: 0,
|
||||
points: 10,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (questions.length === 0) {
|
||||
throw new Error("Question not found");
|
||||
}
|
||||
|
||||
const question = questions[index % questions.length];
|
||||
if (!question) throw new Error("Question not found");
|
||||
|
|
|
|||
|
|
@ -3,7 +3,10 @@ import type { Question } from "../party-types";
|
|||
import type { PartyAnalytics } from "./question-utils";
|
||||
import { buildQuestionWindow, getQuestionRange } from "./question-utils";
|
||||
|
||||
type NumericQuestion = Omit<Extract<Question, { type: "numeric" }>, "startTimestamp" | "endTimestamp">;
|
||||
type NumericQuestion = Omit<
|
||||
Extract<Question, { type: "numeric" }>,
|
||||
"startTimestamp" | "endTimestamp"
|
||||
>;
|
||||
|
||||
type BuildNumericQuestionInput = {
|
||||
db: typeof db;
|
||||
|
|
|
|||
|
|
@ -10,8 +10,13 @@ export type PartyAnalytics = {
|
|||
mostSharedGenres?: { name: string }[];
|
||||
};
|
||||
storyClusters?: {
|
||||
tracks?: { name: string }[];
|
||||
tracks?: {
|
||||
name: string;
|
||||
artists?: { name: string }[];
|
||||
albumName?: string;
|
||||
}[];
|
||||
artists?: { name: string }[];
|
||||
genres?: { name: string }[];
|
||||
}[];
|
||||
memberProfiles?: { userId: string }[];
|
||||
pairwise?: { userIdA: string; userIdB: string }[];
|
||||
|
|
@ -78,16 +83,50 @@ export function getQuestionRange(
|
|||
};
|
||||
}
|
||||
|
||||
export function getTopGenreName(analytics: PartyAnalytics): string {
|
||||
return analytics?.groupSummary?.mostSharedGenres?.[0]?.name ?? "Hip-Hop";
|
||||
export function getMostSharedGenreNames(analytics: PartyAnalytics): string[] {
|
||||
return (analytics?.groupSummary?.mostSharedGenres ?? []).map(
|
||||
(genre) => genre.name,
|
||||
);
|
||||
}
|
||||
|
||||
export function getTopArtistName(analytics: PartyAnalytics): string {
|
||||
return analytics?.storyClusters?.[0]?.artists?.[0]?.name ?? "Artist A";
|
||||
export function getTopClusterArtists(analytics: PartyAnalytics): string[] {
|
||||
return (analytics?.storyClusters?.[0]?.artists ?? []).map(
|
||||
(artist) => artist.name,
|
||||
);
|
||||
}
|
||||
|
||||
export function getTopTrackName(analytics: PartyAnalytics): string {
|
||||
return analytics?.storyClusters?.[0]?.tracks?.[0]?.name ?? "Track A";
|
||||
export function getTopClusterTracks(
|
||||
analytics: PartyAnalytics,
|
||||
): Array<{ name: string; artists?: { name: string }[]; albumName?: string }> {
|
||||
return analytics?.storyClusters?.[0]?.tracks ?? [];
|
||||
}
|
||||
|
||||
export function buildOrderedOptions(
|
||||
values: Array<string | undefined>,
|
||||
desiredCount: number,
|
||||
): string[] | null {
|
||||
const options = uniqueStrings(
|
||||
values.filter((value): value is string => Boolean(value)),
|
||||
);
|
||||
return options.length >= desiredCount ? options.slice(0, desiredCount) : null;
|
||||
}
|
||||
|
||||
export function buildOptionsWithCorrect(
|
||||
correct: string,
|
||||
candidates: string[],
|
||||
desiredCount: number,
|
||||
): string[] | null {
|
||||
const options = uniqueStrings([
|
||||
correct,
|
||||
...candidates.filter((c) => c !== correct),
|
||||
]);
|
||||
return options.length >= desiredCount ? options.slice(0, desiredCount) : null;
|
||||
}
|
||||
|
||||
export function pickRandom<T>(items: T[]): T | null {
|
||||
if (items.length === 0) return null;
|
||||
const index = Math.floor(Math.random() * items.length);
|
||||
return items[index] ?? null;
|
||||
}
|
||||
|
||||
export function getCurrentLeader(
|
||||
|
|
|
|||
|
|
@ -20,7 +20,9 @@ export function buildSocialQuestion(
|
|||
const diverse = getMostDiverseMember(analytics, members);
|
||||
const aligned = getMostAlignedMember(analytics, members);
|
||||
|
||||
const questions: Array<Omit<ChoiceQuestion, "startTimestamp" | "endTimestamp">> = [
|
||||
const questions: Array<
|
||||
Omit<ChoiceQuestion, "startTimestamp" | "endTimestamp">
|
||||
> = [
|
||||
{
|
||||
type: "choice",
|
||||
text: "Who is leading the quiz right now?",
|
||||
|
|
|
|||
|
|
@ -1,44 +1,47 @@
|
|||
import Elysia from "elysia"
|
||||
import { broadcastQuizState, partyTopic, socketPartyId, userTopic } from "./party-socket"
|
||||
import { getMemberRecord, getPartyStatus } from "../party-data"
|
||||
import { db } from "../db"
|
||||
import Elysia from "elysia";
|
||||
import { db } from "../db";
|
||||
import { getMemberRecord, getPartyStatus } from "../party-data";
|
||||
import {
|
||||
broadcastQuizState,
|
||||
partyTopic,
|
||||
socketPartyId,
|
||||
userTopic,
|
||||
} from "./party-socket";
|
||||
|
||||
export const partySocketApp = new Elysia()
|
||||
.group("/dev-socket", (app) =>
|
||||
app
|
||||
.get("/test", () => ({ ok: 1 }))
|
||||
.ws("/ws", {
|
||||
async open(ws) {
|
||||
let id = "zzxWcTUntIWTHkX8atEOv7Neiu7XEz9t"
|
||||
ws.subscribe(userTopic(id))
|
||||
const membership = await getMemberRecord(db, id);
|
||||
if (!membership) {
|
||||
ws.send(
|
||||
JSON.stringify({
|
||||
type: "snapshot",
|
||||
party: null,
|
||||
members: [],
|
||||
}),
|
||||
);
|
||||
return;
|
||||
}
|
||||
export const partySocketApp = new Elysia().group("/dev-socket", (app) =>
|
||||
app
|
||||
.get("/test", () => ({ ok: 1 }))
|
||||
.ws("/ws", {
|
||||
async open(ws) {
|
||||
const id = "zzxWcTUntIWTHkX8atEOv7Neiu7XEz9t";
|
||||
ws.subscribe(userTopic(id));
|
||||
const membership = await getMemberRecord(db, id);
|
||||
if (!membership) {
|
||||
ws.send(
|
||||
JSON.stringify({
|
||||
type: "snapshot",
|
||||
party: null,
|
||||
members: [],
|
||||
}),
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
socketPartyId.set(ws, membership.partyId);
|
||||
ws.subscribe(partyTopic(membership.partyId));
|
||||
socketPartyId.set(ws, membership.partyId);
|
||||
ws.subscribe(partyTopic(membership.partyId));
|
||||
|
||||
const snapshot = await getPartyStatus(membership.partyId);
|
||||
if (snapshot) {
|
||||
ws.send(
|
||||
JSON.stringify({
|
||||
type: "snapshot",
|
||||
party: snapshot.party,
|
||||
members: snapshot.members,
|
||||
}),
|
||||
);
|
||||
const snapshot = await getPartyStatus(membership.partyId);
|
||||
if (snapshot) {
|
||||
ws.send(
|
||||
JSON.stringify({
|
||||
type: "snapshot",
|
||||
party: snapshot.party,
|
||||
members: snapshot.members,
|
||||
}),
|
||||
);
|
||||
|
||||
await broadcastQuizState(ws, membership.partyId);
|
||||
}
|
||||
}
|
||||
|
||||
})
|
||||
)
|
||||
await broadcastQuizState(ws, membership.partyId);
|
||||
}
|
||||
},
|
||||
}),
|
||||
);
|
||||
|
|
|
|||
|
|
@ -5,8 +5,13 @@ import { partyMember } from "../db/schema";
|
|||
import { generatePartyQuestion } from "../party/question-generator";
|
||||
import type { PartyAnalytics } from "../party/question-utils";
|
||||
import { updatePartyData } from "../party/state";
|
||||
import type { Question, QuizResponse, QuizRound, QuizState } from "../party-types";
|
||||
import { partyAnalysisWorkflow, PartyAnalysisWorkflow } from "./party-analysis";
|
||||
import type {
|
||||
Question,
|
||||
QuizResponse,
|
||||
QuizRound,
|
||||
QuizState,
|
||||
} from "../party-types";
|
||||
import { partyAnalysisWorkflow } from "./party-analysis";
|
||||
|
||||
const TOTAL_QUESTIONS = 5;
|
||||
|
||||
|
|
@ -35,9 +40,9 @@ export class QuizWorkflow extends ConfiguredInstance {
|
|||
answers: {},
|
||||
scores: {},
|
||||
history: [],
|
||||
};
|
||||
};
|
||||
|
||||
await partyAnalysisWorkflow.analyzeParty(partyId)
|
||||
await partyAnalysisWorkflow.analyzeParty(partyId);
|
||||
|
||||
// Initialize quiz state
|
||||
await QuizWorkflow.updatePartyData(partyId, quizState);
|
||||
|
|
@ -76,10 +81,10 @@ export class QuizWorkflow extends ConfiguredInstance {
|
|||
deadlineEpochMS: question.endTimestamp,
|
||||
});
|
||||
|
||||
if (response === null) {
|
||||
if (response === null) {
|
||||
// Timeout - fill in missing players with no answer
|
||||
const now = Date.now();
|
||||
if (now < question.endTimestamp) continue;
|
||||
if (now < question.endTimestamp) continue;
|
||||
for (const memberId of memberIds) {
|
||||
if (!receivedPlayers.has(memberId)) {
|
||||
receivedPlayers.add(memberId);
|
||||
|
|
@ -116,8 +121,7 @@ export class QuizWorkflow extends ConfiguredInstance {
|
|||
}
|
||||
|
||||
for (const [playerId, gained] of QuizWorkflow.scoreRound(round)) {
|
||||
quizState.scores[playerId] =
|
||||
(quizState.scores[playerId] ?? 0) + gained;
|
||||
quizState.scores[playerId] = (quizState.scores[playerId] ?? 0) + gained;
|
||||
}
|
||||
|
||||
await QuizWorkflow.updatePartyData(partyId, quizState);
|
||||
|
|
@ -169,7 +173,10 @@ export class QuizWorkflow extends ConfiguredInstance {
|
|||
const ordered = round.responses
|
||||
.map((response) => ({
|
||||
response,
|
||||
distance: Math.abs((response.selectedValue ?? response.selected) - round.question.correct),
|
||||
distance: Math.abs(
|
||||
(response.selectedValue ?? response.selected) -
|
||||
round.question.correct,
|
||||
),
|
||||
}))
|
||||
.sort((a, b) => a.distance - b.distance);
|
||||
|
||||
|
|
@ -183,9 +190,12 @@ export class QuizWorkflow extends ConfiguredInstance {
|
|||
}
|
||||
}
|
||||
|
||||
const scoringGroups = groups.slice(0, Math.max(0, groups.length - 1));
|
||||
const _scoringGroups = groups.slice(0, Math.max(0, groups.length - 1));
|
||||
if (groups.length <= 1) {
|
||||
return round.responses.map((response) => [response.playerId, round.question.points]);
|
||||
return round.responses.map((response) => [
|
||||
response.playerId,
|
||||
round.question.points,
|
||||
]);
|
||||
}
|
||||
|
||||
return groups.flatMap((group, index) => {
|
||||
|
|
|
|||
Loading…
Reference in a new issue