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

Add conversation search documentation #5018

Merged
merged 15 commits into from
Sep 19, 2023
367 changes: 367 additions & 0 deletions _ml-commons-plugin/conversational-search.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,367 @@
---
layout: default
title: Conversational search
has_children: false
nav_order: 200
---

# Conversational search

Conversational search is an experimental machine learning (ML) feature that enables a new search interface. Whereas traditional document search allows a user to ask a question and receive a list of documents that might contain the answer to that question, conversational search uses large language models (LLM) to read the top N documents and synthesizes those documents into a plaintext "answer" to your question.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

Currently, conversational search utilizes two systems to synthesize documents:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

- [Conversation memory](#conversation-memory)
- [Retreival Augmented Generation (RAG) pipeline](#rag-pipeline)

## Conversation memory

Conversation memory consists of a simple CRUD-life API comprised of two resources: **Conversations** and **Interactions**. Conversations are made up of interactions. An interaction represents a pair of messages: a human input and the AI response. You cannot create any interactions until you've created a conversation.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

To make it easier to build and debug applications that make use of conversation memory, `conversation-meta` and `conversation-interactions` are stored in two system indexes.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

### `conversation-meta` index

In the `conversation-meta` index, you can customize the `name` field to make it easier for end users to know how to continue a conversation with the AI, as shown in the following schema:

```jsx
.plugins-ml-conversation-meta
{
"_meta": {
"schema_version": 1
},
"properties": {
"name": {"type": "keyword"},
"create_time": {"type": "date", "format": "strict_date_time||epoch_millis"},
"user": {"type": "keyword"}
}
}
```

### `conversation-interactions` index

In the `conversation-interactions` index, all of the following fields are set by the user or AI application. Each field is entered as a string:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Introducing a table, so no colon in this case 😄.


| Field | Description |
| :--- | :--- |
| `input` | The question which forms the basis for an interaction. |
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
| `prompt_template` | The prompt template that was used as the framework for this interaction. |
| `response` | The AI’s response to the prompt. |
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
| `origin` | The name of the AI or other system that generated the response. |
| `additional_info` | Any other information that was sent to the “origin” in the prompt. |
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

The goal of the `conversation-interactions` index is to create a clean interaction abstraction and make it easy for the index to reconstruct the exact prompts sent to the LLM, enabling robust debugging and explainability, as shown in the following schema:
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of "goal", do we actually mean "purpose"? Alternatively, you could rewrite as "The conversation-interactions index creates a clean interaction abstraction and makes it easy..."

Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```jsx
.plugins-ml-conversation-interactions
{
"_meta": {
"schema_version": 1
},
"properties": {
"conversation_id": {"type": "keyword"},
"create_time": {"type": "date", "format": "strict_date_time||epoch_millis"},
"input": {"type": "text"},
"prompt_template": {"type": "text"},
"response": {"type": "text"},
"origin": {"type": "keyword"},
"additional_info": {"type": "text"}
}
}
```

## Working with conversations and interactions

When security is enabled, all conversations are in ML Commons exist in a "private" security mode. Only the person who creates a conversation can interact with that conversation. No other users on the cluster can see another user's conversation.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should "Security" be capitalized?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe switch to "The Security Plugin".

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed, but lowercase "plugin".

{: .note}

To begin using conversation memory, enable the following cluster setting:

```json
PUT /_cluster/settings
{
"persistent": {
"plugins.ml_commons.memory_feature_enabled": true
}
}
```

After conversation memory is enabled, you can now use the Memory API to create a conversation.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

To make the conversation easily identifiable, use the optional `name` field in the Memory API, as shown in the following example. This will be your only opportunity to give your conversation a name.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved



```json
POST /_plugins/_ml/memory/conversation
{
"name": Example conversation
}
```

The Memory API responds with the conversation ID, as shown in the following example response:

```json
{ "conversation_id": "4of2c9nhoIuhcr" }
```

You'll use the `conversation_id` to create interactions inside the conversation. To create interactions, enter the `conversation_id` into the Memory API path. Then, customize the [fields](#conversation-interactions-index) in the request body, as shown in the following example:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```json
POST /_plugins/_ml/memory/conversation/4of2c9nhoIuhcr
{
"input": "How do I make an interaction?",
"prompt_template": "Hello OpenAI, can you answer this question? \
Here's some extra info that may help. \
[INFO] \n [QUESTION]",
"response": "Hello, this is OpenAI. Here is the answer to your question.",
"origin": "MyFirstOpenAIWrapper",
"additional_info": "Additional text related to the answer \
A JSON or other semi-structured response"
}
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
```

The Memory API then responds with an interaction ID, as shown in the following response:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```json
{ "interaction_id": "948vh_PoiY2hrnpo" }
```

### Getting conversations

You can get a list of conversations using the following Memory API operation:
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
You can get a list of conversations using the following Memory API operation:
You can get a list of conversations by using the following Memory API operation:


```json
GET /_plugins/_ml/memory/conversation?max_results=3&next_token=0
```

Use the following path parameters to customize your results:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

Parameter | Data type | Description
:--- | :--- | :---
`max_results` | Integer | The maximum number of results returned by the response. Default is `10`.
`next_token` | Integer | Represents the position in the conversation order to retrieve. For example, if three conversations A, B, and C exist, `next_token=1` would return conversations B and C. Default is `0`.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Confirm that my changes retain the intended meaning.


The Memory API responds with the most recent conversation created first, as indicated in the `create_time` field of the response as shown in the following example:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Confirm that my changes retain the intended meaning.


```json
{
"conversations": [
{
"conversation_id": "0y4hto_in1",
"name": "",
"create_time": "2023-4-23 10:25.324662"
}, ... (2 more since we specified max_results=3)
],
"next_token": 3
}
```


If there are fewer conversations than the number set in `max_results`, the response only returns the number of conversations that exist. Lastly, `next_token` provides an ordered position of the sorted list of conversations. When a conversation is added between subsequent GET conversation calls, one of the listed conversations will be duplicated in the result, for example:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```json
GetConversations -> [BCD]EFGH
CreateConversation -> ABCDEFGH
GetConversations(next_token=3) -> ABC[DEF]GH
```

### Getting interactions

To see a list of interactions in a conversation, enter the `conversation_id` at the end of API request, as shown in the following example. You can use the `max_results` and `next_token` to sort the response:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```json
GET /_plugins/_ml/memory/conversation/4of2c9nhoIuhcr
```

The Memory API returns the following interaction information:

```json
{
"interactions": [
{
"interaction_id": "342968y2u4-0",
"conversation_id": "0y4hto_in1",
"create_time": "2023-4-23 10:25.324662",
"input": "How do I make an interaction?",
"prompt_template": "Hello OpenAI, can you answer this question? \
Here's some extra info that may help. \
[INFO] \n [QUESTION]",
"response": "Hello, this is OpenAI. Here is the answer to your question.",
"origin": "MyFirstOpenAIWrapper",
"additional_info": "Additional text related to the answer \
A JSON or other semi-structured response"
}, ... (9 more since max_results defaults to 10)
],
"next_token": 10
}
```

### Deleting conversations

To delete a conversation, use the `DELETE` operation as showing in the following example:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```json
DELETE /_plugins/_ml/memory/conversation/4of2c9nhoIuhcr
```

The Memory API responds with the following:

```json
{ "success": true }
```

Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
## RAG pipeline

RAG is a technique for grounding LLMs and a natural language interface, implemented as a search pipeline.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

### Enabling RAG

Use the following cluster setting to enable the RAG pipeline feature:

```json
PUT /_cluster/settings
{
"persistent": {"plugins.ml_commons.rag_pipline_feature_enabled": "true"}
}
```

### Connecting the model

RAG requires an LLM to function. We recommend using a [connector]({{site.url}}{{site.baseurl}}/ml-commons-plugin/extensibility/connectors/).
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

Use the following steps to set up an HTTP connector using the OpenAI GPT 3.5 model:

1. Use the Connector API to create the HTTP connector:

```json
POST /_plugins/_ml/connectors/_create
{
"name": "OpenAI Chat Connector",
"description": "The connector to public OpenAI model service for GPT 3.5",
"version": 2,
"protocol": "http",
"parameters": {
"endpoint": "[api.openai.com](http://api.openai.com/)",
"model": "gpt-3.5-turbo",
"temperature": 0
},
"credential": {
"openAI_key": "<your OpenAI key>"
},
"actions": [
{
"action_type": "predict",
"method": "POST",
"url": "[https://$](https://%24/){parameters.endpoint}/v1/chat/completions",
"headers": {
"Authorization": "Bearer ${credential.openAI_key}"
},
"request_body": "{ \"model\": \"${parameters.model}\", \"messages\": ${parameters.messages}, \"temperature\": $ {parameters.temperature} }"
}
]
}
```

2. Create a new model group for the connected model. You'll use the `model_group_id` returned by the Register API to register the model:

```json
POST /_plugins/_ml/model_group/_register
{
"name": "public_model_group",
"description": "This is a public model group"
}
```
3. Register the model and deploy the model using the `connector_id` from the Connector API response in Step 1 and `model_group_id` returned in Step 2:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```java
POST /_plugins/_ml/models/_register
{
"name": "openAI-gpt-3.5-turbo",
"function_name": "remote",
"model_group_id": "fp-hSYoBu0R6vVqGMnM1",
"description": "test model",
"connector_id": "f5-iSYoBu0R6vVqGI3PA"
}
```

4. With the model registered, use the `task_id` returned in the registration response to get the `model_id`. You'll use the `model_id` to deploy the model to OpenSearch:

```json
GET /_plugins/_ml/tasks/<task_id>
```

5. Using your `model_id` from step 4, deploy the model:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```
POST /_plugins/_ml/models/<model_id>/_deploy
```

### Setting up the pipeline
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should "the" be "a"? We use "a" in the following sentence.


Next, we'll create a search pipeline for the connector model. Use the following Search API request to create a pipeline:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```json
PUT /_search/pipeline/<pipeline_name>
{
"response_processors": [
{
"retrieval_augmented_generation": {
"tag": "openai_pipeline_demo",
"description": "Demo pipeline Using OpenAI Connector",
"model_id": "<model_id>",
"context_field_list": ["text"]
}
}
]
}
```

`context_field_list` is the list of fields in document sources that the pipeline uses as context for the RAG. For example, when `context_field_list` parses through the following document:
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The second sentence here is incomplete.

Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

```json
{
"_index": "qa_demo",
"_id": "SimKcIoBOVKVCYpk1IL-",
"_source": {
"title": "Abraham Lincoln 2",
"text": "Abraham Lincoln was born on February 12, 1809, the second child of Thomas Lincoln and Nancy Hanks Lincoln, in a log cabin on Sinking Spring Farm near Hodgenville, Kentucky.[2] He was a descendant of Samuel Lincoln, an Englishman who migrated from Hingham, Norfolk, to its namesake, Hingham, Massachusetts, in 1638. The family then migrated west, passing through New Jersey, Pennsylvania, and Virginia.[3] Lincoln was also a descendant of the Harrison family of Virginia; his paternal grandfather and namesake, Captain Abraham Lincoln and wife Bathsheba (née Herring) moved the family from Virginia to Jefferson County, Kentucky.[b] The captain was killed in an Indian raid in 1786.[5] His children, including eight-year-old Thomas, Abraham's father, witnessed the attack.[6][c] Thomas then worked at odd jobs in Kentucky and Tennessee before the family settled in Hardin County, Kentucky, in the early 1800s.[6]\n"
}
}
```

When configured using the preceeding Search API request, the pipeline sends the `text` field from the response to OpenAI model. You can customize `context_field_list` in your RAG pipeline to send any fields that exist in your documents to the LLM.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like either "the" or "a" should precede "OpenAI".


### Using the pipeline

Using the pipeline is similar to submitting [search queries]({{site.url}}{{site.baseurl}}/api-reference/search/#example) to OpenSearch, as shown in the following example:

```json
GET /<index_name>/_search?search_pipeline=<pipeline_name>
{
"query" : {...},
"ext": {
"generative_qa_parameters": {
"llm_model": "gpt-3.5-turbo",
"llm_question": "Was Abraham Lincoln a good politician",
"conversation_id": "_ikaSooBHvd8_FqDUOjZ"
}
}
}
```

The RAG search query uses the following request objects under the `generative_qa_paramters` option:
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

Parameter | Required | Description
:--- | :--- | :---
`llm_question` | Yes | The question the LLM must answer.
`llm_model` | No | Overrides the original model set in the connection in cases where you want to use a different model. For example, using GPT 4 instead of GPT 3.5. This option is required if a default model is not set during pipeline creation.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
`coversation_id` | No | Integrates conversation memory into your RAG pipeline by adding the 10 most recent conversations into the context of search query to the LLM.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like either "a" or "the" should precede "search query".

Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

If you're LLM includes a set token limit, set the `size` field in you OpenSearch query to limit the amount of documents used in the search response. Otherwise, the RAG pipeline will send every document in the search result into the LLM.
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

## Next Steps
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved

- To learn more about ML connectors, see [Creating connectors for third-party ML platforms]({{site.url}}{{site.baseurl}}/ml-commons-plugin/extensibility/connectors/)
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved
- To learn more about the ML framework for OpenSearch, see [ML framework]({{site.url}}{{site.baseurl}}/ml-commons-plugin/ml-framework/)
Naarcha-AWS marked this conversation as resolved.
Show resolved Hide resolved