diff --git a/src/pages/api/trpc/[trpc].ts b/src/pages/api/trpc/[trpc].ts index e79d257..b05a5ab 100644 --- a/src/pages/api/trpc/[trpc].ts +++ b/src/pages/api/trpc/[trpc].ts @@ -3,6 +3,8 @@ import { env } from "@/env.mjs"; import { appRouter } from "@/server/api/root"; import { createTRPCContext } from "@/server/api/trpc"; +export const maxDuration = 150; + // export API handler export default createNextApiHandler({ router: appRouter, diff --git a/src/server/api/routers/activities.ts b/src/server/api/routers/activities.ts index 19f9ea3..308e93e 100644 --- a/src/server/api/routers/activities.ts +++ b/src/server/api/routers/activities.ts @@ -25,67 +25,65 @@ export const activityRouter = createTRPCRouter({ getActivites: protectedProcedure .input(z.object({ mood: z.number(), questId: z.string() })) .query(async ({ ctx, input }) => { - try { - const prisma = ctx.prisma; - - // check for cache - const cacheKey = `${ctx.session.user.id}_affirmation_v2` - const cacheData = await kv.get(cacheKey); - - console.log({ cacheData }) // log for texting + try { + const prisma = ctx.prisma; + const cacheKey = `${ctx.session.user.id}_activity`; + if (process.env.VERCEL_ENV === "production") { + // check for cache + const cacheData = await kv.get(cacheKey); + + if (cacheData) { + return cacheData; + } + } - if(cacheData) { - return cacheData - } + // const quest = await ctx.prisma.quest.findUnique({ where: { id: input.questId }, include: { goal: true }}) + const quest = await prisma.quest.findFirst({ + where: { id: input.questId }, + include: { goal: true }, + }); - // const quest = await ctx.prisma.quest.findUnique({ where: { id: input.questId }, include: { goal: true }}) - const quest = await prisma.quest.findFirst({ - where: { id : input.questId }, - include: { goal: true } - }); - - if(!quest){ - throw new Error('invlaid quest') - } - if (!quest.goal)return []; - const goal = (quest.goal).name; - const prompt = ` + if (!quest) { + throw new Error("invlaid quest"); + } + if (!quest.goal) return []; + const goal = quest.goal.name; + const prompt = ` Imagine yourself as a specialist, someone with a mood of ${input.mood} in a scale of 1-10 - Can you recommend three activities for the day that can empower him/her to reach thier goal? + Can you recommend 5 activities for the day that can empower him/her to reach thier goal? These activities should contribute to thier pursuit of ${goal}. Please provide your response in the form of a JavaScript array of objects, with each object containing a title, duration, and a concise description (maximum 150 characters). `.trim(); - - - const rawActivitiyResponse = await queryOpenAi(prompt).catch((err: unknown) => { - console.log({ err }) - }) - - - console.log({ rawActivitiyResponse }) - if (!rawActivitiyResponse) { - throw new Error("Oops something went wrong"); + + const rawActivitiyResponse = await queryOpenAi(prompt).catch( + (err: unknown) => { + console.log({ err }); } - const activities = parseActivities(rawActivitiyResponse); - console.log({ activities }) + ); - // set cache data - - await kv.set(cacheKey, activities, { - ex: 86400 - }) - - return activities; + + if (!rawActivitiyResponse) { + throw new Error("Oops something went wrong"); } - catch (err: unknown){ - console.log(err) - throw err; + const activities = parseActivities(rawActivitiyResponse); + + + // set cache data + if (process.env.VERCEL_ENV === "production") { + await kv.set(cacheKey, activities, { + ex: 86400, + }); } + return activities; + } catch (err: unknown) { + console.log(err); + throw err; + } }), startQuestActivity: protectedProcedure .input( z.object({ type: z.enum(dailyActivityEnum), - id: z.string() + id: z.string(), }) ) .mutation(async ({ input, ctx }) => { @@ -97,24 +95,24 @@ export const activityRouter = createTRPCRouter({ const { type, id } = input; const quest = await prisma.quest.findFirst({ - where: { userId, id }, - include: { goal: true } + where: { userId, id }, + include: { goal: true }, }); if (!quest) { throw new Error("Quest not found"); } - - if (!quest.goal){ - throw new Error('Invalid quest') + + if (!quest.goal) { + throw new Error("Invalid quest"); } - const isChat = type.toLowerCase() === DailyQuestActivity.Chat.toString().toLowerCase(); - if (isChat){ + const isChat = + type.toLowerCase() === DailyQuestActivity.Chat.toString().toLowerCase(); + if (isChat) { // create a new chat ; - const goal = (quest.goal).name; - await createChat(prisma, goal, userId, quest.id) - + const goal = quest.goal.name; + await createChat(prisma, goal, userId, quest.id); } const currentDay = new Date(); // Current date @@ -154,16 +152,13 @@ export const activityRouter = createTRPCRouter({ const currentProfileScore = profile.score + score; - - const level = getLevel(currentProfileScore); - console.log({ level }) + console.log({ level }); if (!level) { throw new Error("Oops try again"); } - // update user score and create activity await prisma.profile.update({ @@ -190,27 +185,31 @@ export const activityRouter = createTRPCRouter({ .input( z.object({ id: z.string(), - chatSessionId: z.string().optional() + chatSessionId: z.string().optional(), }) ) .mutation(async ({ ctx, input }) => { const prisma = ctx.prisma; - const userId = ctx.session.user.id + const userId = ctx.session.user.id; - const profile = await prisma.profile.findFirst({ where: { userId }}); + const profile = await prisma.profile.findFirst({ where: { userId } }); - if(!profile){ - throw new Error('user not found') + if (!profile) { + throw new Error("user not found"); } - const chatId = input.chatSessionId + const chatId = input.chatSessionId; - if (chatId){ - - const session = await prisma.dailyTherapySession.findUnique({ where: { id: chatId }}); - if (!session){ - throw new Error('Invalid chat session') - } - await prisma.dailyTherapySession.update({ where: { id: chatId }, data: { isActive: false }}) + if (chatId) { + const session = await prisma.dailyTherapySession.findUnique({ + where: { id: chatId }, + }); + if (!session) { + throw new Error("Invalid chat session"); + } + await prisma.dailyTherapySession.update({ + where: { id: chatId }, + data: { isActive: false }, + }); } const activity = await prisma.dailyActivity.findFirst({ @@ -228,61 +227,61 @@ export const activityRouter = createTRPCRouter({ throw new Error("Activiy does not exit"); } - const currentScore = 50 + profile.score; const currentLevel = getLevel(currentScore); await prisma.profile.update({ where: { - userId + userId, }, data: { score: currentScore, level: currentLevel, - } - }) - - + }, + }); const currentActivity = await prisma.dailyActivity.update({ where: { - id: input.id + id: input.id, }, data: { isCompleted: true, - }, include: { - quest: true - } - }) + quest: true, + }, + }); return currentActivity; }), - getActiveDailyActivity: protectedProcedure.input(z.object({ - type: z.enum(dailyActivityEnum) - })).query(async ({ ctx, input }) => { - - - - const activity = ctx.prisma.dailyActivity.findFirst({ + getActiveDailyActivity: protectedProcedure + .input( + z.object({ + type: z.enum(dailyActivityEnum), + }) + ) + .query(async ({ ctx, input }) => { + const activity = ctx.prisma.dailyActivity.findFirst({ where: { type: input.type as DailyQuestActivity, isCompleted: false, quest: { userId: ctx.session.user.id, - } + }, }, include: { - quest: true - } - }) - - return activity; + quest: true, + }, + }); + return activity; }), - getActiveStep: protectedProcedure.input(z.object({ - id: z.string() - })).query(async ({ ctx , input}) => { + getActiveStep: protectedProcedure + .input( + z.object({ + id: z.string(), + }) + ) + .query(async ({ ctx, input }) => { const questId = input.id; const currentDay = new Date(); // Current date @@ -299,12 +298,12 @@ export const activityRouter = createTRPCRouter({ createdAt: { gte: currentDayStart, lt: new Date(currentDayStart.getTime() + 24 * 60 * 60 * 1000), - } + }, }, - }) + }); - return { count } - }) + return { count }; + }), }); /** @@ -316,4 +315,3 @@ Zenith: 15001 - 40000 GodMode: 400001 - INFINITY * */ - diff --git a/src/shared-hooks/useEnterSubmit.tsx b/src/shared-hooks/useEnterSubmit.tsx new file mode 100644 index 0000000..0de0b47 --- /dev/null +++ b/src/shared-hooks/useEnterSubmit.tsx @@ -0,0 +1,23 @@ +import { useRef, type RefObject } from 'react' + +export function useEnterSubmit(): { + formRef: RefObject + onKeyDown: (event: React.KeyboardEvent) => void +} { + const formRef = useRef(null) + + const handleKeyDown = ( + event: React.KeyboardEvent + ): void => { + if ( + event.key === 'Enter' && + !event.shiftKey && + !event.nativeEvent.isComposing + ) { + formRef.current?.requestSubmit() + event.preventDefault() + } + } + + return { formRef, onKeyDown: handleKeyDown } +} \ No newline at end of file diff --git a/src/ui/quest/QuestAffrimationWrapper.tsx b/src/ui/quest/QuestAffrimationWrapper.tsx index 96d89fe..670678f 100644 --- a/src/ui/quest/QuestAffrimationWrapper.tsx +++ b/src/ui/quest/QuestAffrimationWrapper.tsx @@ -47,7 +47,7 @@ export const QuestAffrimationWrapper = () => { onClick={goBack} > - + {/* {!hideInfo && +); + +interface ChatMessageProps { + message: any; + index: number; +} + +const ChatMessage: React.FC = ({ message, index }) => ( + + + {message.sender === "user" ? "You" : "Ada"} + + {message.content} + +); + +interface ChatInputProps { + value: string; + onChange: (value: string) => void; + onSubmit: () => Promise; + isLoading: boolean; +} +const ChatInput: React.FC = ({ value, onChange, onSubmit, isLoading }) => { + + const { formRef, onKeyDown } = useEnterSubmit(); + return
{ + e.preventDefault(); + void onSubmit(); + }} + > + +