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 public doc for enabled_by_type & enabled_by_value #810

Closed
wants to merge 11 commits into from
Closed
Show file tree
Hide file tree
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
Original file line number Diff line number Diff line change
Expand Up @@ -162,5 +162,6 @@ Alternatively, you can test your tool package using the script below to ensure t
* If you encounter a `403 Forbidden Error`, it's likely due to a naming conflict with an existing package. You will need to choose a different name. Package names must be unique on PyPI to avoid confusion and conflicts among users. Before creating a new package, it's recommended to search PyPI (https://pypi.org/) to verify that your chosen name is not already taken. If the name you want is unavailable, consider selecting an alternative name or a variation that clearly differentiates your package from the existing one.

## Advanced features
[Customize your tool icon](add-a-tool-icon.md)
[Add category and tags for tool](add-category-and-tags-for-tool.md)
[Customize your tool icon](add-a-tool-icon.md)
[Add category and tags for tool](add-category-and-tags-for-tool.md)
[Use Enabled By to Support Cascading Settings between Inputs for Tool](how-to-use-enabled-by.md)
217 changes: 217 additions & 0 deletions docs/how-to-guides/develop-a-tool/how-to-use-enabled-by.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,217 @@
# Use Enabled By to Support Cascading Settings between Inputs for Tool

This guide will instruct you on how to use the "enabled by" feature in the tool package. The "enabled by" feature is designed to support cascading settings between inputs for tool.
Cascading settings between inputs are frequently used in situations where the selection in one input field determines what subsequent inputs should be shown.
This approach help in creating a more efficient, user-friendly, and error-free input process. To support cascading settings between tool inputs, we've introduced following two configurations:
* "enabled_by + enabled_by_type": This setting enabled the current input based on the value type of the input shown in "enabled_by" field.
* "enabled_by + enabled_by_value": This setting enabled the current input based on the actual value of the input shown in "enabled_by" field.

> Note: The "enabled_by_type" and "enabled_by_value" configurations should not be used simultaneously for the same input. Both enabled_by_type and enabled_by_value in the tool yaml are of list type, implying that a single input can be enabled by multiple values from the dependent input.
## Prerequisites
To proceed, it's crucial for you to understand the process of developing a tool and generating a tool yaml. For thorough insights and instructions, please refer to [Create and Use Tool Package](create-and-use-tool-package.md).

## Use "enabled_by + enabled_by_type" in tool
Here we take [this existing tool](https://github.com/microsoft/promptflow/blob/eeb1df65e55d93e1a6fe30f58adda1e9d241db61/src/promptflow-tools/promptflow/tools/embedding.py) as example, this tool has four inputs: "connection", "input", "deployment_name", and "model".
The "deployment_name" and "model" are enabled by the type of input "connection". When the type of "connection" input is AzureOpenAIConnection, the "deployment_name" input is enabled and displayed. When the type of "connection" input is OpenAIConnection, the "model" input is enabled and displayed.
Below shows how to support this cascading setting in both tool code and tool yaml.

### Step 1: How to define the rules in the tool
All inputs will be passed to the tool, allowing you to define your own set of rules to determine which input to use. You need to pay attention to some key points:
* When an input can be selected as multiple types, using Union can combine the types. For example, "Union[AzureOpenAIConnection, OpenAIConnection]".
* In tool logic, "isinstance" can be used to determine the input type. Different inputs can be used according to the different types.

Here is an example of how you can support the "enabled by" feature in your tool:


```python
from enum import Enum
from typing import Union

import openai

from promptflow.connections import AzureOpenAIConnection, OpenAIConnection
from promptflow._internal import tool
from promptflow.tools.exception import InvalidConnectionType


class EmbeddingModel(str, Enum):
TEXT_EMBEDDING_ADA_002 = "text-embedding-ada-002"
TEXT_SEARCH_ADA_DOC_001 = "text-search-ada-doc-001"
TEXT_SEARCH_ADA_QUERY_001 = "text-search-ada-query-001"


@tool
def embedding(connection: Union[AzureOpenAIConnection, OpenAIConnection], input: str, deployment_name: str = "", model: EmbeddingModel = EmbeddingModel.TEXT_EMBEDDING_ADA_002):
connection_dict = dict(connection)
# If the connection type is AzureOpenAIConnection, use the deployment_name input.
if isinstance(connection, AzureOpenAIConnection):
return openai.Embedding.create(
input=input,
engine=deployment_name,
**connection_dict,
)["data"][0]["embedding"]
# If the connection type is OpenAIConnection, use the model input.
elif isinstance(connection, OpenAIConnection):
return openai.Embedding.create(
input=input,
model=model,
**connection_dict,
)["data"][0]["embedding"]
else:
error_message = f"Not Support connection type '{type(connection).__name__}' for embedding api. " \
f"Connection type should be in [AzureOpenAIConnection, OpenAIConnection]."
raise InvalidConnectionType(message=error_message)
```

### Step 2: Support "enabled_by_type" in the tool yaml
Once you have generated a tool yaml, you can incorporate the "enabled_by_type" into it. Here is an example showcasing the use of "enabled_by_type" in the tool yaml:

```yaml
promptflow.tools.embedding.embedding:
name: Embedding
description: Use Open AI's embedding model to create an embedding vector representing the input text.
type: python
module: promptflow.tools.embedding
function: embedding
inputs:
connection:
type: [AzureOpenAIConnection, OpenAIConnection]
deployment_name:
type:
- string
# The input deployment_name is enabled by connection
enabled_by: connection
# When the connection type is AzureOpenAIConnection, deployment_name is enabled and displayed.
enabled_by_type: [AzureOpenAIConnection]
capabilities:
completion: false
chat_completion: false
embeddings: true
model_list:
- text-embedding-ada-002
- text-search-ada-doc-001
- text-search-ada-query-001
model:
type:
- string
# The input model is enabled by connection
enabled_by: connection
# When the connection type is OpenAIConnection, model is enabled and displayed.
enabled_by_type: [OpenAIConnection]
enum:
- text-embedding-ada-002
- text-search-ada-doc-001
- text-search-ada-query-001
input:
type:
- string
```



## Use "enabled_by + enabled_by_value" in tool
Assume you want to develop a tool with four inputs: "connection", "input", "deployment_name", and "model". The "deployment_name" and "model" are enabled by "connection" value. When the "connection" value is "azure-open-ai-connection", the "deployment_name" input is enabled and displayed. When the "connection" value is "open-ai-connection", the "model" input is enabled and displayed. You need to support enable by in two part: tool and tool yaml. Here is an example of how you can support the "enabled by" feature in your tool and tool yaml.


### Step 1: How to define the rules in the tool
All inputs will be passed to the tool, allowing you to define your own set of rules to determine which input to use. You need to pay attention to some key points:
* When an input has the potential to have multiple values, an Enum class such as "ConnectionName" as shown in the following example can be created.
* Within the tool's logic, various inputs may be used depending on the respective connection values.


```python
from enum import Enum
from typing import Union

import openai

from promptflow.connections import AzureOpenAIConnection, OpenAIConnection
from promptflow._internal import tool


class EmbeddingModel(str, Enum):
TEXT_EMBEDDING_ADA_002 = "text-embedding-ada-002"
TEXT_SEARCH_ADA_DOC_001 = "text-search-ada-doc-001"
TEXT_SEARCH_ADA_QUERY_001 = "text-search-ada-query-001"

class ConnectionName(str, Enum):
Azure_Open_AI_CONNECTION = "azure-open-ai-connection"
Open_AI_CONNECTION = "open-ai-connection"

@tool
def embedding(connection: Enum, input: str, deployment_name: str = "", model: EmbeddingModel = EmbeddingModel.TEXT_EMBEDDING_ADA_002):
connection_dict = dict(connection)
Copy link
Contributor

Choose a reason for hiding this comment

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

If connection is a str Enum, how it can be used as a dict to send an API call ?

# If the connection value is azure-open-ai-connection, use the deployment_name input.
if connection == ConnectionName.Azure_Open_AI_CONNECTION:
return openai.Embedding.create(
input=input,
engine=deployment_name,
**connection_dict,
)["data"][0]["embedding"]
# If the connection type is open-ai-connection, use the model input.
elif connection == ConnectionName.OPEN_AI_CONNECTION:
return openai.Embedding.create(
input=input,
model=model,
**connection_dict,
)["data"][0]["embedding"]
else:
error_message = f"Not Support connection value '{connection}' for embedding api. " \
f"Connection value should be in [{ConnectionName.Azure_Open_AI_CONNECTION}, {ConnectionName.Open_AI_CONNECTION}]."
raise Exception(error_message)
```

### Step 2: Support "enabled_by_value" in the tool yaml
Once you have generated a tool yaml, you can incorporate the "enabled_by_value" feature into it. Here is an example showcasing the use of "enabled_by_value" in the tool yaml:

```yaml
promptflow.tools.embedding.embedding:
name: Embedding
description: Use Open AI's embedding model to create an embedding vector representing the input text.
type: python
module: promptflow.tools.embedding
function: embedding
inputs:
connection:
type:
- string
enum:
- azure-open-ai-connection
- open-ai-connection
deployment_name:
type:
- string
# The input deployment_name is enabled by connection
enabled_by: connection
# When the connection value is azure-open-ai-connection, deployment_name is enabled and displayed.
enabled_by_value: [azure-open-ai-connection]
capabilities:
completion: false
chat_completion: false
embeddings: true
model_list:
- text-embedding-ada-002
- text-search-ada-doc-001
- text-search-ada-query-001
model:
type:
- string
# The input model is enabled by connection
enabled_by: connection
# When the connection value is open-ai-connection, model is enabled and displayed.
enabled_by_value: [open-ai-connection]
enum:
- text-embedding-ada-002
- text-search-ada-doc-001
- text-search-ada-query-001
input:
type:
- string
```

## Use the tool from VSCode Extension
After you build and share the tool package with "enabled_by" feature, you can use your tool from VSCode Extension according to [Create and Use Tool Package](create-and-use-tool-package.md). For instance, when you select a connection with "AzureOpenAIConnection" type, only "deployment_name" input is enabled and displayed for "enabled_by_type" example.

![enabled_by_type.png](../../media/how-to-guides/develop-a-tool/enabled_by_type.png)


1 change: 1 addition & 0 deletions docs/how-to-guides/develop-a-tool/index.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,5 @@ We provide guides on how to develop a tool and use it.
create-and-use-tool-package
add-a-tool-icon
add-category-and-tags-for-tool
how-to-use-enabled-by
```
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading