Skip to content

Commit

Permalink
Pool the server to get the status of all inprogress conversations (#41)
Browse files Browse the repository at this point in the history
  • Loading branch information
harishmohanraj authored Nov 24, 2023
1 parent 56b6833 commit bf960a6
Show file tree
Hide file tree
Showing 6 changed files with 183 additions and 84 deletions.
4 changes: 4 additions & 0 deletions main.wasp
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,9 @@ app chatApp {
("stripe", "11.15.0"),
("markdown-to-jsx", "7.3.2"),
],
webSocket: {
fn: import { webSocketFn } from "@server/webSocket.js"
},
}

/* 💽 Wasp defines DB entities via Prisma Database Models:
Expand Down Expand Up @@ -129,6 +132,7 @@ entity Conversation {=psl
conversation Json
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
status String?
chat Chat? @relation(fields: [chatId], references: [id])
chatId Int?
user User? @relation(fields: [userId], references: [id])
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
-- AlterTable
ALTER TABLE "Conversation" ADD COLUMN "status" TEXT;
25 changes: 13 additions & 12 deletions src/client/components/ConversationWrapper.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -71,18 +71,18 @@ export default function ConversationWrapper() {
{
chatId: Number(id),
},
{ enabled: !!id }
{ enabled: !!id, refetchInterval: 1000 }
);

useEffect(() => {
if (chatContainerRef.current) {
// Todo: remove the below ignore comment
// @ts-ignore
chatContainerRef.current.scrollTop =
// Todo: remove the below ignore comment
// @ts-ignore
chatContainerRef.current.scrollHeight;
}
// if (chatContainerRef.current) {
// // Todo: remove the below ignore comment
// // @ts-ignore
// chatContainerRef.current.scrollTop =
// // Todo: remove the below ignore comment
// // @ts-ignore
// chatContainerRef.current.scrollHeight;
// }
}, [conversations]);

async function callAgent(userQuery: string) {
Expand All @@ -102,20 +102,21 @@ export default function ConversationWrapper() {
// 2. call backend python server to get agent response
setIsLoading(true);
const response = await getAgentResponse({
message: userQuery,
message: payload.conversations,
conv_id: payload.conversation_id,
});
// 3. add agent response as new conversation in the table
const openAIPayload = {
const openAIResponse = {
conversation_id: conversations.id,
conversations: [
...payload.conversations,
// Todo: remove the below ignore comment
// @ts-ignore
...[{ role: "assistant", content: response.content }],
],
...(response.team_status && { status: response.team_status }),
};
await updateConversation(openAIPayload);
await updateConversation(openAIResponse);
setIsLoading(false);
} catch (err: any) {
setIsLoading(false);
Expand Down
166 changes: 94 additions & 72 deletions src/server/actions.ts
Original file line number Diff line number Diff line change
@@ -1,57 +1,66 @@
import fetch from 'node-fetch';
import HttpError from '@wasp/core/HttpError.js';
import fetch from "node-fetch";
import HttpError from "@wasp/core/HttpError.js";
// import type { RelatedObject } from '@wasp/entities';
import type { Chat } from '@wasp/entities';
import type { Conversation } from '@wasp/entities';
import type { GenerateGptResponse, StripePayment, CreateChat, UpdateConversation, GetAgentResponse } from '@wasp/actions/types';
import type { StripePaymentResult, OpenAIResponse } from './types';
import Stripe from 'stripe';
import type { Chat } from "@wasp/entities";
import type { Conversation } from "@wasp/entities";
import type {
GenerateGptResponse,
StripePayment,
CreateChat,
UpdateConversation,
GetAgentResponse,
} from "@wasp/actions/types";
import type { StripePaymentResult, OpenAIResponse } from "./types";
import Stripe from "stripe";

import { ADS_SERVER_URL } from "./config.js";

const stripe = new Stripe(process.env.STRIPE_KEY!, {
apiVersion: '2022-11-15',
apiVersion: "2022-11-15",
});

// WASP_WEB_CLIENT_URL will be set up by Wasp when deploying to production: https://wasp-lang.dev/docs/deploying
const DOMAIN = process.env.WASP_WEB_CLIENT_URL || 'http://localhost:3000';

// Python ADS_SERVER_URL
const ADS_SERVER_URL = process.env.ADS_SERVER_URL || 'http://127.0.0.1:9000';
const DOMAIN = process.env.WASP_WEB_CLIENT_URL || "http://localhost:3000";

export const stripePayment: StripePayment<void, StripePaymentResult> = async (_args, context) => {
export const stripePayment: StripePayment<void, StripePaymentResult> = async (
_args,
context
) => {
if (!context.user) {
throw new HttpError(401);
}

let customer: Stripe.Customer;
const stripeCustomers = await stripe.customers.list({
email: context.user.email!,
});
if (!stripeCustomers.data.length) {
console.log('creating customer');
console.log("creating customer");
customer = await stripe.customers.create({
email: context.user.email!,
});
} else {
console.log('using existing customer');
console.log("using existing customer");
customer = stripeCustomers.data[0];
}

const session: Stripe.Checkout.Session = await stripe.checkout.sessions.create({
line_items: [
{
price: process.env.SUBSCRIPTION_PRICE_ID!,
quantity: 1,
const session: Stripe.Checkout.Session =
await stripe.checkout.sessions.create({
line_items: [
{
price: process.env.SUBSCRIPTION_PRICE_ID!,
quantity: 1,
},
],
mode: "subscription",
success_url: `${DOMAIN}/checkout?success=true`,
cancel_url: `${DOMAIN}/checkout?canceled=true`,
automatic_tax: { enabled: true },
customer_update: {
address: "auto",
},
],
mode: 'subscription',
success_url: `${DOMAIN}/checkout?success=true`,
cancel_url: `${DOMAIN}/checkout?canceled=true`,
automatic_tax: { enabled: true },
customer_update: {
address: 'auto',
},
customer: customer.id,
});
customer: customer.id,
});

await context.entities.User.update({
where: {
Expand All @@ -64,7 +73,7 @@ export const stripePayment: StripePayment<void, StripePaymentResult> = async (_a
});

if (!session) {
throw new HttpError(402, 'Could not create a Stripe session');
throw new HttpError(402, "Could not create a Stripe session");
} else {
return {
sessionUrl: session.url,
Expand Down Expand Up @@ -93,11 +102,11 @@ export const generateGptResponse: GenerateGptResponse<GptPayload> = async (
// engine:"airt-canada-gpt35-turbo-16k",
messages: [
{
role: 'system',
role: "system",
content: instructions,
},
{
role: 'user',
role: "user",
content: command,
},
],
Expand All @@ -119,20 +128,23 @@ export const generateGptResponse: GenerateGptResponse<GptPayload> = async (
// });
// }

console.log('fetching', payload);
console.log("fetching", payload);
// https://api.openai.com/v1/chat/completions
const response = await fetch('https://airt-openai-canada.openai.azure.com/openai/deployments/airt-canada-gpt35-turbo-16k/chat/completions?api-version=2023-07-01-preview', {
headers: {
'Content-Type': 'application/json',
// Authorization: `Bearer ${process.env.AZURE_OPENAI_API_KEY!}`,
'api-key': `${process.env.AZURE_OPENAI_API_KEY!}`,
},
method: 'POST',
body: JSON.stringify(payload),
});
const response = await fetch(
"https://airt-openai-canada.openai.azure.com/openai/deployments/airt-canada-gpt35-turbo-16k/chat/completions?api-version=2023-07-01-preview",
{
headers: {
"Content-Type": "application/json",
// Authorization: `Bearer ${process.env.AZURE_OPENAI_API_KEY!}`,
"api-key": `${process.env.AZURE_OPENAI_API_KEY!}`,
},
method: "POST",
body: JSON.stringify(payload),
}
);

const json = (await response.json()) as OpenAIResponse;
console.log('response json', json);
console.log("response json", json);
// return context.entities.RelatedObject.create({
// data: {
// content: json?.choices[0].message.content,
Expand All @@ -141,7 +153,7 @@ export const generateGptResponse: GenerateGptResponse<GptPayload> = async (
// });
return {
content: json?.choices[0].message.content,
}
};
} catch (error: any) {
if (!context.user.hasPaid && error?.statusCode != 402) {
await context.entities.User.update({
Expand All @@ -156,12 +168,13 @@ export const generateGptResponse: GenerateGptResponse<GptPayload> = async (
console.error(error);
}

throw new HttpError(500, 'Something went wrong');
throw new HttpError(500, "Something went wrong");
};



export const createChat: CreateChat<void, Conversation> = async (_args, context) => {
export const createChat: CreateChat<void, Conversation> = async (
_args,
context
) => {
if (!context.user) {
throw new HttpError(401);
}
Expand All @@ -175,35 +188,40 @@ export const createChat: CreateChat<void, Conversation> = async (_args, context)
data: {
conversation: [
{
role: 'assistant',
content: `Hi! I am Captn and I am here to help you with digital marketing. I can create and optimise marketing campaigns for you. But before I propose any activities, please let me know a little bit about your business and what your marketing goals are.`,
role: "assistant",
content: `Hi! I am Captn and I am here to help you with digital marketing. I can create and optimise marketing campaigns for you. But before I propose any activities, please let me know a little bit about your business and what your marketing goals are.`,
},
],
],
chat: { connect: { id: chat.id } },
user: { connect: { id: context.user.id } },
},
});
}
};

type UpdateConversationPayload = {
conversation_id: number;
conversations: any;
status: string;
};

export const updateConversation: UpdateConversation<UpdateConversationPayload, Conversation> = async (args, context) => {
export const updateConversation: UpdateConversation<
UpdateConversationPayload,
Conversation
> = async (args, context) => {
if (!context.user) {
throw new HttpError(401);
}
return context.entities.Conversation.update({
where: { id: args.conversation_id },
data: {
conversation: args.conversations
conversation: args.conversations,
...(args.status && { status: args.status }),
},
})
}
});
};

type AgentPayload = {
message: string;
message: any;
conv_id: number;
};

Expand All @@ -215,29 +233,33 @@ export const getAgentResponse: GetAgentResponse<AgentPayload> = async (
throw new HttpError(401);
}

const payload = { message: message, conv_id: conv_id, user_id: context.user.id };
console.log("===========")
console.log("Payload to Python server")
console.log(payload)
console.log("===========")
const payload = {
message: message,
conv_id: conv_id,
user_id: context.user.id,
};
console.log("===========");
console.log("Payload to Python server");
console.log(payload);
console.log("===========");
try {
const response = await fetch(`${ADS_SERVER_URL}/chat`, {
method: 'POST',
const response = await fetch(`${ADS_SERVER_URL}/openai/chat`, {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify(payload),
});

const json = await response.json() as { detail?: string }; // Parse JSON once
const json: any = (await response.json()) as { detail?: string }; // Parse JSON once

if (!response.ok) {
const errorMsg = json.detail || `HTTP error with status code ${response.status}`;
console.error('Server Error:', errorMsg);
const errorMsg =
json.detail || `HTTP error with status code ${response.status}`;
console.error("Server Error:", errorMsg);
throw new Error(errorMsg);
}

return { content: json };

return { content: json["content"], team_status: json["team_status"] };
} catch (error: any) {
throw new HttpError(500, 'Something went wrong. Please try again later');
throw new HttpError(500, "Something went wrong. Please try again later");
}
};
2 changes: 2 additions & 0 deletions src/server/config.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export const ADS_SERVER_URL =
process.env.ADS_SERVER_URL || "http://127.0.0.1:9000";
Loading

0 comments on commit bf960a6

Please sign in to comment.