mirror of
https://github.com/qodo-ai/pr-agent.git
synced 2025-07-04 21:00:40 +08:00
Merge pull request #396 from Codium-ai/hl/custom_labels
Implement Custom Labels for PRs
This commit is contained in:
35
docs/GENERATE_CUSTOM_LABELS.md
Normal file
35
docs/GENERATE_CUSTOM_LABELS.md
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
# Generate Custom Labels
|
||||||
|
The `generte_labels` tool scans the PR code changes, and given a list of labels and their descriptions, it automatically suggests labels that match the PR code changes.
|
||||||
|
|
||||||
|
It can be invoked manually by commenting on any PR:
|
||||||
|
```
|
||||||
|
/generte_labels
|
||||||
|
```
|
||||||
|
For example:
|
||||||
|
|
||||||
|
If we wish to add detect changes to SQL queries in a given PR, we can add the following custom label along with its description:
|
||||||
|
|
||||||
|
<kbd><img src=./../pics/custom_labels_list.png width="768"></kbd>
|
||||||
|
When running the `generte_labels` tool on a PR that includes changes in SQL queries, it will automatically suggest the custom label:
|
||||||
|
<kbd><img src=./../pics/custom_label_published.png width="768"></kbd>
|
||||||
|
|
||||||
|
### Configuration options
|
||||||
|
To enable custom labels, you need to add the following configuration to the [custom_labels file](./../pr_agent/settings/custom_labels.toml):
|
||||||
|
- Change `enable_custom_labels` to True: This will turn off the default labels and enable the custom labels provided in the custom_labels.toml file.
|
||||||
|
- Add the custom labels to the custom_labels.toml file. It should be formatted as follows:
|
||||||
|
```
|
||||||
|
[custom_labels."Custom Label Name"]
|
||||||
|
description = "Description of when AI should suggest this label"
|
||||||
|
```
|
||||||
|
- You can add modify the list to include all the custom labels you wish to use in your repository.
|
||||||
|
|
||||||
|
#### Github Action
|
||||||
|
To use the `generte_labels` tool with Github Action:
|
||||||
|
|
||||||
|
- Add the following file to your repository under `env` section in `.github/workflows/pr_agent.yml`
|
||||||
|
- Comma separated list of custom labels and their descriptions
|
||||||
|
- The number of labels and descriptions should be the same and in the same order (empty descriptions are allowed):
|
||||||
|
```
|
||||||
|
CUSTOM_LABELS: "label1, label2, ..."
|
||||||
|
CUSTOM_LABELS_DESCRIPTION: "label1 description, label2 description, ..."
|
||||||
|
```
|
BIN
pics/custom_label_published.png
Normal file
BIN
pics/custom_label_published.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 253 KiB |
BIN
pics/custom_labels_list.png
Normal file
BIN
pics/custom_labels_list.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 84 KiB |
@ -7,6 +7,7 @@ from pr_agent.tools.pr_add_docs import PRAddDocs
|
|||||||
from pr_agent.tools.pr_code_suggestions import PRCodeSuggestions
|
from pr_agent.tools.pr_code_suggestions import PRCodeSuggestions
|
||||||
from pr_agent.tools.pr_config import PRConfig
|
from pr_agent.tools.pr_config import PRConfig
|
||||||
from pr_agent.tools.pr_description import PRDescription
|
from pr_agent.tools.pr_description import PRDescription
|
||||||
|
from pr_agent.tools.pr_generate_labels import PRGenerateLabels
|
||||||
from pr_agent.tools.pr_information_from_user import PRInformationFromUser
|
from pr_agent.tools.pr_information_from_user import PRInformationFromUser
|
||||||
from pr_agent.tools.pr_questions import PRQuestions
|
from pr_agent.tools.pr_questions import PRQuestions
|
||||||
from pr_agent.tools.pr_reviewer import PRReviewer
|
from pr_agent.tools.pr_reviewer import PRReviewer
|
||||||
@ -31,6 +32,7 @@ command2class = {
|
|||||||
"settings": PRConfig,
|
"settings": PRConfig,
|
||||||
"similar_issue": PRSimilarIssue,
|
"similar_issue": PRSimilarIssue,
|
||||||
"add_docs": PRAddDocs,
|
"add_docs": PRAddDocs,
|
||||||
|
"generate_labels": PRGenerateLabels,
|
||||||
}
|
}
|
||||||
|
|
||||||
commands = list(command2class.keys())
|
commands = list(command2class.keys())
|
||||||
|
@ -304,3 +304,19 @@ def try_fix_yaml(review_text: str) -> dict:
|
|||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
return data
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
def set_custom_labels(variables):
|
||||||
|
labels = get_settings().custom_labels
|
||||||
|
if not labels:
|
||||||
|
# set default labels
|
||||||
|
labels = ['Bug fix', 'Tests', 'Bug fix with tests', 'Refactoring', 'Enhancement', 'Documentation', 'Other']
|
||||||
|
labels_list = "\n - ".join(labels) if labels else ""
|
||||||
|
labels_list = f" - {labels_list}" if labels_list else ""
|
||||||
|
variables["custom_labels"] = labels_list
|
||||||
|
return
|
||||||
|
final_labels = ""
|
||||||
|
for k, v in labels.items():
|
||||||
|
final_labels += f" - {k} ({v['description']})\n"
|
||||||
|
variables["custom_labels"] = final_labels
|
||||||
|
variables["custom_labels_examples"] = f" - {list(labels.keys())[0]}"
|
||||||
|
@ -23,8 +23,10 @@ global_settings = Dynaconf(
|
|||||||
"settings/pr_sort_code_suggestions_prompts.toml",
|
"settings/pr_sort_code_suggestions_prompts.toml",
|
||||||
"settings/pr_information_from_user_prompts.toml",
|
"settings/pr_information_from_user_prompts.toml",
|
||||||
"settings/pr_update_changelog_prompts.toml",
|
"settings/pr_update_changelog_prompts.toml",
|
||||||
|
"settings/pr_custom_labels.toml",
|
||||||
"settings/pr_add_docs.toml",
|
"settings/pr_add_docs.toml",
|
||||||
"settings_prod/.secrets.toml"
|
"settings_prod/.secrets.toml",
|
||||||
|
"settings/custom_labels.toml"
|
||||||
]]
|
]]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -17,8 +17,11 @@ async def run_action():
|
|||||||
OPENAI_KEY = os.environ.get('OPENAI_KEY') or os.environ.get('OPENAI.KEY')
|
OPENAI_KEY = os.environ.get('OPENAI_KEY') or os.environ.get('OPENAI.KEY')
|
||||||
OPENAI_ORG = os.environ.get('OPENAI_ORG') or os.environ.get('OPENAI.ORG')
|
OPENAI_ORG = os.environ.get('OPENAI_ORG') or os.environ.get('OPENAI.ORG')
|
||||||
GITHUB_TOKEN = os.environ.get('GITHUB_TOKEN')
|
GITHUB_TOKEN = os.environ.get('GITHUB_TOKEN')
|
||||||
get_settings().set("CONFIG.PUBLISH_OUTPUT_PROGRESS", False)
|
CUSTOM_LABELS = os.environ.get('CUSTOM_LABELS')
|
||||||
|
CUSTOM_LABELS_DESCRIPTIONS = os.environ.get('CUSTOM_LABELS_DESCRIPTIONS')
|
||||||
|
# CUSTOM_LABELS is a comma separated list of labels (string), convert to list and strip spaces
|
||||||
|
|
||||||
|
get_settings().set("CONFIG.PUBLISH_OUTPUT_PROGRESS", False)
|
||||||
|
|
||||||
# Check if required environment variables are set
|
# Check if required environment variables are set
|
||||||
if not GITHUB_EVENT_NAME:
|
if not GITHUB_EVENT_NAME:
|
||||||
@ -33,6 +36,7 @@ async def run_action():
|
|||||||
if not GITHUB_TOKEN:
|
if not GITHUB_TOKEN:
|
||||||
print("GITHUB_TOKEN not set")
|
print("GITHUB_TOKEN not set")
|
||||||
return
|
return
|
||||||
|
CUSTOM_LABELS_DICT = handle_custom_labels(CUSTOM_LABELS, CUSTOM_LABELS_DESCRIPTIONS)
|
||||||
|
|
||||||
# Set the environment variables in the settings
|
# Set the environment variables in the settings
|
||||||
get_settings().set("OPENAI.KEY", OPENAI_KEY)
|
get_settings().set("OPENAI.KEY", OPENAI_KEY)
|
||||||
@ -40,6 +44,7 @@ async def run_action():
|
|||||||
get_settings().set("OPENAI.ORG", OPENAI_ORG)
|
get_settings().set("OPENAI.ORG", OPENAI_ORG)
|
||||||
get_settings().set("GITHUB.USER_TOKEN", GITHUB_TOKEN)
|
get_settings().set("GITHUB.USER_TOKEN", GITHUB_TOKEN)
|
||||||
get_settings().set("GITHUB.DEPLOYMENT_TYPE", "user")
|
get_settings().set("GITHUB.DEPLOYMENT_TYPE", "user")
|
||||||
|
get_settings().set("CUSTOM_LABELS", CUSTOM_LABELS_DICT)
|
||||||
|
|
||||||
# Load the event payload
|
# Load the event payload
|
||||||
try:
|
try:
|
||||||
@ -88,5 +93,31 @@ async def run_action():
|
|||||||
await PRAgent().handle_request(url, body)
|
await PRAgent().handle_request(url, body)
|
||||||
|
|
||||||
|
|
||||||
|
def handle_custom_labels(CUSTOM_LABELS, CUSTOM_LABELS_DESCRIPTIONS):
|
||||||
|
if CUSTOM_LABELS:
|
||||||
|
CUSTOM_LABELS = [x.strip() for x in CUSTOM_LABELS.split(',')]
|
||||||
|
else:
|
||||||
|
# Set default labels
|
||||||
|
CUSTOM_LABELS = ['Bug fix', 'Tests', 'Bug fix with tests', 'Refactoring', 'Enhancement', 'Documentation',
|
||||||
|
'Other']
|
||||||
|
print(f"Using default labels: {CUSTOM_LABELS}")
|
||||||
|
if CUSTOM_LABELS_DESCRIPTIONS:
|
||||||
|
CUSTOM_LABELS_DESCRIPTIONS = [x.strip() for x in CUSTOM_LABELS_DESCRIPTIONS.split(',')]
|
||||||
|
else:
|
||||||
|
# Set default labels
|
||||||
|
CUSTOM_LABELS_DESCRIPTIONS = ['Fixes a bug in the code', 'Adds or modifies tests',
|
||||||
|
'Fixes a bug in the code and adds or modifies tests',
|
||||||
|
'Refactors the code without changing its functionality',
|
||||||
|
'Adds new features or functionality',
|
||||||
|
'Adds or modifies documentation',
|
||||||
|
'Other changes that do not fit in any of the above categories']
|
||||||
|
print(f"Using default labels: {CUSTOM_LABELS_DESCRIPTIONS}")
|
||||||
|
# create a dictionary of labels and descriptions
|
||||||
|
CUSTOM_LABELS_DICT = dict()
|
||||||
|
for i in range(len(CUSTOM_LABELS)):
|
||||||
|
CUSTOM_LABELS_DICT[CUSTOM_LABELS[i]] = {'description': CUSTOM_LABELS_DESCRIPTIONS[i]}
|
||||||
|
return CUSTOM_LABELS_DICT
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
asyncio.run(run_action())
|
asyncio.run(run_action())
|
@ -33,10 +33,13 @@ add_original_user_description=false
|
|||||||
keep_original_user_title=false
|
keep_original_user_title=false
|
||||||
use_bullet_points=true
|
use_bullet_points=true
|
||||||
extra_instructions = ""
|
extra_instructions = ""
|
||||||
|
|
||||||
# markers
|
# markers
|
||||||
use_description_markers=false
|
use_description_markers=false
|
||||||
include_generated_by_header=true
|
include_generated_by_header=true
|
||||||
|
|
||||||
|
#custom_labels = ['Bug fix', 'Tests', 'Bug fix with tests', 'Refactoring', 'Enhancement', 'Documentation', 'Other']
|
||||||
|
|
||||||
[pr_questions] # /ask #
|
[pr_questions] # /ask #
|
||||||
|
|
||||||
[pr_code_suggestions] # /improve #
|
[pr_code_suggestions] # /improve #
|
||||||
|
16
pr_agent/settings/custom_labels.toml
Normal file
16
pr_agent/settings/custom_labels.toml
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
enable_custom_labels=false
|
||||||
|
|
||||||
|
[custom_labels."Bug fix"]
|
||||||
|
description = "Fixes a bug in the code"
|
||||||
|
[custom_labels."Tests"]
|
||||||
|
description = "Adds or modifies tests"
|
||||||
|
[custom_labels."Bug fix with tests"]
|
||||||
|
description = "Fixes a bug in the code and adds or modifies tests"
|
||||||
|
[custom_labels."Refactoring"]
|
||||||
|
description = "Code refactoring without changing functionality"
|
||||||
|
[custom_labels."Enhancement"]
|
||||||
|
description = "Adds new features or functionality"
|
||||||
|
[custom_labels."Documentation"]
|
||||||
|
description = "Adds or modifies documentation"
|
||||||
|
[custom_labels."Other"]
|
||||||
|
description = "Other changes that do not fit in any of the above categories"
|
72
pr_agent/settings/pr_custom_labels.toml
Normal file
72
pr_agent/settings/pr_custom_labels.toml
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
[pr_custom_labels_prompt]
|
||||||
|
system="""You are CodiumAI-PR-Reviewer, a language model designed to review git pull requests.
|
||||||
|
Your task is to label the type of the PR content.
|
||||||
|
- Make sure not to focus the new PR code (the '+' lines).
|
||||||
|
- If needed, each YAML output should be in block scalar format ('|-')
|
||||||
|
{%- if extra_instructions %}
|
||||||
|
|
||||||
|
Extra instructions from the user:
|
||||||
|
'
|
||||||
|
{{ extra_instructions }}
|
||||||
|
'
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
You must use the following YAML schema to format your answer:
|
||||||
|
```yaml
|
||||||
|
PR Type:
|
||||||
|
type: array
|
||||||
|
{%- if enable_custom_labels %}
|
||||||
|
description: One or more labels that describe the PR type. Don't output the description in the parentheses.
|
||||||
|
{%- endif %}
|
||||||
|
items:
|
||||||
|
type: string
|
||||||
|
enum:
|
||||||
|
{%- if enable_custom_labels %}
|
||||||
|
{{ custom_labels }}
|
||||||
|
{%- else %}
|
||||||
|
- Bug fix
|
||||||
|
- Tests
|
||||||
|
- Refactoring
|
||||||
|
- Enhancement
|
||||||
|
- Documentation
|
||||||
|
- Other
|
||||||
|
{%- endif %}
|
||||||
|
|
||||||
|
Example output:
|
||||||
|
```yaml
|
||||||
|
PR Type:
|
||||||
|
{%- if enable_custom_labels %}
|
||||||
|
{{ custom_labels_examples }}
|
||||||
|
{%- else %}
|
||||||
|
- Bug fix
|
||||||
|
- Tests
|
||||||
|
{%- endif %}
|
||||||
|
```
|
||||||
|
|
||||||
|
Make sure to output a valid YAML. Don't repeat the prompt in the answer, and avoid outputting the 'type' and 'description' fields.
|
||||||
|
"""
|
||||||
|
|
||||||
|
user="""PR Info:
|
||||||
|
Previous title: '{{title}}'
|
||||||
|
Previous description: '{{description}}'
|
||||||
|
Branch: '{{branch}}'
|
||||||
|
{%- if language %}
|
||||||
|
|
||||||
|
Main language: {{language}}
|
||||||
|
{%- endif %}
|
||||||
|
{%- if commit_messages_str %}
|
||||||
|
|
||||||
|
Commit messages:
|
||||||
|
{{commit_messages_str}}
|
||||||
|
{%- endif %}
|
||||||
|
|
||||||
|
|
||||||
|
The PR Git Diff:
|
||||||
|
```
|
||||||
|
{{diff}}
|
||||||
|
```
|
||||||
|
Note that lines in the diff body are prefixed with a symbol that represents the type of change: '-' for deletions, '+' for additions, and ' ' (a space) for unchanged lines.
|
||||||
|
|
||||||
|
Response (should be a valid YAML, and nothing else):
|
||||||
|
```yaml
|
||||||
|
"""
|
@ -19,16 +19,22 @@ PR Title:
|
|||||||
description: an informative title for the PR, describing its main theme
|
description: an informative title for the PR, describing its main theme
|
||||||
PR Type:
|
PR Type:
|
||||||
type: array
|
type: array
|
||||||
|
{%- if enable_custom_labels %}
|
||||||
|
description: One or more labels that describe the PR type. Don't output the description in the parentheses.
|
||||||
|
{%- endif %}
|
||||||
items:
|
items:
|
||||||
type: string
|
type: string
|
||||||
enum:
|
enum:
|
||||||
|
{%- if enable_custom_labels %}
|
||||||
|
{{ custom_labels }}
|
||||||
|
{%- else %}
|
||||||
- Bug fix
|
- Bug fix
|
||||||
- Tests
|
- Tests
|
||||||
- Bug fix with tests
|
|
||||||
- Refactoring
|
- Refactoring
|
||||||
- Enhancement
|
- Enhancement
|
||||||
- Documentation
|
- Documentation
|
||||||
- Other
|
- Other
|
||||||
|
{%- endif %}
|
||||||
PR Description:
|
PR Description:
|
||||||
type: string
|
type: string
|
||||||
description: an informative and concise description of the PR.
|
description: an informative and concise description of the PR.
|
||||||
@ -52,7 +58,11 @@ Example output:
|
|||||||
PR Title: |-
|
PR Title: |-
|
||||||
...
|
...
|
||||||
PR Type:
|
PR Type:
|
||||||
|
{%- if enable_custom_labels %}
|
||||||
|
{{ custom_labels_examples }}
|
||||||
|
{%- else %}
|
||||||
- Bug fix
|
- Bug fix
|
||||||
|
{%- endif %}
|
||||||
PR Description: |-
|
PR Description: |-
|
||||||
...
|
...
|
||||||
PR Main Files Walkthrough:
|
PR Main Files Walkthrough:
|
||||||
|
@ -51,13 +51,22 @@ PR Analysis:
|
|||||||
description: summary of the PR in 2-3 sentences.
|
description: summary of the PR in 2-3 sentences.
|
||||||
Type of PR:
|
Type of PR:
|
||||||
type: string
|
type: string
|
||||||
|
{%- if enable_custom_labels %}
|
||||||
|
description: One or more labels that describe the PR type. Don't output the description in the parentheses.
|
||||||
|
{%- endif %}
|
||||||
|
items:
|
||||||
|
type: string
|
||||||
enum:
|
enum:
|
||||||
|
{%- if enable_custom_labels %}
|
||||||
|
{{ custom_labels }}
|
||||||
|
{%- else %}
|
||||||
- Bug fix
|
- Bug fix
|
||||||
- Tests
|
- Tests
|
||||||
- Refactoring
|
- Refactoring
|
||||||
- Enhancement
|
- Enhancement
|
||||||
- Documentation
|
- Documentation
|
||||||
- Other
|
- Other
|
||||||
|
{%- endif %}
|
||||||
{%- if require_score %}
|
{%- if require_score %}
|
||||||
Score:
|
Score:
|
||||||
type: int
|
type: int
|
||||||
|
@ -7,7 +7,7 @@ from jinja2 import Environment, StrictUndefined
|
|||||||
from pr_agent.algo.ai_handler import AiHandler
|
from pr_agent.algo.ai_handler import AiHandler
|
||||||
from pr_agent.algo.pr_processing import get_pr_diff, retry_with_fallback_models
|
from pr_agent.algo.pr_processing import get_pr_diff, retry_with_fallback_models
|
||||||
from pr_agent.algo.token_handler import TokenHandler
|
from pr_agent.algo.token_handler import TokenHandler
|
||||||
from pr_agent.algo.utils import load_yaml
|
from pr_agent.algo.utils import load_yaml, set_custom_labels
|
||||||
from pr_agent.config_loader import get_settings
|
from pr_agent.config_loader import get_settings
|
||||||
from pr_agent.git_providers import get_git_provider
|
from pr_agent.git_providers import get_git_provider
|
||||||
from pr_agent.git_providers.git_provider import get_main_pr_language
|
from pr_agent.git_providers.git_provider import get_main_pr_language
|
||||||
@ -42,7 +42,10 @@ class PRDescription:
|
|||||||
"diff": "", # empty diff for initial calculation
|
"diff": "", # empty diff for initial calculation
|
||||||
"use_bullet_points": get_settings().pr_description.use_bullet_points,
|
"use_bullet_points": get_settings().pr_description.use_bullet_points,
|
||||||
"extra_instructions": get_settings().pr_description.extra_instructions,
|
"extra_instructions": get_settings().pr_description.extra_instructions,
|
||||||
"commit_messages_str": self.git_provider.get_commit_messages()
|
"commit_messages_str": self.git_provider.get_commit_messages(),
|
||||||
|
"enable_custom_labels": get_settings().enable_custom_labels,
|
||||||
|
"custom_labels": "",
|
||||||
|
"custom_labels_examples": "",
|
||||||
}
|
}
|
||||||
|
|
||||||
self.user_description = self.git_provider.get_user_description()
|
self.user_description = self.git_provider.get_user_description()
|
||||||
@ -140,6 +143,7 @@ class PRDescription:
|
|||||||
variables["diff"] = self.patches_diff # update diff
|
variables["diff"] = self.patches_diff # update diff
|
||||||
|
|
||||||
environment = Environment(undefined=StrictUndefined)
|
environment = Environment(undefined=StrictUndefined)
|
||||||
|
set_custom_labels(variables)
|
||||||
system_prompt = environment.from_string(get_settings().pr_description_prompt.system).render(variables)
|
system_prompt = environment.from_string(get_settings().pr_description_prompt.system).render(variables)
|
||||||
user_prompt = environment.from_string(get_settings().pr_description_prompt.user).render(variables)
|
user_prompt = environment.from_string(get_settings().pr_description_prompt.user).render(variables)
|
||||||
|
|
||||||
@ -156,7 +160,6 @@ class PRDescription:
|
|||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
|
||||||
def _prepare_data(self):
|
def _prepare_data(self):
|
||||||
# Load the AI prediction data into a dictionary
|
# Load the AI prediction data into a dictionary
|
||||||
self.data = load_yaml(self.prediction.strip())
|
self.data = load_yaml(self.prediction.strip())
|
||||||
|
163
pr_agent/tools/pr_generate_labels.py
Normal file
163
pr_agent/tools/pr_generate_labels.py
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
import copy
|
||||||
|
import re
|
||||||
|
from typing import List, Tuple
|
||||||
|
|
||||||
|
from jinja2 import Environment, StrictUndefined
|
||||||
|
|
||||||
|
from pr_agent.algo.ai_handler import AiHandler
|
||||||
|
from pr_agent.algo.pr_processing import get_pr_diff, retry_with_fallback_models
|
||||||
|
from pr_agent.algo.token_handler import TokenHandler
|
||||||
|
from pr_agent.algo.utils import load_yaml, set_custom_labels
|
||||||
|
from pr_agent.config_loader import get_settings
|
||||||
|
from pr_agent.git_providers import get_git_provider
|
||||||
|
from pr_agent.git_providers.git_provider import get_main_pr_language
|
||||||
|
from pr_agent.log import get_logger
|
||||||
|
|
||||||
|
|
||||||
|
class PRGenerateLabels:
|
||||||
|
def __init__(self, pr_url: str, args: list = None):
|
||||||
|
"""
|
||||||
|
Initialize the PRGenerateLabels object with the necessary attributes and objects for generating labels
|
||||||
|
corresponding to the PR using an AI model.
|
||||||
|
Args:
|
||||||
|
pr_url (str): The URL of the pull request.
|
||||||
|
args (list, optional): List of arguments passed to the PRGenerateLabels class. Defaults to None.
|
||||||
|
"""
|
||||||
|
# Initialize the git provider and main PR language
|
||||||
|
self.git_provider = get_git_provider()(pr_url)
|
||||||
|
self.main_pr_language = get_main_pr_language(
|
||||||
|
self.git_provider.get_languages(), self.git_provider.get_files()
|
||||||
|
)
|
||||||
|
self.pr_id = self.git_provider.get_pr_id()
|
||||||
|
|
||||||
|
# Initialize the AI handler
|
||||||
|
self.ai_handler = AiHandler()
|
||||||
|
|
||||||
|
# Initialize the variables dictionary
|
||||||
|
self.vars = {
|
||||||
|
"title": self.git_provider.pr.title,
|
||||||
|
"branch": self.git_provider.get_pr_branch(),
|
||||||
|
"description": self.git_provider.get_pr_description(full=False),
|
||||||
|
"language": self.main_pr_language,
|
||||||
|
"diff": "", # empty diff for initial calculation
|
||||||
|
"use_bullet_points": get_settings().pr_description.use_bullet_points,
|
||||||
|
"extra_instructions": get_settings().pr_description.extra_instructions,
|
||||||
|
"commit_messages_str": self.git_provider.get_commit_messages(),
|
||||||
|
"custom_labels": "",
|
||||||
|
"custom_labels_examples": "",
|
||||||
|
"enable_custom_labels": get_settings().enable_custom_labels,
|
||||||
|
}
|
||||||
|
|
||||||
|
# Initialize the token handler
|
||||||
|
self.token_handler = TokenHandler(
|
||||||
|
self.git_provider.pr,
|
||||||
|
self.vars,
|
||||||
|
get_settings().pr_custom_labels_prompt.system,
|
||||||
|
get_settings().pr_custom_labels_prompt.user,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Initialize patches_diff and prediction attributes
|
||||||
|
self.patches_diff = None
|
||||||
|
self.prediction = None
|
||||||
|
|
||||||
|
async def run(self):
|
||||||
|
"""
|
||||||
|
Generates a PR labels using an AI model and publishes it to the PR.
|
||||||
|
"""
|
||||||
|
|
||||||
|
try:
|
||||||
|
get_logger().info(f"Generating a PR labels {self.pr_id}")
|
||||||
|
if get_settings().config.publish_output:
|
||||||
|
self.git_provider.publish_comment("Preparing PR labels...", is_temporary=True)
|
||||||
|
|
||||||
|
await retry_with_fallback_models(self._prepare_prediction)
|
||||||
|
|
||||||
|
get_logger().info(f"Preparing answer {self.pr_id}")
|
||||||
|
if self.prediction:
|
||||||
|
self._prepare_data()
|
||||||
|
else:
|
||||||
|
return None
|
||||||
|
|
||||||
|
pr_labels = self._prepare_labels()
|
||||||
|
|
||||||
|
if get_settings().config.publish_output:
|
||||||
|
get_logger().info(f"Pushing labels {self.pr_id}")
|
||||||
|
if self.git_provider.is_supported("get_labels"):
|
||||||
|
current_labels = self.git_provider.get_labels()
|
||||||
|
if current_labels is None:
|
||||||
|
current_labels = []
|
||||||
|
self.git_provider.publish_labels(pr_labels + current_labels)
|
||||||
|
self.git_provider.remove_initial_comment()
|
||||||
|
except Exception as e:
|
||||||
|
get_logger().error(f"Error generating PR labels {self.pr_id}: {e}")
|
||||||
|
|
||||||
|
return ""
|
||||||
|
|
||||||
|
async def _prepare_prediction(self, model: str) -> None:
|
||||||
|
"""
|
||||||
|
Prepare the AI prediction for the PR labels based on the provided model.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
model (str): The name of the model to be used for generating the prediction.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
None
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
Any exceptions raised by the 'get_pr_diff' and '_get_prediction' functions.
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
get_logger().info(f"Getting PR diff {self.pr_id}")
|
||||||
|
self.patches_diff = get_pr_diff(self.git_provider, self.token_handler, model)
|
||||||
|
get_logger().info(f"Getting AI prediction {self.pr_id}")
|
||||||
|
self.prediction = await self._get_prediction(model)
|
||||||
|
|
||||||
|
async def _get_prediction(self, model: str) -> str:
|
||||||
|
"""
|
||||||
|
Generate an AI prediction for the PR labels based on the provided model.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
model (str): The name of the model to be used for generating the prediction.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: The generated AI prediction.
|
||||||
|
"""
|
||||||
|
variables = copy.deepcopy(self.vars)
|
||||||
|
variables["diff"] = self.patches_diff # update diff
|
||||||
|
|
||||||
|
environment = Environment(undefined=StrictUndefined)
|
||||||
|
set_custom_labels(variables)
|
||||||
|
system_prompt = environment.from_string(get_settings().pr_custom_labels_prompt.system).render(variables)
|
||||||
|
user_prompt = environment.from_string(get_settings().pr_custom_labels_prompt.user).render(variables)
|
||||||
|
|
||||||
|
if get_settings().config.verbosity_level >= 2:
|
||||||
|
get_logger().info(f"\nSystem prompt:\n{system_prompt}")
|
||||||
|
get_logger().info(f"\nUser prompt:\n{user_prompt}")
|
||||||
|
|
||||||
|
response, finish_reason = await self.ai_handler.chat_completion(
|
||||||
|
model=model,
|
||||||
|
temperature=0.2,
|
||||||
|
system=system_prompt,
|
||||||
|
user=user_prompt
|
||||||
|
)
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
|
def _prepare_data(self):
|
||||||
|
# Load the AI prediction data into a dictionary
|
||||||
|
self.data = load_yaml(self.prediction.strip())
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def _prepare_labels(self) -> List[str]:
|
||||||
|
pr_types = []
|
||||||
|
|
||||||
|
# If the 'PR Type' key is present in the dictionary, split its value by comma and assign it to 'pr_types'
|
||||||
|
if 'PR Type' in self.data:
|
||||||
|
if type(self.data['PR Type']) == list:
|
||||||
|
pr_types = self.data['PR Type']
|
||||||
|
elif type(self.data['PR Type']) == str:
|
||||||
|
pr_types = self.data['PR Type'].split(',')
|
||||||
|
|
||||||
|
return pr_types
|
@ -9,7 +9,7 @@ from yaml import SafeLoader
|
|||||||
from pr_agent.algo.ai_handler import AiHandler
|
from pr_agent.algo.ai_handler import AiHandler
|
||||||
from pr_agent.algo.pr_processing import get_pr_diff, retry_with_fallback_models
|
from pr_agent.algo.pr_processing import get_pr_diff, retry_with_fallback_models
|
||||||
from pr_agent.algo.token_handler import TokenHandler
|
from pr_agent.algo.token_handler import TokenHandler
|
||||||
from pr_agent.algo.utils import convert_to_markdown, load_yaml, try_fix_yaml
|
from pr_agent.algo.utils import convert_to_markdown, load_yaml, try_fix_yaml, set_custom_labels
|
||||||
from pr_agent.config_loader import get_settings
|
from pr_agent.config_loader import get_settings
|
||||||
from pr_agent.git_providers import get_git_provider
|
from pr_agent.git_providers import get_git_provider
|
||||||
from pr_agent.git_providers.git_provider import IncrementalPR, get_main_pr_language
|
from pr_agent.git_providers.git_provider import IncrementalPR, get_main_pr_language
|
||||||
@ -63,6 +63,8 @@ class PRReviewer:
|
|||||||
'answer_str': answer_str,
|
'answer_str': answer_str,
|
||||||
"extra_instructions": get_settings().pr_reviewer.extra_instructions,
|
"extra_instructions": get_settings().pr_reviewer.extra_instructions,
|
||||||
"commit_messages_str": self.git_provider.get_commit_messages(),
|
"commit_messages_str": self.git_provider.get_commit_messages(),
|
||||||
|
"custom_labels": "",
|
||||||
|
"enable_custom_labels": get_settings().enable_custom_labels,
|
||||||
}
|
}
|
||||||
|
|
||||||
self.token_handler = TokenHandler(
|
self.token_handler = TokenHandler(
|
||||||
@ -149,6 +151,7 @@ class PRReviewer:
|
|||||||
variables["diff"] = self.patches_diff # update diff
|
variables["diff"] = self.patches_diff # update diff
|
||||||
|
|
||||||
environment = Environment(undefined=StrictUndefined)
|
environment = Environment(undefined=StrictUndefined)
|
||||||
|
set_custom_labels(variables)
|
||||||
system_prompt = environment.from_string(get_settings().pr_review_prompt.system).render(variables)
|
system_prompt = environment.from_string(get_settings().pr_review_prompt.system).render(variables)
|
||||||
user_prompt = environment.from_string(get_settings().pr_review_prompt.user).render(variables)
|
user_prompt = environment.from_string(get_settings().pr_review_prompt.user).render(variables)
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user