Add Generative AI to a JavaScript Web App
Learn how to integrate genAI with minimal code changes using JS, Cognito credentials to invoke the Amazon Bedrock API in a React single page app.
How Does This Application Work?
An instance of a Large Language Model
First: Chat With Amazon Bedrock
Second: Knowledge Bases for Amazon Bedrock
Let's Deploy React Generative AI Application With Amazon Bedrock and AWS Javascript SDK
Step 1 - Enable AWS Amplify Hosting:
Step 2 - Access to the App URL:
Let's Try React Generative AI Application With Amazon Bedrock Javascript SDK
🚨🚨🚨🚨🚨 This blog is outdated, check the new version Add Generative AI to a JavaScript Web App 2.0 🚨🚨🚨🚨🚨🚨
data:image/s3,"s3://crabby-images/aedb8/aedb86802ae1fcb38686f0a08babd14a9ae31e38" alt="Authentication Authentication"
1
2
3
4
5
6
7
8
9
10
11
12
{ policyName: "amplify-permissions-custom-resources",
policyDocument: {
Version: "2012-10-17",
Statement: [
{
Resource: "*",
Action: ["bedrock:InvokeModel*", "bedrock:List*", "bedrock:Retrieve*"],
Effect: "Allow",
}
]
}
}
- Chat with Amazon Bedrock
- Knowledge Bases for Amazon Bedrock
data:image/s3,"s3://crabby-images/9bd56/9bd5637e6be52ba300f1a6ee875f05522bf3ca5e" alt="Demos Menu Demos menu"
1
2
import { BedrockAgentClient} from "@aws-sdk/client-bedrock-agent"
import { BedrockAgentRuntimeClient} from "@aws-sdk/client-bedrock-agent-runtime"
1
2
3
4
5
6
7
8
9
10
11
12
13
export const getModel = async () => {
const session = await fetchAuthSession(); //Amplify helper to fetch current logged in user
const model = new Bedrock({
model: "anthropic.claude-instant-v1", // model-id you can try others if you want
region: "us-east-1", // app region
streaming: true, // this enables to get the response in streaming manner
credentials: session.credentials, // the user credentials that allows to invoke bedrock service
// try to limit to 1000 tokens for generation
// temperature = 1 means more creative and freedom
modelKwargs: { max_tokens_to_sample: 1000, temperature: 1 },
});
return model;
};
data:image/s3,"s3://crabby-images/8d60a/8d60af9bcb095377a9f04bb09e0b99899440827c" alt="Chat Q&A Chat Q&A"
data:image/s3,"s3://crabby-images/ce1d3/ce1d3dd0d0b530f7b070d3b72b785eae117b8313" alt="Chat Q&A Chat Q&A"
data:image/s3,"s3://crabby-images/0c143/0c1435fa7447dcc3ab056d5a8b3002510ab8c454" alt="Chat with Memory Chat with Memory"
1
2
3
4
5
6
import { Bedrock } from "@langchain/community/llms/bedrock/web";
import { ConversationChain} from "langchain/chains";
import { BufferMemory } from "langchain/memory";
// create a memory object
const memory = new BufferMemory({ humanPrefix: "H", memoryKey:"chat_history"});
chat_history
key in the memory to get past dialogs, hence you use that key as memoryKey in BufferMemory.data:image/s3,"s3://crabby-images/313ad/313adc92f2fbd0ec9950461b6a32cd7644accb86" alt="Knowledge Bases for Amazon Bedrock Knowledge Bases for Amazon Bedrock"
data:image/s3,"s3://crabby-images/8d04f/8d04f3a830b9573c94a22cc685a124223993ccf8" alt="Amazon Bedrock Retrieve => LLM Amazon Bedrock Retrieve => LLM"
1
2
3
4
5
6
7
8
9
import { ListKnowledgeBasesCommand } from "@aws-sdk/client-bedrock-agent"
export const getBedrockKnowledgeBases = async () => {
const session = await fetchAuthSession()
const client = new BedrockAgentClient({ region: "us-east-1", credentials: session.credentials })
const command = new ListKnowledgeBasesCommand({})
const response = await client.send(command)
return response.knowledgeBaseSummaries
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
import { AmazonKnowledgeBaseRetriever } from "@langchain/community/retrievers/amazon_knowledge_base";
export const getBedrockKnowledgeBaseRetriever = async (knowledgeBaseId) => {
const session = await fetchAuthSession();
const retriever = new AmazonKnowledgeBaseRetriever({
topK: 10, // return top 10 documents
knowledgeBaseId: knowledgeBaseId,
region: "us-east-1",
clientOptions: { credentials: session.credentials }
})
return retriever
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
import { ConversationalRetrievalQAChain } from "langchain/chains";
export const getConversationalRetrievalQAChain = async (llm, retriever, memory) => {
const chain = ConversationalRetrievalQAChain.fromLLM(
llm, retriever = retriever)
chain.memory = memory
//Here you modify the default prompt to add the Human prefix and Assistant suffix needed by Claude.
//otherwise you get an exception
//this is the prompt that uses chat history and last question to formulate a complete standalone question
chain.questionGeneratorChain.prompt.template = "Human: " + chain.questionGeneratorChain.prompt.template +"\nAssistant:"
// Here you finally answer the question using the retrieved documents.
chain.combineDocumentsChain.llmChain.prompt.template = `Human: Use the following pieces of context to answer the question at the end. If you don't know the answer, just say that you don't know, don't try to make up an answer.
{context}
Question: {question}
Helpful Answer:
Assistant:`
return chain
}
data:image/s3,"s3://crabby-images/2e180/2e1808f345adf774e23eaf44ea3fe126ef7b7594" alt="Amazon Bedrock Retrieve & Generate Amazon Bedrock Retrieve & Generate"
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
import { BedrockAgentRuntimeClient, RetrieveAndGenerateCommand } from "@aws-sdk/client-bedrock-agent-runtime"
export const ragBedrockKnowledgeBase = async (sessionId, knowledgeBaseId, query) => {
const session = await fetchAuthSession()
const client = new BedrockAgentRuntimeClient({ region: "us-east-1", credentials: session.credentials });
const input = {
input: { text: query }, // user question
retrieveAndGenerateConfiguration: {
type: "KNOWLEDGE_BASE",
knowledgeBaseConfiguration: {
knowledgeBaseId: knowledgeBaseId,
//your existing KnowledgeBase in the same region/ account
// Arn of a Bedrock model, in this case we jump to claude 2.1, the latest. Feel free to use another
modelArn: "arn:aws:bedrock:us-east-1::foundation-model/anthropic.claude-v2:1", // Arn of a Bedrock model
},
}
}
if (sessionId) {
// you can pass the sessionId to continue a dialog.
input.sessionId = sessionId
}
const command = new RetrieveAndGenerateCommand(input);
const response = await client.send(command)
return response
}
- first fork this repo:
1
https://github.com/build-on-aws/building-reactjs-gen-ai-apps-with-amazon-bedrock-javascript-sdk/forks
- Create a New branch:
dev-branch
. - Then follow the steps in Getting started with existing code guide.
- In Step 1 Add repository branch, select main branch and Connecting a monorepo? Pick a folder and enter
reactjs-gen-ai-apps
as a root directory.Add repository branch - For the next Step, Build settings, select
building-a-gen-ai-gen-ai-personal-assistant-reactjs-apps(this app)
as App name, in Enviroment select Create a new envitoment and writedev.
data:image/s3,"s3://crabby-images/1ec9a/1ec9aab9c1761ecfc42bed39aea2ba6a4177ebaa" alt="App build and test settings App build and test settings"
- If there is no existing role, create a new one to service Amplify.
- Deploy your app.
data:image/s3,"s3://crabby-images/bc13e/bc13e507dea5086de58277f6e3e509dbb9d53964" alt="Amplify Deploy Amplify Deploy"
data:image/s3,"s3://crabby-images/52926/529263c594b01d0fd1a5e22163779bdd99455caa" alt="Sing In Window Sing In Window"
data:image/s3,"s3://crabby-images/2333f/2333f573b1596620438d23d3c6975adf4709d677" alt="Backend environments Backend environments"
data:image/s3,"s3://crabby-images/5e6cd/5e6cd2f546a7251bcf48ec736dda5ac5f29aea29" alt="View in Cognito View in Cognito"
hideSignUp: false
in App.jsx, but this can introduce a security flaw by giving anyone access to it.Any opinions in this post are those of the individual author and may not reflect the opinions of AWS.