Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix/link translation #23

Open
wants to merge 5 commits into
base: main
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions prisma/migrations/20240220044932_init/migration.sql
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
-- DropIndex
DROP INDEX "conversation_userId_flowId_key";
3 changes: 1 addition & 2 deletions prisma/schema.prisma
Original file line number Diff line number Diff line change
@@ -13,7 +13,7 @@ datasource db {
}

model conversation {
id String @default(uuid()) @db.Uuid
id String @db.Uuid
createdAt DateTime @default(now()) @db.Timestamptz(3)
updatedAt DateTime @updatedAt
userId String @db.Uuid
@@ -22,7 +22,6 @@ model conversation {
flowId String
feedback feedback?
@@unique([id])
@@unique([userId,flowId])
}

model User {
61 changes: 40 additions & 21 deletions src/app.controller.ts
Original file line number Diff line number Diff line change
@@ -85,13 +85,21 @@ export class AppController {
let startTime = Date.now()
//get userId from headers
const userId = headers["user-id"]
const sessionId = headers["session-id"]
console.log("userId =",userId)
console.log("sessionId =", sessionId)
if(!userId){
return {
"text":"",
"error": "'user-id' should not be empty"
}
}
if(!sessionId) {
return {
"text": "",
"error": "'session-id' should not be empty"
}
}
let messageType = 'intermediate_response'
//setup loggers
let verboseLogger = this.logger.logWithCustomFields({
@@ -152,16 +160,40 @@ export class AppController {
}
})
}
let conversation = await this.conversationService.getConversationState(
userId,
configid
)

//input setup
let prompt: Prompt = {
input: promptDto
}
let userInput = promptDto.text;
let type = "text"

let defaultContext = {
sessionId,
userId,
userQuestion:'',
query: '',
queryType: '',
response: '',
userAadhaarNumber: user.identifier && configid=='3' ? user.identifier : '',
otp: '',
error: '',
currentState: "getUserQuestion",
type: '',
inputType: type,
inputLanguage: prompt.inputLanguage,
lastAadhaarDigits:'',
state:'onGoing',
isOTPVerified: false
}

let conversation = await this.conversationService.getConversationState(
sessionId,
userId,
defaultContext,
configid
)

//handle text and audio
if(promptDto.text){
type = "Text"
@@ -342,6 +374,8 @@ export class AppController {
}
}

conversation.inputType = type;

//get flow
let botFlowMachine;
switch(configid){
@@ -358,23 +392,7 @@ export class AppController {
botFlowMachine = this.promptService.getXstateMachine("botFlowMachine3")
}

let defaultContext = {
userId,
userQuestion:'',
query: '',
queryType: '',
response: '',
userAadhaarNumber: user.identifier && configid=='3' ? user.identifier : '',
otp: '',
error: '',
currentState: "getUserQuestion",
type: '',
inputType: type,
inputLanguage: prompt.inputLanguage,
lastAadhaarDigits:'',
state:'onGoing',
isOTPVerified: false
}


let botFlowService = interpret(botFlowMachine.withContext(conversation || defaultContext)).start();
// verboseLogger("current state when API hit =", botFlowService.state.context.currentState)
@@ -1085,6 +1103,7 @@ export class AppController {
}

conversation = await this.conversationService.saveConversation(
sessionId,
userId,
botFlowService.getSnapshot().context,
botFlowService.state.context.state,
16 changes: 11 additions & 5 deletions src/modules/aiTools/ai-tools.service.ts
Original file line number Diff line number Diff line change
@@ -81,6 +81,11 @@ export class AiToolsService {
text: string
) {
try {
const urlRegex = /(https?:\/\/[^\s]+)|(www\.[^\s]+)/g;
const urls = text.match(urlRegex) || [];

const placeHolder = "9kBjf325" //placeholder which stays the same across languages after translation
const textWithoutUrls = text.replace(urlRegex, placeHolder)
let config = {
"language": {
"sourceLanguage": source,
@@ -89,7 +94,7 @@ export class AiToolsService {
}
let bhashiniConfig: any = await this.getBhashiniConfig('translation',config)

let textArray = text.split("\n")
let textArray = textWithoutUrls.split("\n")
for(let i=0;i<textArray.length;i++){
let response: any = await this.computeBhashini(
bhashiniConfig?.pipelineInferenceAPIEndPoint?.inferenceApiKey?.value,
@@ -112,8 +117,9 @@ export class AiToolsService {
}
textArray[i]=response?.pipelineResponse[0]?.output[0]?.target
}
const translatedText = textArray.join('\n').replace(new RegExp(placeHolder, 'g'), () => urls.shift() || '');
return {
text: textArray.join('\n'),
text: translatedText,
error: null
}
} catch(error){
@@ -254,18 +260,18 @@ export class AiToolsService {
}
}

async getResponseViaWadhwani(text: string) {
async getResponseViaWadhwani(sessionId: string, userId: string, text: string) {
try{
var myHeaders = new Headers();
myHeaders.append("accept", "application/json");
myHeaders.append("X-API-Key", this.configService.get("WADHWANI_API_KEY"));
let response: any = await fetch(`${this.configService.get("WADHWANI_BASE_URL")}/get_bot_response?query=${text}`, {
let response: any = await fetch(`${this.configService.get("WADHWANI_BASE_URL")}/get_bot_response?query=${text}&user_id=${userId}&session_id=${sessionId}`, {
headers: myHeaders,
"method": "GET",
"mode": "cors",
"credentials": "omit"
});
response = (await response.text()).replace(/^\"|\"$/g, '')
response = await response.json()
return response
} catch(error){
console.log(error)
35 changes: 29 additions & 6 deletions src/modules/conversation/conversation.service.ts
Original file line number Diff line number Diff line change
@@ -16,14 +16,16 @@ export class ConversationService {
}

async saveConversation(
sessionId: string,
userId: string,
context: any,
state: string,
flowId: string
): Promise<any> {
return await this.prisma.conversation.upsert({
where: { userId_flowId: {userId, flowId} },
where: { id: sessionId },
create: {
id: sessionId,
userId,
context,
state,
@@ -34,20 +36,41 @@ export class ConversationService {
}

async getConversationState(
sessionId: string,
userId: string,
flowId: string
defaultContext: any,
flowId: string,
): Promise<any | null> {
const conversation: any = await this.prisma.conversation.findFirst({
let conversation: any = await this.prisma.conversation.findFirst({
where: {
userId,
flowId,
state: 'onGoing'
id: sessionId
},
});

if(!conversation) {
conversation = await this.prisma.conversation.create({
data: {
id: sessionId,
userId,
context: defaultContext,
flowId,
state: 'onGoing'
}
})
}

return conversation?.context ? {...conversation.context, id:conversation.id} : null;
}

async getConversationById(sessionId: string): Promise<any | null> {
const conversation: any = await this.prisma.conversation.findFirst({
where: {
id: sessionId
}
});
return conversation?.id ? conversation : null;
}

async createOrUpdateFeedback(
feedback: any
): Promise<feedback> {
4 changes: 2 additions & 2 deletions src/xstate/prompt/prompt.gaurds.ts
Original file line number Diff line number Diff line change
@@ -20,8 +20,8 @@ export const promptGuards = {

ifOTPHasBeenVerified: (context,_) => context.isOTPVerified,

ifInvalidClassifier: (_,event) => event.data == "invalid",
ifInvalidClassifier: (_,event) => event.data.class == "invalid",

ifConvoStarterOrEnder: (_,event) => event.data == "convo"
ifConvoStarterOrEnder: (_,event) => event.data.class == "convo"

}
6 changes: 4 additions & 2 deletions src/xstate/prompt/prompt.machine.ts
Original file line number Diff line number Diff line change
@@ -878,19 +878,21 @@ export const botFlowMachine3:any =
onDone: [
{
cond: "ifConvoStarterOrEnder",
target: "wadhwaniClassifier",
target: "endFlow",
actions: [
assign({
queryType: (_,event) => event.data,
response: (_,event) => event.data.response,
isWadhwaniResponse: "true"
})
]
},
{
cond: "ifInvalidClassifier",
target: "wadhwaniClassifier",
target: "endFlow",
actions: [
assign({
response: (_,event) => event.data.response,
isWadhwaniResponse: "true"
})
]
10 changes: 7 additions & 3 deletions src/xstate/prompt/prompt.service.ts
Original file line number Diff line number Diff line change
@@ -34,14 +34,18 @@ export class PromptServices {

async questionClassifier (context) {
try{
let response: any = await this.aiToolsService.textClassification(context.query)
let response: any = await this.aiToolsService.getResponseViaWadhwani(context.sessionId, context.userId, context.query)
if (response.error) throw new Error(`${response.error}, please try again.`)
if (response == `"Invalid"`) return "convo"
if (response == `"convo_starter"`) return "convo"
if (response == `"convo_ender"`) return "convo"
if (response == `"Installment Not Received"`) return "payment"
else {
return "invalid"
intent = "invalid"
}
return {
class: intent,
response: response.response
}
} catch (error){
return Promise.reject(error)
@@ -198,7 +202,7 @@ export class PromptServices {

async wadhwaniClassifier (context) {
try{
let response: any = await this.aiToolsService.getResponseViaWadhwani(context.query)
let response: any = await this.aiToolsService.getResponseViaWadhwani(context.sessionId, context.userId,context.query)
if (response.error) throw new Error(`${response.error}, please try again.`)
return response;
} catch (error){
Loading