This is a guide to set up the repository with Azure DevOps pipelines for experimenting and evaluating Prompt flows in Azure Machine Learning. The repository contains examples of Prompt flow runs and evaluations for named entity recognition, math coding, and web classification.
This template supports Azure Machine Learning (ML) as a platform for LLMOps, and Azure DevOps pipelines as a platform for Flow operationalization. LLMOps with Prompt flow provides automation of:
- Experimentation by executing prompt flow 'flows'
- Evaluation of prompts along with their variants
- Registration of prompt flow 'flows'
- Deployment of prompt flow 'flows'
- Generation of Docker Image
- Deployment to Kubernetes, Azure Web Apps and Azure ML compute
- A/B deployments
- Role based access control (RBAC) permissions to deployment system managed id to key vault and Azure ML workspace
- Endpoint testing
- Report generation
It is important to understand how Prompt flow works before using this template.
- An Azure subscription. If you don't have an Azure subscription, create a free account before you begin. Try the free or paid version of Machine Learning.
- An Azure Machine Learning workspace.
- Git running on your local machine.
- Azure DevOps organization.
- Azure OpenAI with Model deployed with name
gpt-35-turbo
. - In case of Kubernetes based deployment, Kubernetes resources and associating it with Azure Machine Learning workspace would be required. More details about using Kubernetes as compute in AzureML is available here.
The template deploys real-time online endpoints for flows. These endpoints have Managed ID assigned to them and in many cases they need access to Azure Machine learning workspace and its associated key vault. The template by default provides access to both the key vault and Azure Machine Learning workspace based on this document
This section provides instructions on how to execute template either locally or on Azure, based on the configuration specified in the config.py
file within the llmops
folder.
The config.py
file located in the llmops
folder contains the EXECUTION_TYPE
variable, which determines where the flow will be executed.
- Set
EXECUTION_TYPE = "LOCAL"
to execute the flows on your local machine. - Set
EXECUTION_TYPE = "AZURE"
to execute the flows on Azure.
When the EXECUTION_TYPE
variable is set to "LOCAL"
, the templates will be executed on your local machine. This is useful for local development, testing, and debugging.
Setup the environment by installing the required dependencies using the following command:
pip install -r ./.azure-pipelines/requirements/execute_job_requirements.txt
# select the requirements file based on the usecase you are working on e.g. for web_classification
pip install -r ./web_classification/flows/experiment/requirements.txt
To execute a template locally, use the following command:
python -m llmops.common.prompt_pipeline --base_path ./<example_path> --variants <template_name>
Replace <example_path>
with the path to the specific example you want to run (e.g., ./web_classification
) and <template_name>
with the name of the template (e.g., summarize_text_content.variant_0
). The --variants
argument is optional and can be used to specify the variant of the template to run.
Ensure that you have a .env file with valid values for the required environment variables. The .env file should be located in the root directory of the example you are running. An example .env.sample file is provided which can be renamed to .env file.
When the EXECUTION_TYPE
variable is set to "AZURE"
, the flows will be executed on Azure. This allows you to leverage the scalability and resources provided by Azure for running your experiments and evaluations.
There are two ways to execute templates on Azure:
-
CI/CD Pipelines: Set up a CI/CD pipeline in your Azure DevOps/Github/Jenkins. Configure the pipeline to trigger the execution of the template on Azure. Ensure that the
EXECUTION_TYPE
variable inconfig.py
is set to"AZURE"
. -
Direct Execution from Local Machine: You can also execute templates on Azure directly from your local machine. Set the
EXECUTION_TYPE
variable inconfig.py
to"AZURE"
and use the following command:
Setup the environment by installing the required dependencies using the following command:
pip install -r ./github/requirements/execute_job_requirements.txt
# select the requirements file based on the usecase you are working on e.g. for web_classification
pip install -r ./web_classification/flows/experiment/requirements.txt
python -m llmops.common.prompt_pipeline --base_path ./<example_path> --variants <template_name>
Ensure that you have a .env file with valid values for the required environment variables. The .env file should be located in the root directory of the example you are running. An example .env.sample file is provided which can be renamed to .env file.
- Make sure to review and modify the
config.py
file based on your specific requirements and environment setup. - Ensure that you have the necessary dependencies installed and the required configurations set up for the execution environment you choose LOCAL.
- When executing templates on Azure, ensure that you have the appropriate permissions and credentials set up to access Azure resources.
- Be aware of any costs associated with running experiments and evaluations on Azure, as it may incur charges based on resource usage.
This template utilizes secrets to securely store sensitive information such as API keys, credentials, and other confidential data. Secrets are managed differently depending on the execution environment, whether it's local execution or execution through Azure DevOps pipelines.
For local execution, secrets are stored in a .env file located in the project's root directory. The .env file contains key-value pairs representing the secrets. For example:
AOAI_API_KEY=abcdefghijklmnop
max_total_token=4096
The .env file is alreadyadded to the .gitignore file to prevent it from being committed and pushed to the remote repository, keeping the secrets secure.
When executing the template through Azure DevOps pipelines, secrets are stored as Azure DevOps variable groups. In this template, a secret named 'ENV_VARS' is used to store the secrets. The 'ENV_VARS' secret should contain the same key-value pairs as the .env file. For example:
AOAI_API_KEY=abcdefghijklmnop
AZURE_OPENAI_API_KEY=xxxxxx
Azure DevOps secrets can be accessed and managed in the repository's settings under the "Library" section of "pipelines" tab.
Secrets are referenced in the experiment.yaml, flex.flow.yaml, init.json and env.yaml files using a special syntax: ${SECRET_NAME}. Check out Function_flows as an example.
# experiment.yaml
connections:
- name: aoai
connection_type: AzureOpenAIConnection
api_base: https://xxxxx.openai.azure.com/
api_version: 2023-07-01-preview
api_key: ${api_key}
api_type: azure
# env.yaml
AZURE_OPENAI_API_KEY: ${AZURE_OPENAI_API_KEY}
During execution, the ${api_key} and ${AZURE_OPENAI_API_KEY} placeholders are replaced with the corresponding values from the secrets.
When executing the template through Azure DevOps pipelines, the secrets are loaded from the ENV_VARS Azure DevOps variable group. The placeholders in experiment.yaml and env.yaml are replaced with the values from the ENV_VARS secret. For example, if the ENV_VARS secret contains:
AOAI_API_KEY=abcdefghijklmnop
AZURE_OPENAI_API_KEY=xxxxxx
The placeholders in experiment.yaml, flow.flex.yaml, init.json and env.yaml will be replaced in the same way as in local execution.
To ensure clarity and avoid naming conflicts, it's important to follow a specific naming convention for secrets used in this template. When defining secrets for connections, such as API keys, the secret name should be qualified with the connection name. This helps to distinguish between secrets used for different connections. The naming convention for connection secrets is as follows:
<CONNECTION_NAME>_API_KEY
For example, if you have a connection named AOAI, the corresponding API key secret should be named AOAI_API_KEY. Only API_KEY is supported for connections.
If secrets are used within the init section of the flow.flex.yaml file, the secret name should be qualified with the parent name. This helps to identify the specific flow and context in which the secret is used. The naming convention for flow secrets is as follows:
<PARENT_ELEMENT_NAME>_<SECRET_NAME>
For example, if you have a secret named API_KEY used within the init section of a flow under MODEL_CONFIG, the corresponding secret should be named MODEL_CONFIG_API_KEY.
By adhering to the naming convention for secrets, storing them securely in the .env file or Azure DevOps variable group, and following the best practices for secret security, you can ensure that sensitive information remains protected while allowing seamless execution of the template in different environments.
Create an Azure service principal for the purpose of working with this repository. You can add more depending on number of environments you want to work on (Dev or Prod or Both). Service principals can be created using cloud shell, bash, PowerShell or from Azure UI. If your subscription is a part of organization with multiple tenants, ensure that the Service Principal has access across tenants.
-
Copy the following bash commands to your computer and update the spname (of your choice) and subscriptionId variables with the values for your project. This command will also grant the owner role to the service principal in the subscription provided. This is required for Azure DevOps pipelines to properly use resources in that subscription.
spname="<provide a name to create a new sp name>" roleName="Owner" subscriptionId="<provide your subscription Id>" servicePrincipalName="Azure-ARM-${spname}" # Verify the ID of the active subscription echo "Using subscription ID $subscriptionID" echo "Creating SP for RBAC with name $servicePrincipalName, with role $roleName and in scopes /subscriptions/$subscriptionId" az ad sp create-for-rbac --name $servicePrincipalName --role $roleName --scopes /subscriptions/$subscriptionId --sdk-auth echo "Please ensure that the information created here is properly save for future use."
-
Copy your edited commands into the Azure Shell and run them (Ctrl + Shift + v). If executing the commands on local machine, ensure Azure CLI is installed and successfully able to access after executing
az login
command. Azure CLI can be installed using information available here -
After running these commands, you'll be presented with information related to the service principal. Save this information to a safe location, you'll use it later in the demo to configure Azure DevOps pipelines.
NOTE: The below information should never be part of your repository and its branches. These are important secrets and should never be pushed to any branch in any repository.
```json
{
"clientId": "<service principal client id>",
"clientSecret": "<service principal client secret>",
"subscriptionId": "<Azure subscription id>",
"tenantId": "<Azure tenant id>",
"activeDirectoryEndpointUrl": "https://login.microsoftonline.com",
"resourceManagerEndpointUrl": "https://management.azure.com/",
"activeDirectoryGraphResourceId": "https://graph.windows.net/",
"sqlManagementEndpointUrl": "https://management.core.windows.net:8443/",
"galleryEndpointUrl": "https://gallery.azure.com/",
"managementEndpointUrl": "https://management.core.windows.net/"
}
```
-
Copy the output, braces included. It will be used later in the demo to configure Azure DevOps Repo.
-
Close the Cloud Shell once the service principals are created.
Prompt flow 'flows' require runtime associated with compute instance in Azure Machine Learning workspace. The template provides support for 'automatic runtime' where flows are executed within a runtime provisioned automatically during execution. The first execution might need additional time for provisioning of the runtime.
The template supports using 'automatic runtime' and compute instances. There is no configured required for using automatic runtime. However, if you want to use compute instances, you can create a compute instance in Azure Machine Learning workspace. More details about creating compute instances in Azure Machine Learning workspace is available here.
- Create an Azure DevOps project to establish a repository for source code and CI/CD execution for Prompt flows. Information about creating a new project in Azure DevOps can be found here .
Do not initialize the repository. If you created an empty repo with no README or .gitignore files, you'll see instructions on how to clone the repo to your computer. You'll also see instructions on how to push code in an existing repo into the newly created one. The repo should look like the image shown here.
- Get the url of the repo and note it somewhere for its usage in next steps.
Create a service connection named azure_connection
in Azure DevOps. You can use this document as a reference. Use Azure Resource Manager as a type of the service connection.
From your Azure DevOps project, select Project Settings -> Service connections -> New Service connection -> Azure Resource Manager -> Service Principal (manual)
. Fill the form with relevant data related to Service Principal, test
and save the connection with the name azure_connection
(it is used in the pipelines in this repo). Note: Service principal key is clientSecret
field from the Service principal configs.
Create a new variable group llmops_platform_dev_vg
(follow the documentation) with the following variables:
- AZURE_RM_SVC_CONNECTION: the name of service connection created in previous step.
- rg_name: Name of the resource group containing the Azure ML Workspace
- ws_name: Name of the Azure ML Workspace
- kv_name: Name of the Key Vault associated with the Azure ML Workspace
- COMPUTE_TARGET: Name of the compute cluster used in the Azure ML Workspace (Note: this is only needed if you are executing the Promptflow in AML Pipeline)
Create another variable named 'DOCKER_IMAGE_REGISTRY' within the llmops_platform_dev_vg
variable group with information related to Docker Image Registry. The value for this secret is also a json string with given structure. Create one json object for each available registry. As of now, Azure Container Registry are supported and more will get added soon. Information for each registry can be obtained from Azure Container Registry resource.
It is important to note that there can be any number of variables and each storing Azure Container Registry details as shown next and they can be named anything permissible based on your preference. You should use the same variable in your use-case related CI pipelines. The template by default uses 'DOCKER_IMAGE_REGISTRY' for this purpose.
[
{
"registry_name" : "xxxxxxxx",
"registry_server" : "xxxx.azurecr.io",
"registry_username" : "xxxxxxxxx",
"registry_password": "xxxxxxxxxxxxxx"
}
]
Create another Azure DevOps variable within variable group named 'ENV_VARS'. The data for this secret is same as you put in .env file. Each line is a name=value pair and the values are separated by a newline. The values can be anything permissible based on your preference. The keys should be in upper case. You should use the same key in your use-case related CI pipelines. This includes you python code, experiment.yaml and pipelines. The template by default uses 'ENV_VARS' for this purpose.
The environment variables for Prompt Flow Connections should follow a specific naming convention to ensure clarity and consistency. The format is as follows:
<CONNECTION_NAME>_API_KEY
<CONNECTION_NAME>
: Replace this with the name of the Prompt Flow Connection for which the API key is associated. The connection name should be in uppercase._API_KEY
: This suffix indicates that the environment variable represents an API key.
```.env
AOAI_API_KEY=xxxxxxxxxxxx
ANY_OTHER_VALUE=xxxxxxxxxxxx
Prompt flow Connections helps securely store and manage secret keys or other sensitive credentials required for interacting with LLM and other external tools for example Azure OpenAI.
This repository has 3 examples and all the examples uses connection named aoai
inside, we need to set up a connection with this name if we haven’t created it before.
This repository has all the examples using Azure OpenAI model gpt-35-turbo
deployed with the same name gpt-35-turbo
, we need to set up this deployment if we haven’t created it before.
Please go to Azure Machine Learning workspace portal, click Prompt flow
-> Connections
-> Create
-> Azure OpenAI
, then follow the instruction to create your own connections called aoai
. Learn more on connections. The samples uses a connection named "aoai" connecting to a gpt-35-turbo model deployed with the same name in Azure OpenAI. This connection should be created before executing the out-of-box flows provided with the template.
The configuration for connection used while authoring the repo:
Clone this Github Repository LLMOps Prompt flow Template locally on your workstation. This repo has reusable LLMOps code that can be used across multiple projects.
git clone --branch main https://github.com/microsoft/llmops-promptflow-template.git
cd llmops-promptflow-template
git branch
git fetch origin development:development
git checkout development
git branch
- By default, the cloned repo will point to github remote repo. This should be changed and start pointing to Azure DevOps repository. Update the url pointed by git origin to the Azure DevOps repo url copied from previous step using the following commands. Replace the url with your url and push both main and development branches to Azure DevOps repo.
git remote set-url origin https://[email protected]/yoururl/test-llmops-template/_git/test-llmops-template
git push --all
As a result the code for LLMOps Prompt flow template will now be available in Azure DevOps repository.
Development
branch should be the default branch in Azure DevOps. If not already set as default branch, it should be configured to be the default branch. This template assumes that the team works at a development branch as a primary source for coding. Later, you can implement Azure Pipeline that mode code from the development branch into qa/main or execute a release process right away. Release management is not in scope of this template.
- Create two Azure Pipelines [how to create a basic Azure Pipeline] for each scenario (e.g. named_entity_recognition). Both Azure Pipelines should be created based on existing YAML files:
-
The first one is based on the named_entity_recognition_pr_dev_pipeline.yml, and it helps to maintain code quality for all PRs including integration tests for the Azure ML experiment. Usually, we recommend to have a toy dataset for the integration tests to make sure that the Prompt flow job can be completed fast enough - there is not a goal to check prompt quality and we just need to make sure that our job can be executed.
-
The second Azure Pipeline is based on named_entity_recognition_ci_dev_pipeline.yml is executed automatically once new PR has been merged into the development or main branch. The main idea of this pipeline is to execute bulk run, evaluation on the full dataset for all prompt variants. Both the pipelines can be modified and extended based on the project's requirements.
These following steps should be executed twice - once for PR pipeline and again for CI pipeline.
In resultant page, select the location of the code - select Azure Repos Git
In next page, select the repository. There should be only visible repository.
Select Existing Azure Pipeline YAML file
from resultant page.
select the CI yaml file named_entity_recognition_ci_dev_pipeline.yml
for named_entity_recognition scenario from the development
branch.
Save the new pipeline.
Rename the saved pipeline as named-entity-ci
.
Perform the same steps as before for PR pipeline and name is named-entity-pr
. By the end of this step both the CI and PR pipeline for named_entity_recognition should be available in the Azure DevOps repository.
- Setup a policy for the development branch. At this stage we have an Azure Pipeline that should be executed on every PR to the development branch. At the same time successful completion of the build is not a requirement. So, it's important to add our PR build as a policy. Pay special attention that named_entity_recognition_pr_dev_pipeline.yml has various paths in place. We are using these paths to limit number of runs if the current PR doesn't affect ML component (for example, PR modifies a documentation file). Therefore, setting up the policy you need to make sure that you are using the same set of paths in the Path filter field.
From your Azure DevOps project, select Repos -> Branches -> more options button against development branch -> Branch policies -> Build validation
and fill the form using the appropriate PR pipeline.
More details about how to create a policy can be found here.
There is another azure devops pipeline added :web_classification_pf_in_aml_pipeline_workflow.yml
- It is used to run the promptflow in AML Pipeline as a parallel component.
- You can use this to run other use cases as well, all you need to do is change the use_case_base_path to other use cases, like math_coding, named_entity_recognition.
From local machine, create a new git branch featurebranch
from development
branch.
git checkout -b featurebranch
Important: The pipeline expects the variables
rg_name
,ws_name
andkv_name
to be available in the variable groupllmops_platform_dev_vg
(this should have already been created and set here). These variables should contain the values of the Azure resources in the dev environment. To add a new environment, you must create a new variable groupllmops_platform_<env>_vg
and populate it with the variable values from the new environment. This variable group can then be used in the new pipelines created for that environment.
The rest of the pipeline configurations will be read from the experiment.yaml
file (see file description and specs); and from the configs/deployment_config.json
file for the deployment.
Before running the deployment pipelines, you need to make changes to configs/deployment_config.json
:
- Update the
ENDPOINT_NAME
andCURRENT_DEPLOYMENT_NAME
if you want to deploy to Azure Machine Learning compute - Or update the
CONNECTION_NAMES
,REGISTRY_NAME
,REGISTRY_RG_NAME
,APP_PLAN_NAME
,WEB_APP_NAME
,WEB_APP_RG_NAME
,WEB_APP_SKU
, andUSER_MANAGED_ID
if you want to deploy to Azure Web App.
Modify the configuration values in deployment_config.json
file for each environment. These are required for deploying Prompt flows in Azure ML. Ensure the values for ENDPOINT_NAME
and CURRENT_DEPLOYMENT_NAME
are changed before pushing the changes to remote repository.
ENV_NAME
: This indicates the environment name, referring to the "development" or "production" or any other environment where the prompt will be deployed and used in real-world scenarios.TEST_FILE_PATH
: The value represents the file path containing sample input used for testing the deployed model.ENDPOINT_NAME
: The value represents the name or identifier of the deployed endpoint for the prompt flow.ENDPOINT_DESC
: It provides a description of the endpoint. It describes the purpose of the endpoint, which is to serve a prompt flow online.DEPLOYMENT_DESC
: It provides a description of the deployment itself.PRIOR_DEPLOYMENT_NAME
: The name of prior deployment. Used during A/B deployment. The value is "" if there is only a single deployment. Refer to CURRENT_DEPLOYMENT_NAME property for the first deployment.PRIOR_DEPLOYMENT_TRAFFIC_ALLOCATION
: The traffic allocation of prior deployment. Used during A/B deployment. The value is "" if there is only a single deployment. Refer to CURRENT_DEPLOYMENT_TRAFFIC_ALLOCATION property for the first deployment.CURRENT_DEPLOYMENT_NAME
: The name of current deployment.CURRENT_DEPLOYMENT_TRAFFIC_ALLOCATION
: The traffic allocation of current deployment. A value of 100 indicates that all traffic is directed to this deployment.DEPLOYMENT_VM_SIZE
: This parameter specifies the size or configuration of the virtual machine instances used for the deployment.DEPLOYMENT_INSTANCE_COUNT
:This parameter specifies the number of instances (virtual machines) that should be deployed for this particular configuration.ENVIRONMENT_VARIABLES
: This parameter represents a set of environment variables that can be passed to the deployment.
Kubernetes deployments have additional properties - COMPUTE_NAME
, DEPLOYMENT_VM_SIZE
, CPU_ALLOCATION
and MEMORY_ALLOCATION
related to infrastructure and resource requirements. These should also be updates with your values before executing Kubernetes based deployments.
Azure Web App deployments do not have similar properties as that of Kubernetes and Azure ML compute. For Azure Web App, following properties should be updated.
ENV_NAME
: This indicates the environment name, referring to the "development" or "production" or any other environment where the prompt will be deployed and used in real-world scenarios.TEST_FILE_PATH
: The value represents the file path containing sample input used for testing the deployed model.CONNECTION_NAMES
: The name of the connections used in standard flow in json aray format. e.g. ["aoai", "another_connection"].REGISTRY_NAME
: This is the name of the Container Registry that is available in theDOCKER_IMAGE_REGISTRY
variable inllmops_platform_dev_vg
variable group. Based on this name, appropriate registry details will be used forDocker
image management.REGISTRY_RG_NAME
: This is the name of the resource group related to the Container Registry. It is used for downloading the Docker Image.APP_PLAN_NAME
: Name of the App Services plan. It will be provisioned by the pipeline.WEB_APP_NAME
: Name of the Web App. It will be provisioned by the pipeline.WEB_APP_RG_NAME
: Name of the resource group related to App Service plan and Web App. It will be provisioned by the pipeline.WEB_APP_SKU
: This is theSKU
(size) of the Web App. e.g. "B3"USER_MANAGED_ID
: This is the name of the user defined managed id created during deployment associated with the Web App.
NOTE: For Docker based deployments, ensure to add promptflow, promptflow[azure], promptflow-tools packages in each flow's requirements.txt file
. Check-out existing use-cases (named_entity_recognition, math_coding, web_classification) for examples.`
Now, push the new feature branch to the newly forked repo.
git add .
git commit -m "changed code"
git push -u origin featurebranch
- Raise a new PR to merge code from
featurebranch
to thedevelopment
branch.
This should start the process of executing the Math_coding PR pipeline.
- After the execution is complete, the code can be merged to the
development
branch.
Now a new PR can be opened from development
branch to the main
branch. This should execute both the PR as well as the CI pipeline.
To make any changes to your experiment set-up, refer to the experiment.yaml
in the base path of your use-case. This file configures your use-case including flows and datasets. (see file description and specs for more details).
Add your data into data folder under use case folder. It supports jsonl files and the examples already contains data files for both running and evaluating Prompt flows.
The flows folder contains one folder for each standard and evaluation flow. Each example in the repository already has these flows.
The environment folder
contains dockerfile file for webapp and function app deployments. Any additional dependencies needed by the flow should be added to it. This file is used during deployment process..
The sample-request.json file contains a single test record used for testing the online endpoint after deployment from the pipeline. Each example has its own sample-request.json file and for custom flows, it should be updated to reflect test data needed for the scenario.
Manual approval is enabled by default. The template uses a dummy [email protected]
email Id. Replace this with valid email address in code.
There are three examples in this template. While named_entity_recognition and math_coding has same functionality, web_classification has multiple evaluation flows and datasets for dev environment. This is a flow in general across all examples.
This Azure DevOps CI pipelines contains the following steps:
Run Prompts in Flow
- Upload bulk run dataset
- Bulk run prompt flow based on dataset.
- Bulk run each prompt variant
Run promptflow in AML Pipeline as parallel component
- It reuses the already registered data assets for input.
- Runs the promptflow in AML Pipeline as a parallel component, where we can control the concurrency and parallelism of the promptflow execution. For more details refer here.
- The output of the promptflow is stored in the Azure ML workspace.
Evaluate Results
- Upload ground test dataset
- Execution of multiple evaluation flows for a single bulk run (only for web_classification)
- Evaluation of the bulk run result using single evaluation flow (for others)
Manual Approval
- approval required before deployment
Register Prompt flow LLM App
- Register Prompt flow as a Model in Azure Machine Learning Model Registry
Deploy and Test LLM App
- Deploy the Flow as a model to development environment either as Kubernetes or Azure ML Compute endpoint
- Assign RBAC permissions to the newly deployed endpoint related to Key Vault and Azure ML workspace
- Test the model/promptflow realtime endpoint.
-
After the CI pipeline for an example scenario has run successfully, depending on the configuration it will either deploy to
-
Once pipeline execution completes, it would have successfully completed the test using data from sample-request.json file as well.
The example scenario can be run and deployed both for Dev environments. When you are satisfied with the performance of the prompt evaluation pipeline, Prompt flow model, and deployment in development, additional pipelines similar to dev
pipelines can be replicated and deployed in the Production environment.
The sample Prompt flow run & evaluation and Azure DevOps pipelines can be used as a starting point to adapt your own prompt engineering code and data.