From efdd71ba33bbeac6642fbf9a0c7f5c5271e8b674 Mon Sep 17 00:00:00 2001 From: Bentlybro Date: Tue, 10 Sep 2024 19:17:44 +0100 Subject: [PATCH 01/43] Add Github blocks + "DEVELOPER_TOOLS" category --- .../autogpt_server/blocks/github_blocks.py | 1903 +++++++++++++++++ .../autogpt_server/data/block.py | 1 + 2 files changed, 1904 insertions(+) create mode 100644 rnd/autogpt_server/autogpt_server/blocks/github_blocks.py diff --git a/rnd/autogpt_server/autogpt_server/blocks/github_blocks.py b/rnd/autogpt_server/autogpt_server/blocks/github_blocks.py new file mode 100644 index 000000000000..4967dd65498d --- /dev/null +++ b/rnd/autogpt_server/autogpt_server/blocks/github_blocks.py @@ -0,0 +1,1903 @@ +import base64 + +import requests +from pydantic import BaseModel, ConfigDict, Field + +from autogpt_server.data.block import Block, BlockCategory, BlockOutput, BlockSchema +from autogpt_server.data.model import BlockSecret, SchemaField, SecretField + + +class GithubCredentials(BaseModel): + github_oauth_token: BlockSecret = SecretField(key="github_oauth_token") + + model_config = ConfigDict(title="GitHub Credentials") + + +class GithubCommentBlock(Block): + class Input(BlockSchema): + issue_url: str = SchemaField( + description="URL of the GitHub issue or pull request", + placeholder="https://github.com/owner/repo/issues/1", + ) + comment: str = SchemaField( + description="Comment to post on the issue or pull request", + placeholder="Enter your comment", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the comment posting operation") + error: str = SchemaField( + description="Error message if the comment posting failed" + ) + + def __init__(self): + super().__init__( + id="0001c3d4-5678-90ef-1234-567890abcdef", + description="This block posts a comment on a specified GitHub issue or pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubCommentBlock.Input, + output_schema=GithubCommentBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "comment": "This is a test comment.", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Comment posted successfully")], + test_mock={ + "post_comment": lambda *args, **kwargs: "Comment posted successfully" + }, + ) + + @staticmethod + def post_comment(creds: GithubCredentials, issue_url: str, comment: str) -> str: + try: + if "/pull/" in issue_url: + api_url = ( + issue_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/issues/" + ) + + "/comments" + ) + else: + api_url = ( + issue_url.replace("github.com", "api.github.com/repos") + + "/comments" + ) + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + data = {"body": comment} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Comment posted successfully" + except Exception as e: + return f"Failed to post comment: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.post_comment( + input_data.creds, + input_data.issue_url, + input_data.comment, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubMakeIssueBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + title: str = SchemaField( + description="Title of the issue", placeholder="Enter the issue title" + ) + body: str = SchemaField( + description="Body of the issue", placeholder="Enter the issue body" + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the issue creation operation") + error: str = SchemaField( + description="Error message if the issue creation failed" + ) + + def __init__(self): + super().__init__( + id="0002d3e4-5678-90ab-1234-567890abcdef", + description="This block creates a new issue on a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubMakeIssueBlock.Input, + output_schema=GithubMakeIssueBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "title": "Test Issue", + "body": "This is a test issue.", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Issue created successfully")], + test_mock={ + "create_issue": lambda *args, **kwargs: "Issue created successfully" + }, + ) + + @staticmethod + def create_issue( + creds: GithubCredentials, repo_url: str, title: str, body: str + ) -> str: + try: + api_url = repo_url.replace("github.com", "api.github.com/repos") + "/issues" + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + data = {"title": title, "body": body} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Issue created successfully" + except Exception as e: + return f"Failed to create issue: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.create_issue( + input_data.creds, + input_data.repo_url, + input_data.title, + input_data.body, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubMakePRBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + title: str = SchemaField( + description="Title of the pull request", + placeholder="Enter the pull request title", + ) + body: str = SchemaField( + description="Body of the pull request", + placeholder="Enter the pull request body", + ) + head: str = SchemaField( + description="The name of the branch where your changes are implemented. For cross-repository pull requests in the same network, namespace head with a user like this: username:branch.", + placeholder="Enter the head branch", + ) + base: str = SchemaField( + description="The name of the branch you want the changes pulled into.", + placeholder="Enter the base branch", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the pull request creation operation" + ) + error: str = SchemaField( + description="Error message if the pull request creation failed" + ) + + def __init__(self): + super().__init__( + id="0003q3r4-5678-90ab-1234-567890abcdef", + description="This block creates a new pull request on a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubMakePRBlock.Input, + output_schema=GithubMakePRBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "title": "Test Pull Request", + "body": "This is a test pull request.", + "head": "feature-branch", + "base": "main", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Pull request created successfully")], + test_mock={ + "create_pr": lambda *args, **kwargs: "Pull request created successfully" + }, + ) + + @staticmethod + def create_pr( + creds: GithubCredentials, + repo_url: str, + title: str, + body: str, + head: str, + base: str, + ) -> str: + response = None + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/pulls" + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + data = {"title": title, "body": body, "head": head, "base": base} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Pull request created successfully" + except requests.exceptions.HTTPError as http_err: + if response and response.status_code == 422: + error_details = response.json() + return f"Failed to create pull request: {error_details.get('message', 'Unknown error')}" + return f"Failed to create pull request: {str(http_err)}" + except Exception as e: + return f"Failed to create pull request: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.create_pr( + input_data.creds, + input_data.repo_url, + input_data.title, + input_data.body, + input_data.head, + input_data.base, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubReadIssueBlock(Block): + class Input(BlockSchema): + issue_url: str = SchemaField( + description="URL of the GitHub issue", + placeholder="https://github.com/owner/repo/issues/1", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + title: str = SchemaField(description="Title of the issue") + body: str = SchemaField(description="Body of the issue") + user: str = SchemaField(description="User who created the issue") + error: str = SchemaField( + description="Error message if reading the issue failed" + ) + + def __init__(self): + super().__init__( + id="0004e3f4-5678-90ab-1234-567890abcdef", + description="This block reads the body, title, and user of a specified GitHub issue using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadIssueBlock.Input, + output_schema=GithubReadIssueBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[ + ("title", "Title of the issue"), + ("body", "This is the body of the issue."), + ("user", "username") + ], + test_mock={ + "read_issue": lambda *args, **kwargs: ( + "Title of the issue", + "This is the body of the issue.", + "username" + ) + }, + ) + + @staticmethod + def read_issue(creds: GithubCredentials, issue_url: str) -> tuple[str, str, str]: + try: + api_url = issue_url.replace("github.com", "api.github.com/repos") + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + title = data.get("title", "No title found") + body = data.get("body", "No body content found") + user = data.get("user", {}).get("login", "No user found") + + return title, body, user + except Exception as e: + return f"Failed to read issue: {str(e)}", "", "" + + def run(self, input_data: Input) -> BlockOutput: + title, body, user = self.read_issue( + input_data.creds, + input_data.issue_url, + ) + if "Failed" in title: + yield "error", title + else: + yield "title", title + yield "body", body + yield "user", user + + +class GithubReadPRBlock(Block): + class Input(BlockSchema): + pr_url: str = SchemaField( + description="URL of the GitHub pull request", + placeholder="https://github.com/owner/repo/pull/1", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + include_pr_changes: bool = SchemaField( + description="Whether to include the changes made in the pull request", + default=False, + ) + + class Output(BlockSchema): + title: str = SchemaField(description="Title of the pull request") + body: str = SchemaField(description="Body of the pull request") + user: str = SchemaField(description="User who created the pull request") + changes: str = SchemaField(description="Changes made in the pull request") + error: str = SchemaField( + description="Error message if reading the pull request failed" + ) + + def __init__(self): + super().__init__( + id="0005g3h4-5678-90ab-1234-567890abcdeg", + description="This block reads the body, title, user, and changes of a specified GitHub pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadPRBlock.Input, + output_schema=GithubReadPRBlock.Output, + test_input={ + "pr_url": "https://github.com/owner/repo/pull/1", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + "include_pr_changes": True, + }, + test_output=[ + ("title", "Title of the pull request"), + ("body", "This is the body of the pull request."), + ("user", "username"), + ("changes", "List of changes made in the pull request."), + ], + test_mock={ + "read_pr": lambda *args, **kwargs: ( + "Title of the pull request", + "This is the body of the pull request.", + "username" + ), + "read_pr_changes": lambda *args, **kwargs: "List of changes made in the pull request.", + }, + ) + + @staticmethod + def read_pr(creds: GithubCredentials, pr_url: str) -> tuple[str, str, str]: + try: + api_url = pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/issues/" + ) + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + title = data.get("title", "No title found") + body = data.get("body", "No body content found") + user = data.get("user", {}).get("login", "No user found") + + return title, body, user + except Exception as e: + return f"Failed to read pull request: {str(e)}", "", "" + + @staticmethod + def read_pr_changes(creds: GithubCredentials, pr_url: str) -> str: + try: + api_url = ( + pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/pulls/" + ) + + "/files" + ) + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + files = response.json() + changes = [] + for file in files: + filename = file.get("filename") + patch = file.get("patch") + if filename and patch: + changes.append(f"File: {filename}\n{patch}") + + return "\n\n".join(changes) + except Exception as e: + return f"Failed to read PR changes: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + title, body, user = self.read_pr( + input_data.creds, + input_data.pr_url, + ) + if "Failed" in title: + yield "error", title + else: + yield "title", title + yield "body", body + yield "user", user + + if input_data.include_pr_changes: + changes = self.read_pr_changes( + input_data.creds, + input_data.pr_url, + ) + if "Failed" in changes: + yield "error", changes + else: + yield "changes", changes + else: + yield "changes", "Changes not included" + + +class GithubListIssuesBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + issues: list[dict[str, str]] = SchemaField( + description="List of issues with their URLs" + ) + error: str = SchemaField(description="Error message if listing issues failed") + + def __init__(self): + super().__init__( + id="0006h3i4-5678-90ab-1234-567890abcdef", + description="This block lists all issues for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubListIssuesBlock.Input, + output_schema=GithubListIssuesBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[ + ( + "issues", + [ + { + "title": "Issue 1", + "url": "https://github.com/owner/repo/issues/1", + } + ], + ) + ], + test_mock={ + "list_issues": lambda *args, **kwargs: [ + { + "title": "Issue 1", + "url": "https://github.com/owner/repo/issues/1", + } + ] + }, + ) + + @staticmethod + def list_issues(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: + try: + api_url = repo_url.replace("github.com", "api.github.com/repos") + "/issues" + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + issues = [ + {"title": issue["title"], "url": issue["html_url"]} for issue in data + ] + + return issues + except Exception as e: + return [{"title": "Error", "url": f"Failed to list issues: {str(e)}"}] + + def run(self, input_data: Input) -> BlockOutput: + issues = self.list_issues( + input_data.creds, + input_data.repo_url, + ) + if any("Failed" in issue["url"] for issue in issues): + yield "error", issues[0]["url"] + else: + yield "issues", issues + + +class GithubReadTagsBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + tags: list[dict[str, str]] = SchemaField( + description="List of tags with their names and URLs" + ) + error: str = SchemaField(description="Error message if listing tags failed") + + def __init__(self): + super().__init__( + id="0007g3h4-5678-90ab-1234-567890abcdef", + description="This block lists all tags for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadTagsBlock.Input, + output_schema=GithubReadTagsBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[ + ( + "tags", + [ + { + "name": "v1.0.0", + "url": "https://github.com/owner/repo/tree/v1.0.0", + } + ], + ) + ], + test_mock={ + "list_tags": lambda *args, **kwargs: [ + { + "name": "v1.0.0", + "url": "https://github.com/owner/repo/tree/v1.0.0", + } + ] + }, + ) + + @staticmethod + def list_tags(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/tags" + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + tags = [ + { + "name": tag["name"], + "url": f"https://github.com/{repo_path}/tree/{tag['name']}", + } + for tag in data + ] + + return tags + except Exception as e: + return [{"name": "Error", "url": f"Failed to list tags: {str(e)}"}] + + def run(self, input_data: Input) -> BlockOutput: + tags = self.list_tags( + input_data.creds, + input_data.repo_url, + ) + if any("Failed" in tag["url"] for tag in tags): + yield "error", tags[0]["url"] + else: + yield "tags", tags + + +class GithubReadBranchesBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + branches: list[dict[str, str]] = SchemaField( + description="List of branches with their names and URLs" + ) + error: str = SchemaField(description="Error message if listing branches failed") + + def __init__(self): + super().__init__( + id="0008i3j4-5678-90ab-1234-567890abcdef", + description="This block lists all branches for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadBranchesBlock.Input, + output_schema=GithubReadBranchesBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[ + ( + "branches", + [ + { + "name": "main", + "url": "https://github.com/owner/repo/tree/main", + } + ], + ) + ], + test_mock={ + "list_branches": lambda *args, **kwargs: [ + { + "name": "main", + "url": "https://github.com/owner/repo/tree/main", + } + ] + }, + ) + + @staticmethod + def list_branches(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: + try: + api_url = ( + repo_url.replace("github.com", "api.github.com/repos") + "/branches" + ) + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + branches = [ + {"name": branch["name"], "url": branch["commit"]["url"]} + for branch in data + ] + + return branches + except Exception as e: + return [{"name": "Error", "url": f"Failed to list branches: {str(e)}"}] + + def run(self, input_data: Input) -> BlockOutput: + branches = self.list_branches( + input_data.creds, + input_data.repo_url, + ) + if any("Failed" in branch["url"] for branch in branches): + yield "error", branches[0]["url"] + else: + yield "branches", branches + + +class GithubReadDiscussionsBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + num_discussions: int = SchemaField( + description="Number of discussions to fetch", default=5 + ) + + class Output(BlockSchema): + discussions: list[dict[str, str]] = SchemaField( + description="List of discussions with their titles and URLs" + ) + error: str = SchemaField( + description="Error message if listing discussions failed" + ) + + def __init__(self): + super().__init__( + id="0009j3k4-5678-90ab-1234-567890abcdef", + description="This block lists recent discussions for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadDiscussionsBlock.Input, + output_schema=GithubReadDiscussionsBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + "num_discussions": 3, + }, + test_output=[ + ( + "discussions", + [ + { + "title": "Discussion 1", + "url": "https://github.com/owner/repo/discussions/1", + } + ], + ) + ], + test_mock={ + "list_discussions": lambda *args, **kwargs: [ + { + "title": "Discussion 1", + "url": "https://github.com/owner/repo/discussions/1", + } + ] + }, + ) + + @staticmethod + def list_discussions( + creds: GithubCredentials, repo_url: str, num_discussions: int + ) -> list[dict[str, str]]: + try: + repo_path = repo_url.replace("https://github.com/", "") + owner, repo = repo_path.split("/") + query = """ + query($owner: String!, $repo: String!, $num: Int!) { + repository(owner: $owner, name: $repo) { + discussions(first: $num) { + nodes { + title + url + } + } + } + } + """ + variables = {"owner": owner, "repo": repo, "num": num_discussions} + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.post( + "https://api.github.com/graphql", + json={"query": query, "variables": variables}, + headers=headers, + ) + response.raise_for_status() + + data = response.json() + discussions = [ + {"title": discussion["title"], "url": discussion["url"]} + for discussion in data["data"]["repository"]["discussions"]["nodes"] + ] + + return discussions + except Exception as e: + return [{"title": "Error", "url": f"Failed to list discussions: {str(e)}"}] + + def run(self, input_data: Input) -> BlockOutput: + discussions = self.list_discussions( + input_data.creds, input_data.repo_url, input_data.num_discussions + ) + if any("Failed" in discussion["url"] for discussion in discussions): + yield "error", discussions[0]["url"] + else: + yield "discussions", discussions + + +class GithubReadReleasesBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + releases: list[dict[str, str]] = SchemaField( + description="List of releases with their names and URLs" + ) + error: str = SchemaField(description="Error message if listing releases failed") + + def __init__(self): + super().__init__( + id="0010k3l4-5678-90ab-1234-567890abcdef", + description="This block lists all releases for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadReleasesBlock.Input, + output_schema=GithubReadReleasesBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[ + ( + "releases", + [ + { + "name": "v1.0.0", + "url": "https://github.com/owner/repo/releases/tag/v1.0.0", + } + ], + ) + ], + test_mock={ + "list_releases": lambda *args, **kwargs: [ + { + "name": "v1.0.0", + "url": "https://github.com/owner/repo/releases/tag/v1.0.0", + } + ] + }, + ) + + @staticmethod + def list_releases(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/releases" + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + releases = [ + {"name": release["name"], "url": release["html_url"]} + for release in data + ] + + return releases + except Exception as e: + return [{"name": "Error", "url": f"Failed to list releases: {str(e)}"}] + + def run(self, input_data: Input) -> BlockOutput: + releases = self.list_releases( + input_data.creds, + input_data.repo_url, + ) + if any("Failed" in release["url"] for release in releases): + yield "error", releases[0]["url"] + else: + yield "releases", releases + + +class GithubAddLabelBlock(Block): + class Input(BlockSchema): + issue_url: str = SchemaField( + description="URL of the GitHub issue or pull request", + placeholder="https://github.com/owner/repo/issues/1", + ) + label: str = SchemaField( + description="Label to add to the issue or pull request", + placeholder="Enter the label", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the label addition operation") + error: str = SchemaField( + description="Error message if the label addition failed" + ) + + def __init__(self): + super().__init__( + id="0011l3m4-5678-90ab-1234-567890abcdef", + description="This block adds a label to a specified GitHub issue or pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubAddLabelBlock.Input, + output_schema=GithubAddLabelBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "label": "bug", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Label added successfully")], + test_mock={"add_label": lambda *args, **kwargs: "Label added successfully"}, + ) + + @staticmethod + def add_label(creds: GithubCredentials, issue_url: str, label: str) -> str: + try: + # Convert the provided GitHub URL to the API URL + if "/pull/" in issue_url: + api_url = issue_url.replace("github.com", "api.github.com/repos").replace("/pull/", "/issues/") + "/labels" + else: + api_url = issue_url.replace("github.com", "api.github.com/repos") + "/labels" + + # Log the constructed API URL for debugging + print(f"Constructed API URL: {api_url}") + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + data = {"labels": [label]} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Label added successfully" + except requests.exceptions.HTTPError as http_err: + return f"HTTP error occurred: {http_err} - {response.text}" + except Exception as e: + return f"Failed to add label: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.add_label( + input_data.creds, + input_data.issue_url, + input_data.label, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubRemoveLabelBlock(Block): + class Input(BlockSchema): + issue_url: str = SchemaField( + description="URL of the GitHub issue or pull request", + placeholder="https://github.com/owner/repo/issues/1", + ) + label: str = SchemaField( + description="Label to remove from the issue or pull request", + placeholder="Enter the label", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the label removal operation") + error: str = SchemaField( + description="Error message if the label removal failed" + ) + + def __init__(self): + super().__init__( + id="0012m3n4-5678-90ab-1234-567890abcdef", + description="This block removes a label from a specified GitHub issue or pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubRemoveLabelBlock.Input, + output_schema=GithubRemoveLabelBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "label": "bug", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Label removed successfully")], + test_mock={ + "remove_label": lambda *args, **kwargs: "Label removed successfully" + }, + ) + + @staticmethod + def remove_label(creds: GithubCredentials, issue_url: str, label: str) -> str: + try: + # Convert the provided GitHub URL to the API URL + if "/pull/" in issue_url: + api_url = issue_url.replace("github.com", "api.github.com/repos").replace("/pull/", "/issues/") + f"/labels/{label}" + else: + api_url = issue_url.replace("github.com", "api.github.com/repos") + f"/labels/{label}" + + # Log the constructed API URL for debugging + print(f"Constructed API URL: {api_url}") + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.delete(api_url, headers=headers) + response.raise_for_status() + + return "Label removed successfully" + except requests.exceptions.HTTPError as http_err: + return f"HTTP error occurred: {http_err} - {response.text}" + except Exception as e: + return f"Failed to remove label: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.remove_label( + input_data.creds, + input_data.issue_url, + input_data.label, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubAssignReviewerBlock(Block): + class Input(BlockSchema): + pr_url: str = SchemaField( + description="URL of the GitHub pull request", + placeholder="https://github.com/owner/repo/pull/1", + ) + reviewer: str = SchemaField( + description="Username of the reviewer to assign", + placeholder="Enter the reviewer's username", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the reviewer assignment operation" + ) + error: str = SchemaField( + description="Error message if the reviewer assignment failed" + ) + + def __init__(self): + super().__init__( + id="0014o3p4-5678-90ab-1234-567890abcdef", + description="This block assigns a reviewer to a specified GitHub pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubAssignReviewerBlock.Input, + output_schema=GithubAssignReviewerBlock.Output, + test_input={ + "pr_url": "https://github.com/owner/repo/pull/1", + "reviewer": "reviewer_username", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Reviewer assigned successfully")], + test_mock={ + "assign_reviewer": lambda *args, **kwargs: "Reviewer assigned successfully" + }, + ) + + @staticmethod + def assign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> str: + try: + # Convert the PR URL to the appropriate API endpoint + api_url = ( + pr_url.replace("github.com", "api.github.com/repos") + .replace("/pull/", "/pulls/") + + "/requested_reviewers" + ) + + # Log the constructed API URL for debugging + print(f"Constructed API URL: {api_url}") + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + data = {"reviewers": [reviewer]} + + # Log the request data for debugging + print(f"Request data: {data}") + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Reviewer assigned successfully" + except requests.exceptions.HTTPError as http_err: + if response.status_code == 422: + return f"Failed to assign reviewer: The reviewer '{reviewer}' may not have permission or the pull request is not in a valid state. Detailed error: {response.text}" + else: + return f"HTTP error occurred: {http_err} - {response.text}" + except Exception as e: + return f"Failed to assign reviewer: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.assign_reviewer( + input_data.creds, + input_data.pr_url, + input_data.reviewer, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubUnassignReviewerBlock(Block): + class Input(BlockSchema): + pr_url: str = SchemaField( + description="URL of the GitHub pull request", + placeholder="https://github.com/owner/repo/pull/1", + ) + reviewer: str = SchemaField( + description="Username of the reviewer to unassign", + placeholder="Enter the reviewer's username", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the reviewer unassignment operation" + ) + error: str = SchemaField( + description="Error message if the reviewer unassignment failed" + ) + + def __init__(self): + super().__init__( + id="0015p3q4-5678-90ab-1234-567890abcdef", + description="This block unassigns a reviewer from a specified GitHub pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubUnassignReviewerBlock.Input, + output_schema=GithubUnassignReviewerBlock.Output, + test_input={ + "pr_url": "https://github.com/owner/repo/pull/1", + "reviewer": "reviewer_username", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Reviewer unassigned successfully")], + test_mock={ + "unassign_reviewer": lambda *args, **kwargs: "Reviewer unassigned successfully" + }, + ) + + @staticmethod + def unassign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> str: + try: + api_url = ( + pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/pulls/" + ) + + "/requested_reviewers" + ) + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + data = {"reviewers": [reviewer]} + + response = requests.delete(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Reviewer unassigned successfully" + except Exception as e: + return f"Failed to unassign reviewer: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.unassign_reviewer( + input_data.creds, + input_data.pr_url, + input_data.reviewer, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubListReviewersBlock(Block): + class Input(BlockSchema): + pr_url: str = SchemaField( + description="URL of the GitHub pull request", + placeholder="https://github.com/owner/repo/pull/1", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + reviewers: list[dict[str, str]] = SchemaField( + description="List of reviewers with their usernames and URLs" + ) + error: str = SchemaField( + description="Error message if listing reviewers failed" + ) + + def __init__(self): + super().__init__( + id="0016q3r4-5678-90ab-1234-567890abcdef", + description="This block lists all reviewers for a specified GitHub pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubListReviewersBlock.Input, + output_schema=GithubListReviewersBlock.Output, + test_input={ + "pr_url": "https://github.com/owner/repo/pull/1", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[ + ( + "reviewers", + [ + { + "username": "reviewer1", + "url": "https://github.com/reviewer1", + } + ], + ) + ], + test_mock={ + "list_reviewers": lambda *args, **kwargs: [ + { + "username": "reviewer1", + "url": "https://github.com/reviewer1", + } + ] + }, + ) + + @staticmethod + def list_reviewers(creds: GithubCredentials, pr_url: str) -> list[dict[str, str]]: + try: + api_url = ( + pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/pulls/" + ) + + "/requested_reviewers" + ) + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + reviewers = [ + {"username": reviewer["login"], "url": reviewer["html_url"]} + for reviewer in data.get("users", []) + ] + + return reviewers + except Exception as e: + return [{"username": "Error", "url": f"Failed to list reviewers: {str(e)}"}] + + def run(self, input_data: Input) -> BlockOutput: + reviewers = self.list_reviewers( + input_data.creds, + input_data.pr_url, + ) + if any("Failed" in reviewer["url"] for reviewer in reviewers): + yield "error", reviewers[0]["url"] + else: + yield "reviewers", reviewers + + +class GithubAssignIssueBlock(Block): + class Input(BlockSchema): + issue_url: str = SchemaField( + description="URL of the GitHub issue", + placeholder="https://github.com/owner/repo/issues/1", + ) + assignee: str = SchemaField( + description="Username to assign to the issue", + placeholder="Enter the username", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the issue assignment operation" + ) + error: str = SchemaField( + description="Error message if the issue assignment failed" + ) + + def __init__(self): + super().__init__( + id="0004r3s5-6789-01bc-2345-678901bcdefg", + description="This block assigns a user to a specified GitHub issue using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubAssignIssueBlock.Input, + output_schema=GithubAssignIssueBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "assignee": "username1", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Issue assigned successfully")], + test_mock={ + "assign_issue": lambda *args, **kwargs: "Issue assigned successfully" + }, + ) + + @staticmethod + def assign_issue( + creds: GithubCredentials, + issue_url: str, + assignee: str, + ) -> str: + try: + # Extracting repo path and issue number from the issue URL + repo_path, issue_number = issue_url.replace("https://github.com/", "").split("/issues/") + api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + data = {"assignees": [assignee]} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Issue assigned successfully" + except requests.exceptions.HTTPError as http_err: + return f"Failed to assign issue: {str(http_err)}" + except Exception as e: + return f"Failed to assign issue: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.assign_issue( + input_data.creds, + input_data.issue_url, + input_data.assignee, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubUnassignIssueBlock(Block): + class Input(BlockSchema): + issue_url: str = SchemaField( + description="URL of the GitHub issue", + placeholder="https://github.com/owner/repo/issues/1", + ) + assignee: str = SchemaField( + description="Username to unassign from the issue", + placeholder="Enter the username", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the issue unassignment operation" + ) + error: str = SchemaField( + description="Error message if the issue unassignment failed" + ) + + def __init__(self): + super().__init__( + id="0005r3s6-7890-12cd-3456-789012cdefgh", + description="This block unassigns a user from a specified GitHub issue using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubUnassignIssueBlock.Input, + output_schema=GithubUnassignIssueBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "assignee": "username1", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Issue unassigned successfully")], + test_mock={ + "unassign_issue": lambda *args, **kwargs: "Issue unassigned successfully" + }, + ) + + @staticmethod + def unassign_issue( + creds: GithubCredentials, + issue_url: str, + assignee: str, + ) -> str: + try: + # Extracting repo path and issue number from the issue URL + repo_path, issue_number = issue_url.replace("https://github.com/", "").split("/issues/") + api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" + + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + data = {"assignees": [assignee]} + + response = requests.delete(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Issue unassigned successfully" + except requests.exceptions.HTTPError as http_err: + return f"Failed to unassign issue: {str(http_err)}" + except Exception as e: + return f"Failed to unassign issue: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.unassign_issue( + input_data.creds, + input_data.issue_url, + input_data.assignee, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubReadCodeownersFileBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + codeowners_content: str = SchemaField( + description="Content of the CODEOWNERS file" + ) + error: str = SchemaField(description="Error message if the file reading failed") + + def __init__(self): + super().__init__( + id="0006r3s7-8901-23de-4567-890123defghi", + description="This block reads the CODEOWNERS file from the master branch of a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadCodeownersFileBlock.Input, + output_schema=GithubReadCodeownersFileBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("codeowners_content", "# CODEOWNERS content")], + test_mock={ + "read_codeowners": lambda *args, **kwargs: "# CODEOWNERS content" + }, + ) + + @staticmethod + def read_codeowners(creds: GithubCredentials, repo_url: str) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/contents/.github/CODEOWNERS?ref=master" + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + content = response.json() + return base64.b64decode(content["content"]).decode("utf-8") + except requests.exceptions.HTTPError as http_err: + return f"Failed to read CODEOWNERS file: {str(http_err)}" + except Exception as e: + return f"Failed to read CODEOWNERS file: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + content = self.read_codeowners( + input_data.creds, + input_data.repo_url, + ) + if "Failed" not in content: + yield "codeowners_content", content + else: + yield "error", content + + +class GithubReadFileFromMasterBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + file_path: str = SchemaField( + description="Path to the file in the repository", + placeholder="path/to/file", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + file_content: str = SchemaField( + description="Content of the file from the master branch" + ) + error: str = SchemaField(description="Error message if the file reading failed") + + def __init__(self): + super().__init__( + id="0007r3s8-9012-34ef-5678-901234efghij", + description="This block reads the content of a specified file from the master branch of a GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadFileFromMasterBlock.Input, + output_schema=GithubReadFileFromMasterBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "file_path": "path/to/file", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("file_content", "File content")], + test_mock={"read_file": lambda *args, **kwargs: "File content"}, + ) + + @staticmethod + def read_file(creds: GithubCredentials, repo_url: str, file_path: str) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/contents/{file_path}?ref=master" + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + content = response.json() + return base64.b64decode(content["content"]).decode("utf-8") + except requests.exceptions.HTTPError as http_err: + return f"Failed to read file: {str(http_err)}" + except Exception as e: + return f"Failed to read file: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + content = self.read_file( + input_data.creds, + input_data.repo_url, + input_data.file_path, + ) + if "Failed" not in content: + yield "file_content", content + else: + yield "error", content + + +class GithubReadFileFolderRepoBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + path: str = SchemaField( + description="Path to the file/folder in the repository", + placeholder="path/to/file_or_folder", + ) + branch: str = SchemaField( + description="Branch name to read from", + placeholder="branch_name", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + content: str = SchemaField( + description="Content of the file/folder/repo from the specified branch" + ) + error: str = SchemaField(description="Error message if the reading failed") + + def __init__(self): + super().__init__( + id="0008r3s9-0123-45fg-6789-012345fghijk", + description="This block reads the content of a specified file, folder, or repository from a specified branch using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadFileFolderRepoBlock.Input, + output_schema=GithubReadFileFolderRepoBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "path": "path/to/file_or_folder", + "branch": "branch_name", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("content", "File or folder content")], + test_mock={ + "read_content": lambda *args, **kwargs: "File or folder content" + }, + ) + + @staticmethod + def read_content( + creds: GithubCredentials, repo_url: str, path: str, branch: str + ) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = ( + f"https://api.github.com/repos/{repo_path}/contents/{path}?ref={branch}" + ) + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + content = response.json() + if "content" in content: + return base64.b64decode(content["content"]).decode("utf-8") + else: + return content # Return the directory content as JSON + + except requests.exceptions.HTTPError as http_err: + return f"Failed to read content: {str(http_err)}" + except Exception as e: + return f"Failed to read content: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + content = self.read_content( + input_data.creds, + input_data.repo_url, + input_data.path, + input_data.branch, + ) + if "Failed" not in content: + yield "content", content + else: + yield "error", content + + +class GithubMakeBranchBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + new_branch: str = SchemaField( + description="Name of the new branch", + placeholder="new_branch_name", + ) + source_branch: str = SchemaField( + description="Name of the source branch", + placeholder="source_branch_name", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the branch creation operation") + error: str = SchemaField( + description="Error message if the branch creation failed" + ) + + def __init__(self): + super().__init__( + id="0008r3s9-0123-45fg-6789-012345fghijp", + description="This block creates a new branch from a specified source branch using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubMakeBranchBlock.Input, + output_schema=GithubMakeBranchBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "new_branch": "new_branch_name", + "source_branch": "source_branch_name", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Branch created successfully")], + test_mock={ + "create_branch": lambda *args, **kwargs: "Branch created successfully" + }, + ) + + @staticmethod + def create_branch( + creds: GithubCredentials, repo_url: str, new_branch: str, source_branch: str + ) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + ref_api_url = f"https://api.github.com/repos/{repo_path}/git/refs/heads/{source_branch}" + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(ref_api_url, headers=headers) + response.raise_for_status() + + sha = response.json()["object"]["sha"] + + create_branch_api_url = f"https://api.github.com/repos/{repo_path}/git/refs" + data = {"ref": f"refs/heads/{new_branch}", "sha": sha} + + response = requests.post(create_branch_api_url, headers=headers, json=data) + response.raise_for_status() + + return "Branch created successfully" + except requests.exceptions.HTTPError as http_err: + return f"Failed to create branch: {str(http_err)}" + except Exception as e: + return f"Failed to create branch: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.create_branch( + input_data.creds, + input_data.repo_url, + input_data.new_branch, + input_data.source_branch, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubDeleteBranchBlock(Block): + class Input(BlockSchema): + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + branch: str = SchemaField( + description="Name of the branch to delete", + placeholder="branch_name", + ) + creds: GithubCredentials = Field( + description="GitHub OAuth credentials", + default=GithubCredentials(), + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the branch deletion operation") + error: str = SchemaField( + description="Error message if the branch deletion failed" + ) + + def __init__(self): + super().__init__( + id="0008r3s9-0123-45fg-6789-012345fghijq", + description="This block deletes a specified branch using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubDeleteBranchBlock.Input, + output_schema=GithubDeleteBranchBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "branch": "branch_name", + "creds": { + "github_oauth_token": "your-github-oauth-token", + }, + }, + test_output=[("status", "Branch deleted successfully")], + test_mock={ + "delete_branch": lambda *args, **kwargs: "Branch deleted successfully" + }, + ) + + @staticmethod + def delete_branch(creds: GithubCredentials, repo_url: str, branch: str) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = ( + f"https://api.github.com/repos/{repo_path}/git/refs/heads/{branch}" + ) + headers = { + "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Accept": "application/vnd.github.v3+json", + } + + response = requests.delete(api_url, headers=headers) + response.raise_for_status() + + return "Branch deleted successfully" + except requests.exceptions.HTTPError as http_err: + return f"Failed to delete branch: {str(http_err)}" + except Exception as e: + return f"Failed to delete branch: {str(e)}" + + def run(self, input_data: Input) -> BlockOutput: + status = self.delete_branch( + input_data.creds, + input_data.repo_url, + input_data.branch, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status diff --git a/rnd/autogpt_server/autogpt_server/data/block.py b/rnd/autogpt_server/autogpt_server/data/block.py index 0ecf56e497a5..19dcdcd24122 100644 --- a/rnd/autogpt_server/autogpt_server/data/block.py +++ b/rnd/autogpt_server/autogpt_server/data/block.py @@ -36,6 +36,7 @@ class BlockCategory(Enum): INPUT = "Block that interacts with input of the graph." OUTPUT = "Block that interacts with output of the graph." LOGIC = "Programming logic to control the flow of your agent" + DEVELOPER_TOOLS = "Developer tools like Github." # added this so all the github blocks are in the same category for now, this is to be changed def dict(self) -> dict[str, str]: return {"category": self.name, "description": self.value} From 4cec829b356f8d7861d9603e9e1687375c04080a Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Wed, 11 Sep 2024 17:40:20 +0200 Subject: [PATCH 02/43] move github_blocks.py -> github.py --- .../blocks/{github_blocks.py => github.py} | 50 +++++++++++++------ 1 file changed, 35 insertions(+), 15 deletions(-) rename rnd/autogpt_server/autogpt_server/blocks/{github_blocks.py => github.py} (98%) diff --git a/rnd/autogpt_server/autogpt_server/blocks/github_blocks.py b/rnd/autogpt_server/autogpt_server/blocks/github.py similarity index 98% rename from rnd/autogpt_server/autogpt_server/blocks/github_blocks.py rename to rnd/autogpt_server/autogpt_server/blocks/github.py index 4967dd65498d..b46f452c2aa6 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/github_blocks.py +++ b/rnd/autogpt_server/autogpt_server/blocks/github.py @@ -310,13 +310,13 @@ def __init__(self): test_output=[ ("title", "Title of the issue"), ("body", "This is the body of the issue."), - ("user", "username") + ("user", "username"), ], test_mock={ "read_issue": lambda *args, **kwargs: ( "Title of the issue", "This is the body of the issue.", - "username" + "username", ) }, ) @@ -404,7 +404,7 @@ def __init__(self): "read_pr": lambda *args, **kwargs: ( "Title of the pull request", "This is the body of the pull request.", - "username" + "username", ), "read_pr_changes": lambda *args, **kwargs: "List of changes made in the pull request.", }, @@ -984,10 +984,17 @@ def add_label(creds: GithubCredentials, issue_url: str, label: str) -> str: try: # Convert the provided GitHub URL to the API URL if "/pull/" in issue_url: - api_url = issue_url.replace("github.com", "api.github.com/repos").replace("/pull/", "/issues/") + "/labels" + api_url = ( + issue_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/issues/" + ) + + "/labels" + ) else: - api_url = issue_url.replace("github.com", "api.github.com/repos") + "/labels" - + api_url = ( + issue_url.replace("github.com", "api.github.com/repos") + "/labels" + ) + # Log the constructed API URL for debugging print(f"Constructed API URL: {api_url}") @@ -1064,10 +1071,18 @@ def remove_label(creds: GithubCredentials, issue_url: str, label: str) -> str: try: # Convert the provided GitHub URL to the API URL if "/pull/" in issue_url: - api_url = issue_url.replace("github.com", "api.github.com/repos").replace("/pull/", "/issues/") + f"/labels/{label}" + api_url = ( + issue_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/issues/" + ) + + f"/labels/{label}" + ) else: - api_url = issue_url.replace("github.com", "api.github.com/repos") + f"/labels/{label}" - + api_url = ( + issue_url.replace("github.com", "api.github.com/repos") + + f"/labels/{label}" + ) + # Log the constructed API URL for debugging print(f"Constructed API URL: {api_url}") @@ -1145,8 +1160,9 @@ def assign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> str try: # Convert the PR URL to the appropriate API endpoint api_url = ( - pr_url.replace("github.com", "api.github.com/repos") - .replace("/pull/", "/pulls/") + pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/pulls/" + ) + "/requested_reviewers" ) @@ -1405,9 +1421,11 @@ def assign_issue( ) -> str: try: # Extracting repo path and issue number from the issue URL - repo_path, issue_number = issue_url.replace("https://github.com/", "").split("/issues/") + repo_path, issue_number = issue_url.replace( + "https://github.com/", "" + ).split("/issues/") api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" - + headers = { "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", "Accept": "application/vnd.github.v3+json", @@ -1486,9 +1504,11 @@ def unassign_issue( ) -> str: try: # Extracting repo path and issue number from the issue URL - repo_path, issue_number = issue_url.replace("https://github.com/", "").split("/issues/") + repo_path, issue_number = issue_url.replace( + "https://github.com/", "" + ).split("/issues/") api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" - + headers = { "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", "Accept": "application/vnd.github.v3+json", From e7fa68222710e695fcc8c6554a179fe8cc959355 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Wed, 11 Sep 2024 17:47:09 +0200 Subject: [PATCH 03/43] remove duplicate OAuth handler files --- .../autogpt_server/integrations/github.py | 99 ------------------- .../autogpt_server/integrations/google.py | 96 ------------------ .../autogpt_server/integrations/notion.py | 76 -------------- .../autogpt_server/integrations/oauth.py | 48 --------- 4 files changed, 319 deletions(-) delete mode 100644 rnd/autogpt_server/autogpt_server/integrations/github.py delete mode 100644 rnd/autogpt_server/autogpt_server/integrations/google.py delete mode 100644 rnd/autogpt_server/autogpt_server/integrations/notion.py delete mode 100644 rnd/autogpt_server/autogpt_server/integrations/oauth.py diff --git a/rnd/autogpt_server/autogpt_server/integrations/github.py b/rnd/autogpt_server/autogpt_server/integrations/github.py deleted file mode 100644 index af2483046152..000000000000 --- a/rnd/autogpt_server/autogpt_server/integrations/github.py +++ /dev/null @@ -1,99 +0,0 @@ -import time -from typing import Optional -from urllib.parse import urlencode - -import requests -from autogpt_libs.supabase_integration_credentials_store import OAuth2Credentials - -from autogpt_server.integrations.oauth import BaseOAuthHandler - - -class GitHubOAuthHandler(BaseOAuthHandler): - """ - Based on the documentation at: - - [Authorizing OAuth apps - GitHub Docs](https://docs.github.com/en/apps/oauth-apps/building-oauth-apps/authorizing-oauth-apps) - - [Refreshing user access tokens - GitHub Docs](https://docs.github.com/en/apps/creating-github-apps/authenticating-with-a-github-app/refreshing-user-access-tokens) - - Notes: - - By default, token expiration is disabled on GitHub Apps. This means the access - token doesn't expire and no refresh token is returned by the authorization flow. - - When token expiration gets enabled, any existing tokens will remain non-expiring. - - When token expiration gets disabled, token refreshes will return a non-expiring - access token *with no refresh token*. - """ # noqa - - PROVIDER_NAME = "github" - - def __init__(self, client_id: str, client_secret: str, redirect_uri: str): - self.client_id = client_id - self.client_secret = client_secret - self.redirect_uri = redirect_uri - self.auth_base_url = "https://github.com/login/oauth/authorize" - self.token_url = "https://github.com/login/oauth/access_token" - - def get_login_url(self, scopes: list[str], state: str) -> str: - params = { - "client_id": self.client_id, - "redirect_uri": self.redirect_uri, - "scope": " ".join(scopes), - "state": state, - } - return f"{self.auth_base_url}?{urlencode(params)}" - - def exchange_code_for_tokens(self, code: str) -> OAuth2Credentials: - return self._request_tokens({"code": code, "redirect_uri": self.redirect_uri}) - - def _refresh_tokens(self, credentials: OAuth2Credentials) -> OAuth2Credentials: - if not credentials.refresh_token: - return credentials - - return self._request_tokens( - { - "refresh_token": credentials.refresh_token.get_secret_value(), - "grant_type": "refresh_token", - } - ) - - def _request_tokens( - self, - params: dict[str, str], - current_credentials: Optional[OAuth2Credentials] = None, - ) -> OAuth2Credentials: - request_body = { - "client_id": self.client_id, - "client_secret": self.client_secret, - **params, - } - headers = {"Accept": "application/json"} - response = requests.post(self.token_url, data=request_body, headers=headers) - response.raise_for_status() - token_data: dict = response.json() - - now = int(time.time()) - new_credentials = OAuth2Credentials( - provider=self.PROVIDER_NAME, - title=current_credentials.title if current_credentials else "GitHub", - access_token=token_data["access_token"], - # Token refresh responses have an empty `scope` property (see docs), - # so we have to get the scope from the existing credentials object. - scopes=( - token_data.get("scope", "").split(",") - or (current_credentials.scopes if current_credentials else []) - ), - # Refresh token and expiration intervals are only given if token expiration - # is enabled in the GitHub App's settings. - refresh_token=token_data.get("refresh_token"), - access_token_expires_at=( - now + expires_in - if (expires_in := token_data.get("expires_in", None)) - else None - ), - refresh_token_expires_at=( - now + expires_in - if (expires_in := token_data.get("refresh_token_expires_in", None)) - else None - ), - ) - if current_credentials: - new_credentials.id = current_credentials.id - return new_credentials diff --git a/rnd/autogpt_server/autogpt_server/integrations/google.py b/rnd/autogpt_server/autogpt_server/integrations/google.py deleted file mode 100644 index edcba32df8c6..000000000000 --- a/rnd/autogpt_server/autogpt_server/integrations/google.py +++ /dev/null @@ -1,96 +0,0 @@ -from autogpt_libs.supabase_integration_credentials_store import OAuth2Credentials -from google.auth.transport.requests import Request -from google.oauth2.credentials import Credentials -from google_auth_oauthlib.flow import Flow -from pydantic import SecretStr - -from .oauth import BaseOAuthHandler - - -class GoogleOAuthHandler(BaseOAuthHandler): - """ - Based on the documentation at https://developers.google.com/identity/protocols/oauth2/web-server - """ # noqa - - PROVIDER_NAME = "google" - - def __init__(self, client_id: str, client_secret: str, redirect_uri: str): - self.client_id = client_id - self.client_secret = client_secret - self.redirect_uri = redirect_uri - self.token_uri = "https://oauth2.googleapis.com/token" - - def get_login_url(self, scopes: list[str], state: str) -> str: - flow = self._setup_oauth_flow(scopes) - flow.redirect_uri = self.redirect_uri - authorization_url, _ = flow.authorization_url( - access_type="offline", - include_granted_scopes="true", - state=state, - prompt="consent", - ) - return authorization_url - - def exchange_code_for_tokens(self, code: str) -> OAuth2Credentials: - flow = self._setup_oauth_flow(None) - flow.redirect_uri = self.redirect_uri - flow.fetch_token(code=code) - - google_creds = flow.credentials - # Google's OAuth library is poorly typed so we need some of these: - assert google_creds.token - assert google_creds.refresh_token - assert google_creds.expiry - assert google_creds.scopes - return OAuth2Credentials( - provider=self.PROVIDER_NAME, - title="Google", - access_token=SecretStr(google_creds.token), - refresh_token=SecretStr(google_creds.refresh_token), - access_token_expires_at=int(google_creds.expiry.timestamp()), - refresh_token_expires_at=None, - scopes=google_creds.scopes, - ) - - def _refresh_tokens(self, credentials: OAuth2Credentials) -> OAuth2Credentials: - # Google credentials should ALWAYS have a refresh token - assert credentials.refresh_token - - google_creds = Credentials( - token=credentials.access_token.get_secret_value(), - refresh_token=credentials.refresh_token.get_secret_value(), - token_uri=self.token_uri, - client_id=self.client_id, - client_secret=self.client_secret, - scopes=credentials.scopes, - ) - # Google's OAuth library is poorly typed so we need some of these: - assert google_creds.refresh_token - assert google_creds.scopes - - google_creds.refresh(Request()) - assert google_creds.expiry - - return OAuth2Credentials( - id=credentials.id, - provider=self.PROVIDER_NAME, - title=credentials.title, - access_token=SecretStr(google_creds.token), - refresh_token=SecretStr(google_creds.refresh_token), - access_token_expires_at=int(google_creds.expiry.timestamp()), - refresh_token_expires_at=None, - scopes=google_creds.scopes, - ) - - def _setup_oauth_flow(self, scopes: list[str] | None) -> Flow: - return Flow.from_client_config( - { - "web": { - "client_id": self.client_id, - "client_secret": self.client_secret, - "auth_uri": "https://accounts.google.com/o/oauth2/auth", - "token_uri": self.token_uri, - } - }, - scopes=scopes, - ) diff --git a/rnd/autogpt_server/autogpt_server/integrations/notion.py b/rnd/autogpt_server/autogpt_server/integrations/notion.py deleted file mode 100644 index aed2ccba4376..000000000000 --- a/rnd/autogpt_server/autogpt_server/integrations/notion.py +++ /dev/null @@ -1,76 +0,0 @@ -from base64 import b64encode -from urllib.parse import urlencode - -import requests -from autogpt_libs.supabase_integration_credentials_store import OAuth2Credentials - -from autogpt_server.integrations.oauth import BaseOAuthHandler - - -class NotionOAuthHandler(BaseOAuthHandler): - """ - Based on the documentation at https://developers.notion.com/docs/authorization - - Notes: - - Notion uses non-expiring access tokens and therefore doesn't have a refresh flow - - Notion doesn't use scopes - """ - - PROVIDER_NAME = "notion" - - def __init__(self, client_id: str, client_secret: str, redirect_uri: str): - self.client_id = client_id - self.client_secret = client_secret - self.redirect_uri = redirect_uri - self.auth_base_url = "https://api.notion.com/v1/oauth/authorize" - self.token_url = "https://api.notion.com/v1/oauth/token" - - def get_login_url(self, scopes: list[str], state: str) -> str: - params = { - "client_id": self.client_id, - "redirect_uri": self.redirect_uri, - "response_type": "code", - "owner": "user", - "state": state, - } - return f"{self.auth_base_url}?{urlencode(params)}" - - def exchange_code_for_tokens(self, code: str) -> OAuth2Credentials: - request_body = { - "grant_type": "authorization_code", - "code": code, - "redirect_uri": self.redirect_uri, - } - auth_str = b64encode(f"{self.client_id}:{self.client_secret}".encode()).decode() - headers = { - "Authorization": f"Basic {auth_str}", - "Accept": "application/json", - } - response = requests.post(self.token_url, json=request_body, headers=headers) - response.raise_for_status() - token_data = response.json() - - return OAuth2Credentials( - provider=self.PROVIDER_NAME, - title=token_data.get("workspace_name", "Notion"), - access_token=token_data["access_token"], - refresh_token=None, - access_token_expires_at=None, # Notion tokens don't expire - refresh_token_expires_at=None, - scopes=[], - metadata={ - "owner": token_data["owner"], - "bot_id": token_data["bot_id"], - "workspace_id": token_data["workspace_id"], - "workspace_name": token_data.get("workspace_name"), - "workspace_icon": token_data.get("workspace_icon"), - }, - ) - - def _refresh_tokens(self, credentials: OAuth2Credentials) -> OAuth2Credentials: - # Notion doesn't support token refresh - return credentials - - def needs_refresh(self, credentials: OAuth2Credentials) -> bool: - # Notion access tokens don't expire - return False diff --git a/rnd/autogpt_server/autogpt_server/integrations/oauth.py b/rnd/autogpt_server/autogpt_server/integrations/oauth.py deleted file mode 100644 index 5fefe5b54dbd..000000000000 --- a/rnd/autogpt_server/autogpt_server/integrations/oauth.py +++ /dev/null @@ -1,48 +0,0 @@ -import time -from abc import ABC, abstractmethod -from typing import ClassVar - -from autogpt_libs.supabase_integration_credentials_store import OAuth2Credentials - - -class BaseOAuthHandler(ABC): - PROVIDER_NAME: ClassVar[str] - - @abstractmethod - def __init__(self, client_id: str, client_secret: str, redirect_uri: str): ... - - @abstractmethod - def get_login_url(self, scopes: list[str], state: str) -> str: - """Constructs a login URL that the user can be redirected to""" - ... - - @abstractmethod - def exchange_code_for_tokens(self, code: str) -> OAuth2Credentials: - """Exchanges the acquired authorization code from login for a set of tokens""" - ... - - @abstractmethod - def _refresh_tokens(self, credentials: OAuth2Credentials) -> OAuth2Credentials: - """Implements the token refresh mechanism""" - ... - - def refresh_tokens(self, credentials: OAuth2Credentials) -> OAuth2Credentials: - if credentials.provider != self.PROVIDER_NAME: - raise ValueError( - f"{self.__class__.__name__} can not refresh tokens " - f"for other provider '{credentials.provider}'" - ) - return self._refresh_tokens(credentials) - - def get_access_token(self, credentials: OAuth2Credentials) -> str: - """Returns a valid access token, refreshing it first if needed""" - if self.needs_refresh(credentials): - credentials = self.refresh_tokens(credentials) - return credentials.access_token.get_secret_value() - - def needs_refresh(self, credentials: OAuth2Credentials) -> bool: - """Indicates whether the given tokens need to be refreshed""" - return ( - credentials.access_token_expires_at is not None - and credentials.access_token_expires_at < int(time.time()) + 300 - ) From 8ae5378d0b0b53c24cac1967c49953e658fca431 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 12 Sep 2024 01:40:41 +0200 Subject: [PATCH 04/43] feat(server): Add Supabase support to `AppService` --- rnd/autogpt_server/autogpt_server/util/service.py | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/rnd/autogpt_server/autogpt_server/util/service.py b/rnd/autogpt_server/autogpt_server/util/service.py index 6e707b904d97..db63d14c02d2 100644 --- a/rnd/autogpt_server/autogpt_server/util/service.py +++ b/rnd/autogpt_server/autogpt_server/util/service.py @@ -13,7 +13,7 @@ from autogpt_server.data.queue import AsyncEventQueue, AsyncRedisEventQueue from autogpt_server.util.process import AppProcess from autogpt_server.util.retry import conn_retry -from autogpt_server.util.settings import Config +from autogpt_server.util.settings import Config, Secrets logger = logging.getLogger(__name__) T = TypeVar("T") @@ -48,6 +48,7 @@ class AppService(AppProcess): event_queue: AsyncEventQueue = AsyncRedisEventQueue() use_db: bool = False use_redis: bool = False + use_supabase: bool = False def __init__(self, port): self.port = port @@ -76,6 +77,11 @@ def run(self): self.shared_event_loop.run_until_complete(db.connect()) if self.use_redis: self.shared_event_loop.run_until_complete(self.event_queue.connect()) + if self.use_supabase: + from supabase import create_client + + secrets = Secrets() + self.supabase = create_client(secrets.supabase_url, secrets.supabase_key) # Initialize the async loop. async_thread = threading.Thread(target=self.__start_async_loop) @@ -97,6 +103,9 @@ def cleanup(self): if self.use_redis: logger.info(f"[{self.__class__.__name__}] ⏳ Disconnecting Redis...") self.run_and_wait(self.event_queue.close()) + if self.use_supabase: + logger.info(f"[{self.__class__.__name__}] ⏳ Disconnecting Supabase...") + self.supabase.realtime.remove_all_channels() @conn_retry def __start_pyro(self): From 20c51373d3cb03ffae4086708f3b32e3968ed062 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 12 Sep 2024 01:44:21 +0200 Subject: [PATCH 05/43] Add `**kwargs` to `Block.run(..)` signature to support additional kwargs --- docs/content/server/new_blocks.md | 2 +- .../autogpt_server/blocks/agent.py | 2 +- .../autogpt_server/blocks/basic.py | 21 ++++++++++--------- .../autogpt_server/blocks/block.py | 2 +- .../autogpt_server/blocks/branching.py | 2 +- .../autogpt_server/blocks/csv.py | 2 +- .../autogpt_server/blocks/discord.py | 6 +++--- .../autogpt_server/blocks/email_block.py | 2 +- .../autogpt_server/blocks/http.py | 2 +- .../autogpt_server/blocks/iteration.py | 2 +- .../autogpt_server/blocks/llm.py | 8 +++---- .../autogpt_server/blocks/maths.py | 4 ++-- .../autogpt_server/blocks/medium.py | 2 +- .../autogpt_server/blocks/reddit.py | 4 ++-- .../autogpt_server/blocks/rss.py | 2 +- .../autogpt_server/blocks/search.py | 8 +++---- .../autogpt_server/blocks/talking_head.py | 2 +- .../autogpt_server/blocks/text.py | 8 +++---- .../autogpt_server/blocks/time_blocks.py | 9 ++++---- .../autogpt_server/blocks/youtube.py | 2 +- .../autogpt_server/data/block.py | 10 +++++---- 21 files changed, 52 insertions(+), 50 deletions(-) diff --git a/docs/content/server/new_blocks.md b/docs/content/server/new_blocks.md index 40c02f028e6c..41341d39635f 100644 --- a/docs/content/server/new_blocks.md +++ b/docs/content/server/new_blocks.md @@ -84,7 +84,7 @@ Follow these steps to create and test a new block: 5. **Implement the `run` method with error handling:**, this should contain the main logic of the block: ```python - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: topic = input_data.topic url = f"https://en.wikipedia.org/api/rest_v1/page/summary/{topic}" diff --git a/rnd/autogpt_server/autogpt_server/blocks/agent.py b/rnd/autogpt_server/autogpt_server/blocks/agent.py index 899c561dc1bc..628c492c71e2 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/agent.py +++ b/rnd/autogpt_server/autogpt_server/blocks/agent.py @@ -155,7 +155,7 @@ def get_result(agent: BlockAgent) -> str: raise error or Exception("Failed to get result") - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: # Set up configuration config = ConfigBuilder.build_config_from_env() # Disable commands diff --git a/rnd/autogpt_server/autogpt_server/blocks/basic.py b/rnd/autogpt_server/autogpt_server/blocks/basic.py index 0fc960be9694..887f029aff7c 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/basic.py +++ b/rnd/autogpt_server/autogpt_server/blocks/basic.py @@ -53,7 +53,7 @@ def __init__(self): static_output=True, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: yield "output", input_data.data or input_data.input @@ -75,13 +75,12 @@ def __init__(self): test_output=("status", "printed"), ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: print(">>>>> Print: ", input_data.text) yield "status", "printed" class FindInDictionaryBlock(Block): - class Input(BlockSchema): input: Any = Field(description="Dictionary to lookup from") key: str | int = Field(description="Key to lookup in the dictionary") @@ -115,7 +114,7 @@ def __init__(self): categories={BlockCategory.BASIC}, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: obj = input_data.input key = input_data.key @@ -190,7 +189,7 @@ def __init__(self): ui_type=BlockUIType.INPUT, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: yield "result", input_data.value @@ -269,7 +268,7 @@ def __init__(self): ui_type=BlockUIType.OUTPUT, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: """ Attempts to format the recorded_value using the fmt_string if provided. If formatting fails or no fmt_string is given, returns the original recorded_value. @@ -327,7 +326,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: # If no dictionary is provided, create a new one if input_data.dictionary is None: @@ -398,7 +397,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: # If no list is provided, create a new one if input_data.list is None: @@ -422,7 +421,8 @@ class NoteBlock(Block): class Input(BlockSchema): text: str = SchemaField(description="The text to display in the sticky note.") - class Output(BlockSchema): ... + class Output(BlockSchema): + ... def __init__(self): super().__init__( @@ -436,4 +436,5 @@ def __init__(self): ui_type=BlockUIType.NOTE, ) - def run(self, input_data: Input) -> BlockOutput: ... + def run(self, input_data: Input, **kwargs) -> BlockOutput: + ... diff --git a/rnd/autogpt_server/autogpt_server/blocks/block.py b/rnd/autogpt_server/autogpt_server/blocks/block.py index 99b60039bb2a..16f052410346 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/block.py +++ b/rnd/autogpt_server/autogpt_server/blocks/block.py @@ -31,7 +31,7 @@ def __init__(self): disabled=True, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: code = input_data.code if search := re.search(r"class (\w+)\(Block\):", code): diff --git a/rnd/autogpt_server/autogpt_server/blocks/branching.py b/rnd/autogpt_server/autogpt_server/blocks/branching.py index 4081e6ef456d..b7925f897ae2 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/branching.py +++ b/rnd/autogpt_server/autogpt_server/blocks/branching.py @@ -70,7 +70,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: value1 = input_data.value1 operator = input_data.operator value2 = input_data.value2 diff --git a/rnd/autogpt_server/autogpt_server/blocks/csv.py b/rnd/autogpt_server/autogpt_server/blocks/csv.py index 7044c515b799..64c2b674ba96 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/csv.py +++ b/rnd/autogpt_server/autogpt_server/blocks/csv.py @@ -32,7 +32,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: import csv from io import StringIO diff --git a/rnd/autogpt_server/autogpt_server/blocks/discord.py b/rnd/autogpt_server/autogpt_server/blocks/discord.py index d6bff3d40f10..c1459c823919 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/discord.py +++ b/rnd/autogpt_server/autogpt_server/blocks/discord.py @@ -81,14 +81,14 @@ async def on_message(message): await client.start(token) - def run(self, input_data: "ReadDiscordMessagesBlock.Input") -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: while True: for output_name, output_value in self.__run(input_data): yield output_name, output_value if not input_data.continuous_read: break - def __run(self, input_data: "ReadDiscordMessagesBlock.Input") -> BlockOutput: + def __run(self, input_data: Input) -> BlockOutput: try: loop = asyncio.get_event_loop() future = self.run_bot(input_data.discord_bot_token.get_secret_value()) @@ -187,7 +187,7 @@ def chunk_message(self, message: str, limit: int = 2000) -> list: """Splits a message into chunks not exceeding the Discord limit.""" return [message[i : i + limit] for i in range(0, len(message), limit)] - def run(self, input_data: "SendDiscordMessageBlock.Input") -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: loop = asyncio.get_event_loop() future = self.send_message( diff --git a/rnd/autogpt_server/autogpt_server/blocks/email_block.py b/rnd/autogpt_server/autogpt_server/blocks/email_block.py index 307c07c46c82..2a9f2a698979 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/email_block.py +++ b/rnd/autogpt_server/autogpt_server/blocks/email_block.py @@ -88,7 +88,7 @@ def send_email( except Exception as e: return f"Failed to send email: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: status = self.send_email( input_data.creds, input_data.to_email, diff --git a/rnd/autogpt_server/autogpt_server/blocks/http.py b/rnd/autogpt_server/autogpt_server/blocks/http.py index d12b2fc2756b..e8f13af1e05f 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/http.py +++ b/rnd/autogpt_server/autogpt_server/blocks/http.py @@ -37,7 +37,7 @@ def __init__(self): output_schema=SendWebRequestBlock.Output, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: if isinstance(input_data.body, str): input_data.body = json.loads(input_data.body) diff --git a/rnd/autogpt_server/autogpt_server/blocks/iteration.py b/rnd/autogpt_server/autogpt_server/blocks/iteration.py index 9858541075d4..de6287b2598e 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/iteration.py +++ b/rnd/autogpt_server/autogpt_server/blocks/iteration.py @@ -31,6 +31,6 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: for index, item in enumerate(input_data.items): yield "item", (index, item) diff --git a/rnd/autogpt_server/autogpt_server/blocks/llm.py b/rnd/autogpt_server/autogpt_server/blocks/llm.py index edfc8f706481..5c69311afb7d 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/llm.py +++ b/rnd/autogpt_server/autogpt_server/blocks/llm.py @@ -163,7 +163,7 @@ def llm_call( else: raise ValueError(f"Unsupported LLM provider: {provider}") - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: prompt = [] def trim_prompt(s: str) -> str: @@ -290,7 +290,7 @@ def llm_call(input_data: AIStructuredResponseGeneratorBlock.Input) -> str: raise output_data raise ValueError("Failed to get a response from the LLM.") - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: object_input_data = AIStructuredResponseGeneratorBlock.Input( **{attr: getattr(input_data, attr) for attr in input_data.model_fields}, @@ -332,7 +332,7 @@ def __init__(self): }, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: for output in self._run(input_data): yield output @@ -531,7 +531,7 @@ def llm_call( else: raise ValueError(f"Unsupported LLM provider: {provider}") - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: api_key = ( input_data.api_key.get_secret_value() diff --git a/rnd/autogpt_server/autogpt_server/blocks/maths.py b/rnd/autogpt_server/autogpt_server/blocks/maths.py index 04fa4a97b940..6a057157e6f9 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/maths.py +++ b/rnd/autogpt_server/autogpt_server/blocks/maths.py @@ -51,7 +51,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: operation = input_data.operation a = input_data.a b = input_data.b @@ -105,7 +105,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: collection = input_data.collection try: diff --git a/rnd/autogpt_server/autogpt_server/blocks/medium.py b/rnd/autogpt_server/autogpt_server/blocks/medium.py index d352f8fa1297..8315192e28c9 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/medium.py +++ b/rnd/autogpt_server/autogpt_server/blocks/medium.py @@ -136,7 +136,7 @@ def create_post( return response.json() - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: response = self.create_post( input_data.api_key.get_secret_value(), diff --git a/rnd/autogpt_server/autogpt_server/blocks/reddit.py b/rnd/autogpt_server/autogpt_server/blocks/reddit.py index 1e6f025b7f8e..0eeec64c39bb 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/reddit.py +++ b/rnd/autogpt_server/autogpt_server/blocks/reddit.py @@ -116,7 +116,7 @@ def get_posts(input_data: Input) -> Iterator[praw.reddit.Submission]: subreddit = client.subreddit(input_data.subreddit) return subreddit.new(limit=input_data.post_limit) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: current_time = datetime.now(tz=timezone.utc) for post in self.get_posts(input_data): if input_data.last_minutes: @@ -167,5 +167,5 @@ def reply_post(creds: RedditCredentials, comment: RedditComment) -> str: comment = submission.reply(comment.comment) return comment.id # type: ignore - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: yield "comment_id", self.reply_post(input_data.creds, input_data.data) diff --git a/rnd/autogpt_server/autogpt_server/blocks/rss.py b/rnd/autogpt_server/autogpt_server/blocks/rss.py index 834360823e51..5dffbdb51704 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/rss.py +++ b/rnd/autogpt_server/autogpt_server/blocks/rss.py @@ -86,7 +86,7 @@ def __init__(self): def parse_feed(url: str) -> dict[str, Any]: return feedparser.parse(url) # type: ignore - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: keep_going = True start_time = datetime.now(timezone.utc) - timedelta( minutes=input_data.time_period diff --git a/rnd/autogpt_server/autogpt_server/blocks/search.py b/rnd/autogpt_server/autogpt_server/blocks/search.py index a09203f26836..006263549e87 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/search.py +++ b/rnd/autogpt_server/autogpt_server/blocks/search.py @@ -35,7 +35,7 @@ def __init__(self): test_mock={"get_request": lambda url, json: {"extract": "summary content"}}, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: topic = input_data.topic url = f"https://en.wikipedia.org/api/rest_v1/page/summary/{topic}" @@ -72,7 +72,7 @@ def __init__(self): test_mock={"get_request": lambda url, json: "search content"}, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: # Encode the search query encoded_query = quote(input_data.query) @@ -113,7 +113,7 @@ def __init__(self): test_mock={"get_request": lambda url, json: "scraped content"}, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: # Prepend the Jina-ai Reader URL to the input URL jina_url = f"https://r.jina.ai/{input_data.url}" @@ -166,7 +166,7 @@ def __init__(self): }, ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: units = "metric" if input_data.use_celsius else "imperial" api_key = input_data.api_key.get_secret_value() diff --git a/rnd/autogpt_server/autogpt_server/blocks/talking_head.py b/rnd/autogpt_server/autogpt_server/blocks/talking_head.py index 03dec4b46c7c..081bf1105c92 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/talking_head.py +++ b/rnd/autogpt_server/autogpt_server/blocks/talking_head.py @@ -105,7 +105,7 @@ def get_clip_status(self, api_key: str, clip_id: str) -> dict: response.raise_for_status() return response.json() - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: # Create the clip payload = { diff --git a/rnd/autogpt_server/autogpt_server/blocks/text.py b/rnd/autogpt_server/autogpt_server/blocks/text.py index ae36abc26477..4771aab7d85f 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/text.py +++ b/rnd/autogpt_server/autogpt_server/blocks/text.py @@ -45,7 +45,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: output = input_data.data or input_data.text flags = 0 if not input_data.case_sensitive: @@ -97,7 +97,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: flags = 0 if not input_data.case_sensitive: flags = flags | re.IGNORECASE @@ -147,7 +147,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: # For python.format compatibility: replace all {...} with {{..}}. # But avoid replacing {{...}} to {{{...}}}. fmt = re.sub(r"(?}", input_data.format) @@ -180,6 +180,6 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: combined_text = input_data.delimiter.join(input_data.input) yield "output", combined_text diff --git a/rnd/autogpt_server/autogpt_server/blocks/time_blocks.py b/rnd/autogpt_server/autogpt_server/blocks/time_blocks.py index 10272e0b176e..6aff87c8b4d0 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/time_blocks.py +++ b/rnd/autogpt_server/autogpt_server/blocks/time_blocks.py @@ -27,7 +27,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: current_time = time.strftime("%H:%M:%S") yield "time", current_time @@ -59,7 +59,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: offset = int(input_data.offset) except ValueError: @@ -96,7 +96,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: current_date_time = time.strftime("%Y-%m-%d %H:%M:%S") yield "date_time", current_date_time @@ -129,8 +129,7 @@ def __init__(self): ], ) - def run(self, input_data: Input) -> BlockOutput: - + def run(self, input_data: Input, **kwargs) -> BlockOutput: seconds = int(input_data.seconds) minutes = int(input_data.minutes) hours = int(input_data.hours) diff --git a/rnd/autogpt_server/autogpt_server/blocks/youtube.py b/rnd/autogpt_server/autogpt_server/blocks/youtube.py index cae3bac9f913..25bcfd79c90f 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/youtube.py +++ b/rnd/autogpt_server/autogpt_server/blocks/youtube.py @@ -62,7 +62,7 @@ def extract_video_id(url: str) -> str: def get_transcript(video_id: str): return YouTubeTranscriptApi.get_transcript(video_id) - def run(self, input_data: Input) -> BlockOutput: + def run(self, input_data: Input, **kwargs) -> BlockOutput: try: video_id = self.extract_video_id(input_data.youtube_url) yield "video_id", video_id diff --git a/rnd/autogpt_server/autogpt_server/data/block.py b/rnd/autogpt_server/autogpt_server/data/block.py index 19dcdcd24122..da92cddc4906 100644 --- a/rnd/autogpt_server/autogpt_server/data/block.py +++ b/rnd/autogpt_server/autogpt_server/data/block.py @@ -36,7 +36,7 @@ class BlockCategory(Enum): INPUT = "Block that interacts with input of the graph." OUTPUT = "Block that interacts with output of the graph." LOGIC = "Programming logic to control the flow of your agent" - DEVELOPER_TOOLS = "Developer tools like Github." # added this so all the github blocks are in the same category for now, this is to be changed + DEVELOPER_TOOLS = "Developer tools like Github." # added this so all the github blocks are in the same category for now, this is to be changed def dict(self) -> dict[str, str]: return {"category": self.name, "description": self.value} @@ -179,7 +179,7 @@ def __init__( self.ui_type = ui_type @abstractmethod - def run(self, input_data: BlockSchemaInputType) -> BlockOutput: + def run(self, input_data: BlockSchemaInputType, **kwargs) -> BlockOutput: """ Run the block with the given input data. Args: @@ -210,13 +210,15 @@ def to_dict(self): "uiType": self.ui_type.value, } - def execute(self, input_data: BlockInput) -> BlockOutput: + def execute(self, input_data: BlockInput, **kwargs) -> BlockOutput: if error := self.input_schema.validate_data(input_data): raise ValueError( f"Unable to execute block with invalid input data: {error}" ) - for output_name, output_data in self.run(self.input_schema(**input_data)): + for output_name, output_data in self.run( + self.input_schema(**input_data), **kwargs + ): if error := self.output_schema.validate_field(output_name, output_data): raise ValueError(f"Block produced an invalid output data: {error}") yield output_name, output_data From 08666f485d9ceae00cc330abcb750f52e8259365 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 12 Sep 2024 01:45:45 +0200 Subject: [PATCH 06/43] feat(libs/supabase_integration_credentials_store): Add `CredentialsType` alias --- .../supabase_integration_credentials_store/types.py | 3 +++ rnd/autogpt_server/autogpt_server/server/integrations.py | 5 +++-- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py b/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py index 834da3296721..537e1e744437 100644 --- a/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py +++ b/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py @@ -41,6 +41,9 @@ class APIKeyCredentials(_BaseCredentials): ] +CredentialsType = Literal["api_key", "oauth2"] + + class OAuthState(BaseModel): token: str provider: str diff --git a/rnd/autogpt_server/autogpt_server/server/integrations.py b/rnd/autogpt_server/autogpt_server/server/integrations.py index cf1a3126757d..de98108c019f 100644 --- a/rnd/autogpt_server/autogpt_server/server/integrations.py +++ b/rnd/autogpt_server/autogpt_server/server/integrations.py @@ -1,9 +1,10 @@ import logging -from typing import Annotated, Literal +from typing import Annotated from autogpt_libs.supabase_integration_credentials_store import ( SupabaseIntegrationCredentialsStore, ) +from autogpt_libs.supabase_integration_credentials_store.types import CredentialsType from fastapi import APIRouter, Body, Depends, HTTPException, Path, Query, Request from pydantic import BaseModel from supabase import Client @@ -49,7 +50,7 @@ async def login( class CredentialsMetaResponse(BaseModel): credentials_id: str - credentials_type: Literal["oauth2", "api_key"] + credentials_type: CredentialsType @integrations_api_router.post("/{provider}/callback") From 14c8ca4c63eea288fa1a761f2cd3304974a96163 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 12 Sep 2024 01:59:52 +0200 Subject: [PATCH 07/43] Add strict support for `credentials` fields on blocks --- .../autogpt_server/data/block.py | 46 ++++++++++++++++++- .../autogpt_server/data/model.py | 46 ++++++++++++++++++- 2 files changed, 89 insertions(+), 3 deletions(-) diff --git a/rnd/autogpt_server/autogpt_server/data/block.py b/rnd/autogpt_server/autogpt_server/data/block.py index da92cddc4906..feed34f89d07 100644 --- a/rnd/autogpt_server/autogpt_server/data/block.py +++ b/rnd/autogpt_server/autogpt_server/data/block.py @@ -1,15 +1,17 @@ +import inspect from abc import ABC, abstractmethod from enum import Enum -from typing import Any, ClassVar, Generator, Generic, Type, TypeVar, cast +from typing import Any, ClassVar, Generator, Generic, Type, TypeVar, cast, get_origin import jsonref import jsonschema from prisma.models import AgentBlock from pydantic import BaseModel -from autogpt_server.data.model import ContributorDetails from autogpt_server.util import json +from .model import CREDENTIALS_FIELD_NAME, ContributorDetails, CredentialsMetaInput + BlockData = tuple[str, Any] # Input & Output data should be a tuple of (name, data). BlockInput = dict[str, Any] # Input: 1 input pin consumes 1 data. BlockOutput = Generator[BlockData, None, None] # Output: 1 output pin produces n data. @@ -123,6 +125,46 @@ def get_required_fields(cls) -> set[str]: if field_info.is_required() } + @classmethod + def __pydantic_init_subclass__(cls, **kwargs): + """Validates the schema definition. Rules: + - Only one `CredentialsMetaInput` field may be present. + - This field MUST be called `credentials`. + - A field that is called `credentials` MUST be a `CredentialsMetaInput`. + """ + super().__pydantic_init_subclass__(**kwargs) + credentials_fields = [ + field_name + for field_name, info in cls.model_fields.items() + if ( + inspect.isclass(info.annotation) + and issubclass( + get_origin(info.annotation) or info.annotation, + CredentialsMetaInput, + ) + ) + ] + if len(credentials_fields) > 1: + raise ValueError( + f"{cls.__qualname__} can only have one CredentialsMetaInput field" + ) + elif ( + len(credentials_fields) == 1 + and credentials_fields[0] != CREDENTIALS_FIELD_NAME + ): + raise ValueError( + f"CredentialsMetaInput field on {cls.__qualname__} " + "must be named 'credentials'" + ) + elif ( + len(credentials_fields) == 0 + and CREDENTIALS_FIELD_NAME in cls.model_fields.keys() + ): + raise TypeError( + f"Field 'credentials' on {cls.__qualname__} " + f"must be of type {CredentialsMetaInput.__name__}" + ) + BlockSchemaInputType = TypeVar("BlockSchemaInputType", bound=BlockSchema) BlockSchemaOutputType = TypeVar("BlockSchemaOutputType", bound=BlockSchema) diff --git a/rnd/autogpt_server/autogpt_server/data/model.py b/rnd/autogpt_server/autogpt_server/data/model.py index 1893c67d459c..08c9622d5672 100644 --- a/rnd/autogpt_server/autogpt_server/data/model.py +++ b/rnd/autogpt_server/autogpt_server/data/model.py @@ -1,8 +1,9 @@ from __future__ import annotations import logging -from typing import Any, Callable, ClassVar, Optional, TypeVar +from typing import Any, Callable, ClassVar, Generic, Optional, TypeVar +from autogpt_libs.supabase_integration_credentials_store.types import CredentialsType from pydantic import BaseModel, Field, GetCoreSchemaHandler from pydantic_core import ( CoreSchema, @@ -136,5 +137,48 @@ def SchemaField( ) +CP = TypeVar("CP", bound=str) +CT = TypeVar("CT", bound=CredentialsType) + + +CREDENTIALS_FIELD_NAME = "credentials" + + +class CredentialsMetaInput(BaseModel, Generic[CP, CT]): + id: str + title: str + provider: CP + type: CT + + +def CredentialsField( + provider: CP, + supported_credential_types: set[CT], + *, + title: Optional[str] = None, + description: Optional[str] = None, + **kwargs, +) -> CredentialsMetaInput[CP, CT]: + """ + `CredentialsField` must and can only be used on fields named `credentials`. + This is enforced by the `BlockSchema` base class. + """ + json_extra = { + k: v + for k, v in { + "credentials_provider": provider, + "credentials_types": list(supported_credential_types), + }.items() + if v is not None + } + + return Field( + title=title, + description=description, + json_schema_extra=json_extra, + **kwargs, + ) + + class ContributorDetails(BaseModel): name: str = Field(title="Name", description="The name of the contributor.") From da6d443984008aa69ec2bf07d9563bb477f4b306 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 12 Sep 2024 02:01:49 +0200 Subject: [PATCH 08/43] feat(executor): Fetch credentials for graph execution and pass them down through to the node execution --- .../autogpt_server/data/execution.py | 2 + .../autogpt_server/executor/manager.py | 93 +++++++++++++++++-- 2 files changed, 89 insertions(+), 6 deletions(-) diff --git a/rnd/autogpt_server/autogpt_server/data/execution.py b/rnd/autogpt_server/autogpt_server/data/execution.py index 2ba23a7d3a0f..4da091937481 100644 --- a/rnd/autogpt_server/autogpt_server/data/execution.py +++ b/rnd/autogpt_server/autogpt_server/data/execution.py @@ -4,6 +4,7 @@ from multiprocessing import Manager from typing import Any, Generic, TypeVar +from autogpt_libs.supabase_integration_credentials_store.types import Credentials from prisma.models import ( AgentGraphExecution, AgentNodeExecution, @@ -24,6 +25,7 @@ class GraphExecution(BaseModel): graph_exec_id: str start_node_execs: list["NodeExecution"] graph_id: str + node_input_credentials: dict[str, Credentials] # dict[node_id, Credentials] class NodeExecution(BaseModel): diff --git a/rnd/autogpt_server/autogpt_server/executor/manager.py b/rnd/autogpt_server/autogpt_server/executor/manager.py index 42ca2be57b1b..d5bd72a49fee 100644 --- a/rnd/autogpt_server/autogpt_server/executor/manager.py +++ b/rnd/autogpt_server/autogpt_server/executor/manager.py @@ -9,7 +9,10 @@ from concurrent.futures import Future, ProcessPoolExecutor from contextlib import contextmanager from multiprocessing.pool import AsyncResult, Pool -from typing import TYPE_CHECKING, Any, Coroutine, Generator, TypeVar +from typing import TYPE_CHECKING, Any, Coroutine, Generator, TypeVar, cast + +from autogpt_libs.supabase_integration_credentials_store.types import Credentials +from pydantic import BaseModel if TYPE_CHECKING: from autogpt_server.server.rest_api import AgentServer @@ -35,6 +38,7 @@ upsert_execution_output, ) from autogpt_server.data.graph import Graph, Link, Node, get_graph, get_node +from autogpt_server.data.model import CREDENTIALS_FIELD_NAME, CredentialsMetaInput from autogpt_server.util import json from autogpt_server.util.decorator import error_logged, time_measured from autogpt_server.util.logging import configure_logging @@ -70,6 +74,7 @@ def execute_node( loop: asyncio.AbstractEventLoop, api_client: "AgentServer", data: NodeExecution, + input_credentials: Credentials | None = None, execution_stats: dict[str, Any] | None = None, ) -> ExecutionStream: """ @@ -131,9 +136,15 @@ def update_execution(status: ExecutionStatus): ) update_execution(ExecutionStatus.RUNNING) + extra_exec_kwargs = {} + if input_credentials: + extra_exec_kwargs["credentials"] = input_credentials + output_size = 0 try: - for output_name, output_data in node_block.execute(input_data): + for output_name, output_data in node_block.execute( + input_data, **extra_exec_kwargs + ): output_size += len(json.dumps(output_data)) logger.info( "Node produced output", @@ -432,7 +443,10 @@ def on_node_executor_sigterm(cls): @classmethod @error_logged def on_node_execution( - cls, q: ExecutionQueue[NodeExecution], node_exec: NodeExecution + cls, + q: ExecutionQueue[NodeExecution], + node_exec: NodeExecution, + input_credentials: Credentials | None, ): log_metadata = get_log_metadata( graph_eid=node_exec.graph_exec_id, @@ -444,7 +458,7 @@ def on_node_execution( execution_stats = {} timing_info, _ = cls._on_node_execution( - q, node_exec, log_metadata, execution_stats + q, node_exec, input_credentials, log_metadata, execution_stats ) execution_stats["walltime"] = timing_info.wall_time execution_stats["cputime"] = timing_info.cpu_time @@ -459,6 +473,7 @@ def _on_node_execution( cls, q: ExecutionQueue[NodeExecution], node_exec: NodeExecution, + input_credentials: Credentials | None, log_metadata: dict, stats: dict[str, Any] | None = None, ): @@ -468,7 +483,7 @@ def _on_node_execution( extra={"json_fields": {**log_metadata}}, ) for execution in execute_node( - cls.loop, cls.agent_server_client, node_exec, stats + cls.loop, cls.agent_server_client, node_exec, input_credentials, stats ): q.add(execution) logger.info( @@ -605,7 +620,11 @@ def callback(_): ) running_executions[exec_data.node_id] = cls.executor.apply_async( cls.on_node_execution, - (queue, exec_data), + ( + queue, + exec_data, + graph_exec.node_input_credentials.get(exec_data.node_id), + ), callback=make_exec_callback(exec_data), ) @@ -650,11 +669,17 @@ class ExecutionManager(AppService): def __init__(self): super().__init__(port=Config().execution_manager_port) self.use_db = True + self.use_supabase = True self.pool_size = Config().num_graph_workers self.queue = ExecutionQueue[GraphExecution]() self.active_graph_runs: dict[str, tuple[Future, threading.Event]] = {} def run_service(self): + from autogpt_libs.supabase_integration_credentials_store import ( + SupabaseIntegrationCredentialsStore, + ) + + self.credentials_store = SupabaseIntegrationCredentialsStore(self.supabase) self.executor = ProcessPoolExecutor( max_workers=self.pool_size, initializer=Executor.on_graph_executor_start, @@ -695,7 +720,10 @@ def add_execution( graph: Graph | None = self.run_and_wait(get_graph(graph_id, user_id=user_id)) if not graph: raise Exception(f"Graph #{graph_id} not found.") + graph.validate_graph(for_run=True) + node_input_credentials = self._get_node_input_credentials(graph, user_id) + nodes_input = [] for node in graph.starting_nodes: input_data = {} @@ -741,6 +769,7 @@ def add_execution( graph_id=graph_id, graph_exec_id=graph_exec_id, start_node_execs=starting_node_execs, + node_input_credentials=node_input_credentials, ) self.queue.add(graph_exec) @@ -787,6 +816,58 @@ def cancel_execution(self, graph_exec_id: str) -> None: ) self.agent_server_client.send_execution_update(exec_update.model_dump()) + def _get_node_input_credentials( + self, graph: Graph, user_id: str + ) -> dict[str, Credentials]: + """Gets all credentials for all nodes of the graph""" + + node_credentials: dict[str, Credentials] = {} + + for node in graph.nodes: + block = get_block(node.block_id) + if not block: + raise ValueError(f"Unknown block {node.block_id} for node #{node.id}") + + # Find any fields of type CredentialsMetaInput + model_fields = cast(type[BaseModel], block.input_schema).model_fields + if CREDENTIALS_FIELD_NAME not in model_fields: + continue + + field = model_fields[CREDENTIALS_FIELD_NAME] + + # The BlockSchema class enforces that a `credentials` field is always a + # `CredentialsMetaInput`, so we can safely assume this here. + credentials_meta_type = cast(CredentialsMetaInput, field.annotation) + credentials_meta = credentials_meta_type.model_validate( + node.input_default[CREDENTIALS_FIELD_NAME] + ) + # Fetch the corresponding Credentials and perform sanity checks + credentials = self.credentials_store.get_creds_by_id( + user_id, credentials_meta.id + ) + if not credentials: + raise ValueError( + f"Unknown credentials #{credentials_meta.id} " + f"for node #{node.id}" + ) + if ( + credentials.provider != credentials_meta.provider + or credentials.type != credentials_meta.type + ): + logger.warning( + f"Invalid credentials #{credentials.id} for node #{node.id}: " + "type/provider mismatch: " + f"{credentials_meta.type}<>{credentials.type};" + f"{credentials_meta.provider}<>{credentials.provider}" + ) + raise ValueError( + f"Invalid credentials #{credentials.id} for node #{node.id}: " + "type/provider mismatch" + ) + node_credentials[node.id] = credentials + + return node_credentials + def llprint(message: str): """ From 56d379f2a7c3d2001cf7a4fcc3d1214db5d23bd8 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 12 Sep 2024 02:03:27 +0200 Subject: [PATCH 09/43] feat(server/blocks): Integrate GitHub blocks with new credentials infrastructure --- .../types.py | 6 + .../autogpt_server/blocks/github.py | 526 +++++++++++------- 2 files changed, 320 insertions(+), 212 deletions(-) diff --git a/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py b/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py index 537e1e744437..0274552c01c3 100644 --- a/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py +++ b/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py @@ -27,6 +27,9 @@ class OAuth2Credentials(_BaseCredentials): scopes: list[str] metadata: dict[str, Any] = Field(default_factory=dict) + def bearer(self) -> str: + return f"Bearer {self.access_token.get_secret_value()}" + class APIKeyCredentials(_BaseCredentials): type: Literal["api_key"] = "api_key" @@ -34,6 +37,9 @@ class APIKeyCredentials(_BaseCredentials): expires_at: Optional[int] """Unix timestamp (seconds) indicating when the API key expires (if at all)""" + def bearer(self) -> str: + return f"Bearer {self.api_key.get_secret_value()}" + Credentials = Annotated[ OAuth2Credentials | APIKeyCredentials, diff --git a/rnd/autogpt_server/autogpt_server/blocks/github.py b/rnd/autogpt_server/autogpt_server/blocks/github.py index b46f452c2aa6..4800e578b14b 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/github.py +++ b/rnd/autogpt_server/autogpt_server/blocks/github.py @@ -1,20 +1,34 @@ import base64 +from typing import Literal import requests -from pydantic import BaseModel, ConfigDict, Field +from autogpt_libs.supabase_integration_credentials_store.types import ( + APIKeyCredentials, + OAuth2Credentials, +) from autogpt_server.data.block import Block, BlockCategory, BlockOutput, BlockSchema -from autogpt_server.data.model import BlockSecret, SchemaField, SecretField +from autogpt_server.data.model import ( + CredentialsField, + CredentialsMetaInput, + SchemaField, +) +GithubCredentials = APIKeyCredentials | OAuth2Credentials +GithubCredentialsInput = CredentialsMetaInput[ + Literal["github"], Literal["api_key", "oauth2"] +] -class GithubCredentials(BaseModel): - github_oauth_token: BlockSecret = SecretField(key="github_oauth_token") - - model_config = ConfigDict(title="GitHub Credentials") +GITHUB_CREDS_FIELD: GithubCredentialsInput = CredentialsField( + provider="github", + supported_credential_types={"api_key", "oauth2"}, + description="GitHub OAuth credentials", +) class GithubCommentBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD issue_url: str = SchemaField( description="URL of the GitHub issue or pull request", placeholder="https://github.com/owner/repo/issues/1", @@ -23,10 +37,6 @@ class Input(BlockSchema): description="Comment to post on the issue or pull request", placeholder="Enter your comment", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField(description="Status of the comment posting operation") @@ -44,7 +54,7 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "comment": "This is a test comment.", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -55,7 +65,9 @@ def __init__(self): ) @staticmethod - def post_comment(creds: GithubCredentials, issue_url: str, comment: str) -> str: + def post_comment( + credentials: GithubCredentials, issue_url: str, comment: str + ) -> str: try: if "/pull/" in issue_url: api_url = ( @@ -71,7 +83,7 @@ def post_comment(creds: GithubCredentials, issue_url: str, comment: str) -> str: ) headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } data = {"body": comment} @@ -83,9 +95,15 @@ def post_comment(creds: GithubCredentials, issue_url: str, comment: str) -> str: except Exception as e: return f"Failed to post comment: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.post_comment( - input_data.creds, + credentials, input_data.issue_url, input_data.comment, ) @@ -97,6 +115,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubMakeIssueBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -107,10 +126,6 @@ class Input(BlockSchema): body: str = SchemaField( description="Body of the issue", placeholder="Enter the issue body" ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField(description="Status of the issue creation operation") @@ -129,7 +144,7 @@ def __init__(self): "repo_url": "https://github.com/owner/repo", "title": "Test Issue", "body": "This is a test issue.", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -141,12 +156,12 @@ def __init__(self): @staticmethod def create_issue( - creds: GithubCredentials, repo_url: str, title: str, body: str + credentials: GithubCredentials, repo_url: str, title: str, body: str ) -> str: try: api_url = repo_url.replace("github.com", "api.github.com/repos") + "/issues" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } data = {"title": title, "body": body} @@ -158,9 +173,15 @@ def create_issue( except Exception as e: return f"Failed to create issue: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.create_issue( - input_data.creds, + credentials, input_data.repo_url, input_data.title, input_data.body, @@ -173,6 +194,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubMakePRBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -193,10 +215,6 @@ class Input(BlockSchema): description="The name of the branch you want the changes pulled into.", placeholder="Enter the base branch", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField( @@ -219,7 +237,7 @@ def __init__(self): "body": "This is a test pull request.", "head": "feature-branch", "base": "main", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -231,7 +249,7 @@ def __init__(self): @staticmethod def create_pr( - creds: GithubCredentials, + credentials: GithubCredentials, repo_url: str, title: str, body: str, @@ -243,7 +261,7 @@ def create_pr( repo_path = repo_url.replace("https://github.com/", "") api_url = f"https://api.github.com/repos/{repo_path}/pulls" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } data = {"title": title, "body": body, "head": head, "base": base} @@ -260,9 +278,15 @@ def create_pr( except Exception as e: return f"Failed to create pull request: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.create_pr( - input_data.creds, + credentials, input_data.repo_url, input_data.title, input_data.body, @@ -277,14 +301,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadIssueBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD issue_url: str = SchemaField( description="URL of the GitHub issue", placeholder="https://github.com/owner/repo/issues/1", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): title: str = SchemaField(description="Title of the issue") @@ -303,7 +324,7 @@ def __init__(self): output_schema=GithubReadIssueBlock.Output, test_input={ "issue_url": "https://github.com/owner/repo/issues/1", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -322,12 +343,14 @@ def __init__(self): ) @staticmethod - def read_issue(creds: GithubCredentials, issue_url: str) -> tuple[str, str, str]: + def read_issue( + credentials: GithubCredentials, issue_url: str + ) -> tuple[str, str, str]: try: api_url = issue_url.replace("github.com", "api.github.com/repos") headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -343,9 +366,15 @@ def read_issue(creds: GithubCredentials, issue_url: str) -> tuple[str, str, str] except Exception as e: return f"Failed to read issue: {str(e)}", "", "" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: title, body, user = self.read_issue( - input_data.creds, + credentials, input_data.issue_url, ) if "Failed" in title: @@ -358,14 +387,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadPRBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD pr_url: str = SchemaField( description="URL of the GitHub pull request", placeholder="https://github.com/owner/repo/pull/1", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) include_pr_changes: bool = SchemaField( description="Whether to include the changes made in the pull request", default=False, @@ -389,7 +415,7 @@ def __init__(self): output_schema=GithubReadPRBlock.Output, test_input={ "pr_url": "https://github.com/owner/repo/pull/1", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, "include_pr_changes": True, @@ -411,14 +437,14 @@ def __init__(self): ) @staticmethod - def read_pr(creds: GithubCredentials, pr_url: str) -> tuple[str, str, str]: + def read_pr(credentials: GithubCredentials, pr_url: str) -> tuple[str, str, str]: try: api_url = pr_url.replace("github.com", "api.github.com/repos").replace( "/pull/", "/issues/" ) headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -435,7 +461,7 @@ def read_pr(creds: GithubCredentials, pr_url: str) -> tuple[str, str, str]: return f"Failed to read pull request: {str(e)}", "", "" @staticmethod - def read_pr_changes(creds: GithubCredentials, pr_url: str) -> str: + def read_pr_changes(credentials: GithubCredentials, pr_url: str) -> str: try: api_url = ( pr_url.replace("github.com", "api.github.com/repos").replace( @@ -445,7 +471,7 @@ def read_pr_changes(creds: GithubCredentials, pr_url: str) -> str: ) headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -464,9 +490,15 @@ def read_pr_changes(creds: GithubCredentials, pr_url: str) -> str: except Exception as e: return f"Failed to read PR changes: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: title, body, user = self.read_pr( - input_data.creds, + credentials, input_data.pr_url, ) if "Failed" in title: @@ -478,7 +510,7 @@ def run(self, input_data: Input) -> BlockOutput: if input_data.include_pr_changes: changes = self.read_pr_changes( - input_data.creds, + credentials, input_data.pr_url, ) if "Failed" in changes: @@ -491,14 +523,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubListIssuesBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): issues: list[dict[str, str]] = SchemaField( @@ -515,7 +544,7 @@ def __init__(self): output_schema=GithubListIssuesBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -541,11 +570,13 @@ def __init__(self): ) @staticmethod - def list_issues(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: + def list_issues( + credentials: GithubCredentials, repo_url: str + ) -> list[dict[str, str]]: try: api_url = repo_url.replace("github.com", "api.github.com/repos") + "/issues" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -561,9 +592,15 @@ def list_issues(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]] except Exception as e: return [{"title": "Error", "url": f"Failed to list issues: {str(e)}"}] - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: issues = self.list_issues( - input_data.creds, + credentials, input_data.repo_url, ) if any("Failed" in issue["url"] for issue in issues): @@ -574,14 +611,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadTagsBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): tags: list[dict[str, str]] = SchemaField( @@ -598,7 +632,7 @@ def __init__(self): output_schema=GithubReadTagsBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -624,12 +658,14 @@ def __init__(self): ) @staticmethod - def list_tags(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: + def list_tags( + credentials: GithubCredentials, repo_url: str + ) -> list[dict[str, str]]: try: repo_path = repo_url.replace("https://github.com/", "") api_url = f"https://api.github.com/repos/{repo_path}/tags" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -649,9 +685,15 @@ def list_tags(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: except Exception as e: return [{"name": "Error", "url": f"Failed to list tags: {str(e)}"}] - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: tags = self.list_tags( - input_data.creds, + credentials, input_data.repo_url, ) if any("Failed" in tag["url"] for tag in tags): @@ -662,14 +704,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadBranchesBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): branches: list[dict[str, str]] = SchemaField( @@ -686,7 +725,7 @@ def __init__(self): output_schema=GithubReadBranchesBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -712,13 +751,15 @@ def __init__(self): ) @staticmethod - def list_branches(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: + def list_branches( + credentials: GithubCredentials, repo_url: str + ) -> list[dict[str, str]]: try: api_url = ( repo_url.replace("github.com", "api.github.com/repos") + "/branches" ) headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -735,9 +776,15 @@ def list_branches(creds: GithubCredentials, repo_url: str) -> list[dict[str, str except Exception as e: return [{"name": "Error", "url": f"Failed to list branches: {str(e)}"}] - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: branches = self.list_branches( - input_data.creds, + credentials, input_data.repo_url, ) if any("Failed" in branch["url"] for branch in branches): @@ -748,14 +795,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadDiscussionsBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) num_discussions: int = SchemaField( description="Number of discussions to fetch", default=5 ) @@ -777,7 +821,7 @@ def __init__(self): output_schema=GithubReadDiscussionsBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, "num_discussions": 3, @@ -805,7 +849,7 @@ def __init__(self): @staticmethod def list_discussions( - creds: GithubCredentials, repo_url: str, num_discussions: int + credentials: GithubCredentials, repo_url: str, num_discussions: int ) -> list[dict[str, str]]: try: repo_path = repo_url.replace("https://github.com/", "") @@ -824,7 +868,7 @@ def list_discussions( """ variables = {"owner": owner, "repo": repo, "num": num_discussions} headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -845,9 +889,15 @@ def list_discussions( except Exception as e: return [{"title": "Error", "url": f"Failed to list discussions: {str(e)}"}] - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: discussions = self.list_discussions( - input_data.creds, input_data.repo_url, input_data.num_discussions + credentials, input_data.repo_url, input_data.num_discussions ) if any("Failed" in discussion["url"] for discussion in discussions): yield "error", discussions[0]["url"] @@ -857,14 +907,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadReleasesBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): releases: list[dict[str, str]] = SchemaField( @@ -881,7 +928,7 @@ def __init__(self): output_schema=GithubReadReleasesBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -907,12 +954,14 @@ def __init__(self): ) @staticmethod - def list_releases(creds: GithubCredentials, repo_url: str) -> list[dict[str, str]]: + def list_releases( + credentials: GithubCredentials, repo_url: str + ) -> list[dict[str, str]]: try: repo_path = repo_url.replace("https://github.com/", "") api_url = f"https://api.github.com/repos/{repo_path}/releases" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -929,9 +978,15 @@ def list_releases(creds: GithubCredentials, repo_url: str) -> list[dict[str, str except Exception as e: return [{"name": "Error", "url": f"Failed to list releases: {str(e)}"}] - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: releases = self.list_releases( - input_data.creds, + credentials, input_data.repo_url, ) if any("Failed" in release["url"] for release in releases): @@ -942,6 +997,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubAddLabelBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD issue_url: str = SchemaField( description="URL of the GitHub issue or pull request", placeholder="https://github.com/owner/repo/issues/1", @@ -950,10 +1006,6 @@ class Input(BlockSchema): description="Label to add to the issue or pull request", placeholder="Enter the label", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField(description="Status of the label addition operation") @@ -971,7 +1023,7 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "label": "bug", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -980,7 +1032,7 @@ def __init__(self): ) @staticmethod - def add_label(creds: GithubCredentials, issue_url: str, label: str) -> str: + def add_label(credentials: GithubCredentials, issue_url: str, label: str) -> str: try: # Convert the provided GitHub URL to the API URL if "/pull/" in issue_url: @@ -999,7 +1051,7 @@ def add_label(creds: GithubCredentials, issue_url: str, label: str) -> str: print(f"Constructed API URL: {api_url}") headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } data = {"labels": [label]} @@ -1009,13 +1061,19 @@ def add_label(creds: GithubCredentials, issue_url: str, label: str) -> str: return "Label added successfully" except requests.exceptions.HTTPError as http_err: - return f"HTTP error occurred: {http_err} - {response.text}" + return f"HTTP error occurred: {http_err} - {http_err.response.text}" except Exception as e: return f"Failed to add label: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.add_label( - input_data.creds, + credentials, input_data.issue_url, input_data.label, ) @@ -1027,6 +1085,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubRemoveLabelBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD issue_url: str = SchemaField( description="URL of the GitHub issue or pull request", placeholder="https://github.com/owner/repo/issues/1", @@ -1035,10 +1094,6 @@ class Input(BlockSchema): description="Label to remove from the issue or pull request", placeholder="Enter the label", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField(description="Status of the label removal operation") @@ -1056,7 +1111,7 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "label": "bug", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1067,7 +1122,7 @@ def __init__(self): ) @staticmethod - def remove_label(creds: GithubCredentials, issue_url: str, label: str) -> str: + def remove_label(credentials: GithubCredentials, issue_url: str, label: str) -> str: try: # Convert the provided GitHub URL to the API URL if "/pull/" in issue_url: @@ -1087,7 +1142,7 @@ def remove_label(creds: GithubCredentials, issue_url: str, label: str) -> str: print(f"Constructed API URL: {api_url}") headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -1096,13 +1151,19 @@ def remove_label(creds: GithubCredentials, issue_url: str, label: str) -> str: return "Label removed successfully" except requests.exceptions.HTTPError as http_err: - return f"HTTP error occurred: {http_err} - {response.text}" + return f"HTTP error occurred: {http_err} - {http_err.response.text}" except Exception as e: return f"Failed to remove label: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.remove_label( - input_data.creds, + credentials, input_data.issue_url, input_data.label, ) @@ -1114,6 +1175,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubAssignReviewerBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD pr_url: str = SchemaField( description="URL of the GitHub pull request", placeholder="https://github.com/owner/repo/pull/1", @@ -1122,10 +1184,6 @@ class Input(BlockSchema): description="Username of the reviewer to assign", placeholder="Enter the reviewer's username", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField( @@ -1145,7 +1203,7 @@ def __init__(self): test_input={ "pr_url": "https://github.com/owner/repo/pull/1", "reviewer": "reviewer_username", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1156,7 +1214,9 @@ def __init__(self): ) @staticmethod - def assign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> str: + def assign_reviewer( + credentials: GithubCredentials, pr_url: str, reviewer: str + ) -> str: try: # Convert the PR URL to the appropriate API endpoint api_url = ( @@ -1170,7 +1230,7 @@ def assign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> str print(f"Constructed API URL: {api_url}") headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } data = {"reviewers": [reviewer]} @@ -1183,16 +1243,22 @@ def assign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> str return "Reviewer assigned successfully" except requests.exceptions.HTTPError as http_err: - if response.status_code == 422: - return f"Failed to assign reviewer: The reviewer '{reviewer}' may not have permission or the pull request is not in a valid state. Detailed error: {response.text}" + if http_err.response.status_code == 422: + return f"Failed to assign reviewer: The reviewer '{reviewer}' may not have permission or the pull request is not in a valid state. Detailed error: {http_err.response.text}" else: - return f"HTTP error occurred: {http_err} - {response.text}" + return f"HTTP error occurred: {http_err} - {http_err.response.text}" except Exception as e: return f"Failed to assign reviewer: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.assign_reviewer( - input_data.creds, + credentials, input_data.pr_url, input_data.reviewer, ) @@ -1204,6 +1270,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubUnassignReviewerBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD pr_url: str = SchemaField( description="URL of the GitHub pull request", placeholder="https://github.com/owner/repo/pull/1", @@ -1212,10 +1279,6 @@ class Input(BlockSchema): description="Username of the reviewer to unassign", placeholder="Enter the reviewer's username", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField( @@ -1235,7 +1298,7 @@ def __init__(self): test_input={ "pr_url": "https://github.com/owner/repo/pull/1", "reviewer": "reviewer_username", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1246,7 +1309,9 @@ def __init__(self): ) @staticmethod - def unassign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> str: + def unassign_reviewer( + credentials: GithubCredentials, pr_url: str, reviewer: str + ) -> str: try: api_url = ( pr_url.replace("github.com", "api.github.com/repos").replace( @@ -1255,7 +1320,7 @@ def unassign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> s + "/requested_reviewers" ) headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } data = {"reviewers": [reviewer]} @@ -1267,9 +1332,15 @@ def unassign_reviewer(creds: GithubCredentials, pr_url: str, reviewer: str) -> s except Exception as e: return f"Failed to unassign reviewer: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.unassign_reviewer( - input_data.creds, + credentials, input_data.pr_url, input_data.reviewer, ) @@ -1281,14 +1352,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubListReviewersBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD pr_url: str = SchemaField( description="URL of the GitHub pull request", placeholder="https://github.com/owner/repo/pull/1", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): reviewers: list[dict[str, str]] = SchemaField( @@ -1307,7 +1375,7 @@ def __init__(self): output_schema=GithubListReviewersBlock.Output, test_input={ "pr_url": "https://github.com/owner/repo/pull/1", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1333,7 +1401,9 @@ def __init__(self): ) @staticmethod - def list_reviewers(creds: GithubCredentials, pr_url: str) -> list[dict[str, str]]: + def list_reviewers( + credentials: GithubCredentials, pr_url: str + ) -> list[dict[str, str]]: try: api_url = ( pr_url.replace("github.com", "api.github.com/repos").replace( @@ -1342,7 +1412,7 @@ def list_reviewers(creds: GithubCredentials, pr_url: str) -> list[dict[str, str] + "/requested_reviewers" ) headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -1359,9 +1429,15 @@ def list_reviewers(creds: GithubCredentials, pr_url: str) -> list[dict[str, str] except Exception as e: return [{"username": "Error", "url": f"Failed to list reviewers: {str(e)}"}] - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: reviewers = self.list_reviewers( - input_data.creds, + credentials, input_data.pr_url, ) if any("Failed" in reviewer["url"] for reviewer in reviewers): @@ -1372,6 +1448,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubAssignIssueBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD issue_url: str = SchemaField( description="URL of the GitHub issue", placeholder="https://github.com/owner/repo/issues/1", @@ -1380,10 +1457,6 @@ class Input(BlockSchema): description="Username to assign to the issue", placeholder="Enter the username", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField( @@ -1403,7 +1476,7 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "assignee": "username1", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1415,7 +1488,7 @@ def __init__(self): @staticmethod def assign_issue( - creds: GithubCredentials, + credentials: GithubCredentials, issue_url: str, assignee: str, ) -> str: @@ -1427,7 +1500,7 @@ def assign_issue( api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } data = {"assignees": [assignee]} @@ -1441,9 +1514,15 @@ def assign_issue( except Exception as e: return f"Failed to assign issue: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.assign_issue( - input_data.creds, + credentials, input_data.issue_url, input_data.assignee, ) @@ -1455,6 +1534,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubUnassignIssueBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD issue_url: str = SchemaField( description="URL of the GitHub issue", placeholder="https://github.com/owner/repo/issues/1", @@ -1463,10 +1543,6 @@ class Input(BlockSchema): description="Username to unassign from the issue", placeholder="Enter the username", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField( @@ -1486,7 +1562,7 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "assignee": "username1", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1498,7 +1574,7 @@ def __init__(self): @staticmethod def unassign_issue( - creds: GithubCredentials, + credentials: GithubCredentials, issue_url: str, assignee: str, ) -> str: @@ -1510,7 +1586,7 @@ def unassign_issue( api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } data = {"assignees": [assignee]} @@ -1524,9 +1600,15 @@ def unassign_issue( except Exception as e: return f"Failed to unassign issue: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.unassign_issue( - input_data.creds, + credentials, input_data.issue_url, input_data.assignee, ) @@ -1538,14 +1620,11 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadCodeownersFileBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): codeowners_content: str = SchemaField( @@ -1562,7 +1641,7 @@ def __init__(self): output_schema=GithubReadCodeownersFileBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1573,12 +1652,12 @@ def __init__(self): ) @staticmethod - def read_codeowners(creds: GithubCredentials, repo_url: str) -> str: + def read_codeowners(credentials: GithubCredentials, repo_url: str) -> str: try: repo_path = repo_url.replace("https://github.com/", "") api_url = f"https://api.github.com/repos/{repo_path}/contents/.github/CODEOWNERS?ref=master" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -1592,9 +1671,15 @@ def read_codeowners(creds: GithubCredentials, repo_url: str) -> str: except Exception as e: return f"Failed to read CODEOWNERS file: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: content = self.read_codeowners( - input_data.creds, + credentials, input_data.repo_url, ) if "Failed" not in content: @@ -1605,6 +1690,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadFileFromMasterBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -1613,10 +1699,6 @@ class Input(BlockSchema): description="Path to the file in the repository", placeholder="path/to/file", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): file_content: str = SchemaField( @@ -1634,7 +1716,7 @@ def __init__(self): test_input={ "repo_url": "https://github.com/owner/repo", "file_path": "path/to/file", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1643,12 +1725,12 @@ def __init__(self): ) @staticmethod - def read_file(creds: GithubCredentials, repo_url: str, file_path: str) -> str: + def read_file(credentials: GithubCredentials, repo_url: str, file_path: str) -> str: try: repo_path = repo_url.replace("https://github.com/", "") api_url = f"https://api.github.com/repos/{repo_path}/contents/{file_path}?ref=master" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -1662,9 +1744,15 @@ def read_file(creds: GithubCredentials, repo_url: str, file_path: str) -> str: except Exception as e: return f"Failed to read file: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: content = self.read_file( - input_data.creds, + credentials, input_data.repo_url, input_data.file_path, ) @@ -1676,6 +1764,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubReadFileFolderRepoBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -1688,10 +1777,6 @@ class Input(BlockSchema): description="Branch name to read from", placeholder="branch_name", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): content: str = SchemaField( @@ -1710,7 +1795,7 @@ def __init__(self): "repo_url": "https://github.com/owner/repo", "path": "path/to/file_or_folder", "branch": "branch_name", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1722,7 +1807,7 @@ def __init__(self): @staticmethod def read_content( - creds: GithubCredentials, repo_url: str, path: str, branch: str + credentials: GithubCredentials, repo_url: str, path: str, branch: str ) -> str: try: repo_path = repo_url.replace("https://github.com/", "") @@ -1730,7 +1815,7 @@ def read_content( f"https://api.github.com/repos/{repo_path}/contents/{path}?ref={branch}" ) headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -1748,9 +1833,15 @@ def read_content( except Exception as e: return f"Failed to read content: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: content = self.read_content( - input_data.creds, + credentials, input_data.repo_url, input_data.path, input_data.branch, @@ -1763,6 +1854,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubMakeBranchBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -1775,10 +1867,6 @@ class Input(BlockSchema): description="Name of the source branch", placeholder="source_branch_name", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField(description="Status of the branch creation operation") @@ -1797,7 +1885,7 @@ def __init__(self): "repo_url": "https://github.com/owner/repo", "new_branch": "new_branch_name", "source_branch": "source_branch_name", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1809,13 +1897,16 @@ def __init__(self): @staticmethod def create_branch( - creds: GithubCredentials, repo_url: str, new_branch: str, source_branch: str + credentials: GithubCredentials, + repo_url: str, + new_branch: str, + source_branch: str, ) -> str: try: repo_path = repo_url.replace("https://github.com/", "") ref_api_url = f"https://api.github.com/repos/{repo_path}/git/refs/heads/{source_branch}" headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -1836,9 +1927,15 @@ def create_branch( except Exception as e: return f"Failed to create branch: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.create_branch( - input_data.creds, + credentials, input_data.repo_url, input_data.new_branch, input_data.source_branch, @@ -1851,6 +1948,7 @@ def run(self, input_data: Input) -> BlockOutput: class GithubDeleteBranchBlock(Block): class Input(BlockSchema): + credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -1859,10 +1957,6 @@ class Input(BlockSchema): description="Name of the branch to delete", placeholder="branch_name", ) - creds: GithubCredentials = Field( - description="GitHub OAuth credentials", - default=GithubCredentials(), - ) class Output(BlockSchema): status: str = SchemaField(description="Status of the branch deletion operation") @@ -1880,7 +1974,7 @@ def __init__(self): test_input={ "repo_url": "https://github.com/owner/repo", "branch": "branch_name", - "creds": { + "credentials": { "github_oauth_token": "your-github-oauth-token", }, }, @@ -1891,14 +1985,16 @@ def __init__(self): ) @staticmethod - def delete_branch(creds: GithubCredentials, repo_url: str, branch: str) -> str: + def delete_branch( + credentials: GithubCredentials, repo_url: str, branch: str + ) -> str: try: repo_path = repo_url.replace("https://github.com/", "") api_url = ( f"https://api.github.com/repos/{repo_path}/git/refs/heads/{branch}" ) headers = { - "Authorization": f"Bearer {creds.github_oauth_token.get_secret_value()}", + "Authorization": credentials.bearer(), "Accept": "application/vnd.github.v3+json", } @@ -1911,9 +2007,15 @@ def delete_branch(creds: GithubCredentials, repo_url: str, branch: str) -> str: except Exception as e: return f"Failed to delete branch: {str(e)}" - def run(self, input_data: Input) -> BlockOutput: + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: status = self.delete_branch( - input_data.creds, + credentials, input_data.repo_url, input_data.branch, ) From c8f0110944faddaf301281dbcd52a1147be917a4 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Mon, 16 Sep 2024 17:22:00 +0200 Subject: [PATCH 10/43] Fix schema output with extra properties on complex input fields --- rnd/autogpt_server/autogpt_server/data/block.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/rnd/autogpt_server/autogpt_server/data/block.py b/rnd/autogpt_server/autogpt_server/data/block.py index feed34f89d07..87b896fdadd0 100644 --- a/rnd/autogpt_server/autogpt_server/data/block.py +++ b/rnd/autogpt_server/autogpt_server/data/block.py @@ -52,7 +52,7 @@ def jsonschema(cls) -> dict[str, Any]: if cls.cached_jsonschema: return cls.cached_jsonschema - model = jsonref.replace_refs(cls.model_json_schema()) + model = jsonref.replace_refs(cls.model_json_schema(), merge_props=True) def ref_to_dict(obj): if isinstance(obj, dict): From 57d543af41607e06b21337c392f3a1fcc2dd842f Mon Sep 17 00:00:00 2001 From: Krzysztof Czerwinski <34861343+kcze@users.noreply.github.com> Date: Wed, 18 Sep 2024 22:26:33 +0100 Subject: [PATCH 11/43] feat(server): Server endpoints to add `APIKeyCredentials` and remove credentials (#8094) * Add API Key credentials endpoint * Add endpoint to remove credentials * Use HTTP codes in endpoints * fix HTTP status codes * make `title` mandatory for API keys (because they have no `username` * add client methods --------- Co-authored-by: Reinier van der Leer --- .../src/lib/autogpt-server-api/client.ts | 40 +++++++++++-- .../src/lib/autogpt-server-api/types.ts | 39 ++++++++++++ .../autogpt_server/server/integrations.py | 60 ++++++++++++++++++- 3 files changed, 133 insertions(+), 6 deletions(-) diff --git a/rnd/autogpt_builder/src/lib/autogpt-server-api/client.ts b/rnd/autogpt_builder/src/lib/autogpt-server-api/client.ts index 56d950d9cdc1..694af30fe6c2 100644 --- a/rnd/autogpt_builder/src/lib/autogpt-server-api/client.ts +++ b/rnd/autogpt_builder/src/lib/autogpt-server-api/client.ts @@ -1,12 +1,15 @@ import { createClient } from "../supabase/client"; import { + APIKeyCredentials, Block, + CredentialsMetaResponse, Graph, GraphCreatable, GraphUpdateable, GraphMeta, GraphExecuteResponse, NodeExecutionResult, + OAuth2Credentials, User, } from "./types"; @@ -141,14 +144,42 @@ export default class AutoGPTServerAPI { ).map(parseNodeExecutionResultTimestamps); } + async createAPIKeyCredentials( + credentials: Omit, + ): Promise { + return this._request( + "POST", + `/integrations/${credentials.provider}/credentials`, + credentials, + ); + } + + async listCredentials(provider: string): Promise { + return this._get(`/integrations/${provider}/credentials`); + } + + async getCredentials( + provider: string, + id: string, + ): Promise { + return this._get(`/integrations/${provider}/credentials/${id}`); + } + + async deleteCredentials(provider: string, id: string): Promise { + return this._request( + "DELETE", + `/integrations/${provider}/credentials/${id}`, + ); + } + private async _get(path: string) { return this._request("GET", path); } private async _request( - method: "GET" | "POST" | "PUT" | "PATCH", + method: "GET" | "POST" | "PUT" | "PATCH" | "DELETE", path: string, - payload?: { [key: string]: any }, + payload?: Record, ) { if (method != "GET") { console.debug(`${method} ${path} payload:`, payload); @@ -158,10 +189,11 @@ export default class AutoGPTServerAPI { (await this.supabaseClient?.auth.getSession())?.data.session ?.access_token || ""; + const hasRequestBody = method !== "GET" && payload !== undefined; const response = await fetch(this.baseUrl + path, { method, headers: - method != "GET" + hasRequestBody ? { "Content-Type": "application/json", Authorization: token ? `Bearer ${token}` : "", @@ -169,7 +201,7 @@ export default class AutoGPTServerAPI { : { Authorization: token ? `Bearer ${token}` : "", }, - body: JSON.stringify(payload), + body: hasRequestBody ? JSON.stringify(payload) : undefined, }); const response_data = await response.json(); diff --git a/rnd/autogpt_builder/src/lib/autogpt-server-api/types.ts b/rnd/autogpt_builder/src/lib/autogpt-server-api/types.ts index 876180f1c926..d81c8a6871d1 100644 --- a/rnd/autogpt_builder/src/lib/autogpt-server-api/types.ts +++ b/rnd/autogpt_builder/src/lib/autogpt-server-api/types.ts @@ -79,6 +79,8 @@ export type BlockIOBooleanSubSchema = BlockIOSubSchemaMeta & { default?: boolean; }; +export type CredentialsType = "api_key" | "oauth2"; + export type BlockIONullSubSchema = BlockIOSubSchemaMeta & { type: "null"; }; @@ -179,6 +181,43 @@ export type NodeExecutionResult = { end_time?: Date; }; +/* Mirror of autogpt_server/server/integrations.py:CredentialsMetaResponse */ +export type CredentialsMetaResponse = { + id: string; + type: CredentialsType; + title?: string; + scopes?: Array; + username?: string; +}; + +/* Mirror of autogpt_libs/supabase_integration_credentials_store/types.py:_BaseCredentials */ +type BaseCredentials = { + id: string; + type: CredentialsType; + title?: string; + provider: string; +} + +/* Mirror of autogpt_libs/supabase_integration_credentials_store/types.py:OAuth2Credentials */ +export type OAuth2Credentials = BaseCredentials & { + type: "oauth2"; + scopes: string[]; + username?: string; + access_token: string; + access_token_expires_at?: number; + refresh_token?: string; + refresh_token_expires_at?: number; + metadata: Record; +} + +/* Mirror of autogpt_libs/supabase_integration_credentials_store/types.py:APIKeyCredentials */ +export type APIKeyCredentials = BaseCredentials & { + type: "api_key"; + title: string; + api_key: string; + expires_at?: number; +} + export type User = { id: string; email: string; diff --git a/rnd/autogpt_server/autogpt_server/server/integrations.py b/rnd/autogpt_server/autogpt_server/server/integrations.py index 1c5a717077a5..f05da0e159cb 100644 --- a/rnd/autogpt_server/autogpt_server/server/integrations.py +++ b/rnd/autogpt_server/autogpt_server/server/integrations.py @@ -5,12 +5,22 @@ SupabaseIntegrationCredentialsStore, ) from autogpt_libs.supabase_integration_credentials_store.types import ( + APIKeyCredentials, Credentials, CredentialsType, OAuth2Credentials, ) -from fastapi import APIRouter, Body, Depends, HTTPException, Path, Query, Request -from pydantic import BaseModel +from fastapi import ( + APIRouter, + Body, + Depends, + HTTPException, + Path, + Query, + Request, + Response, +) +from pydantic import BaseModel, SecretStr from supabase import Client from autogpt_server.integrations.oauth import HANDLERS_BY_NAME, BaseOAuthHandler @@ -128,6 +138,52 @@ async def get_credential( return credential +@integrations_api_router.post("/{provider}/credentials", status_code=201) +async def create_api_key_credentials( + provider: Annotated[str, Path(title="The provider to create credentials for")], + api_key: Annotated[str, Body(title="The API key to store")], + title: Annotated[str, Body(title="Optional title for the credentials")], + expires_at: Annotated[ + int | None, Body(title="Unix timestamp when the key expires") + ], + user_id: Annotated[str, Depends(get_user_id)], + store: Annotated[SupabaseIntegrationCredentialsStore, Depends(get_store)], +): + new_credentials = APIKeyCredentials( + provider=provider, + api_key=SecretStr(api_key), + title=title, + expires_at=expires_at, + ) + + try: + store.add_creds(user_id, new_credentials) + except Exception as e: + raise HTTPException( + status_code=500, detail=f"Failed to store credentials: {str(e)}" + ) + return {"id": new_credentials.id} + + +@integrations_api_router.delete("/{provider}/credentials/{cred_id}", status_code=204) +async def delete_credential( + provider: Annotated[str, Path(title="The provider to delete credentials for")], + cred_id: Annotated[str, Path(title="The ID of the credentials to delete")], + user_id: Annotated[str, Depends(get_user_id)], + store: Annotated[SupabaseIntegrationCredentialsStore, Depends(get_store)], +): + creds = store.get_creds_by_id(user_id, cred_id) + if not creds: + raise HTTPException(status_code=404, detail="Credentials not found") + if creds.provider != provider: + raise HTTPException( + status_code=404, detail="Credentials do not match the specified provider" + ) + + store.delete_creds_by_id(user_id, cred_id) + return Response(status_code=204) + + # -------- UTILITIES --------- # From f4a92dcd5fc9154d52506a955d5ad12ccad28f83 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 19 Sep 2024 00:05:25 +0200 Subject: [PATCH 12/43] fix conflict resolve in integrations API router --- .../autogpt_server/server/routers/integrations.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/rnd/autogpt_server/autogpt_server/server/routers/integrations.py b/rnd/autogpt_server/autogpt_server/server/routers/integrations.py index 13c1df1381c5..d1e5a30928af 100644 --- a/rnd/autogpt_server/autogpt_server/server/routers/integrations.py +++ b/rnd/autogpt_server/autogpt_server/server/routers/integrations.py @@ -138,7 +138,7 @@ async def get_credential( return credential -@integrations_api_router.post("/{provider}/credentials", status_code=201) +@router.post("/{provider}/credentials", status_code=201) async def create_api_key_credentials( provider: Annotated[str, Path(title="The provider to create credentials for")], api_key: Annotated[str, Body(title="The API key to store")], @@ -165,7 +165,7 @@ async def create_api_key_credentials( return {"id": new_credentials.id} -@integrations_api_router.delete("/{provider}/credentials/{cred_id}", status_code=204) +@router.delete("/{provider}/credentials/{cred_id}", status_code=204) async def delete_credential( provider: Annotated[str, Path(title="The provider to delete credentials for")], cred_id: Annotated[str, Path(title="The ID of the credentials to delete")], From 1b54bd3c808ccd14028a8f78d508068309906c4f Mon Sep 17 00:00:00 2001 From: Krzysztof Czerwinski <34861343+kcze@users.noreply.github.com> Date: Thu, 19 Sep 2024 02:55:44 +0100 Subject: [PATCH 13/43] feat(builder): Authorization UI on for integration blocks (#8067) ## Front end ### UI components - `CredentialsInput` for use on `CustomNode`: allows user to add/select credentials for a service. - `APIKeyCredentialsModal`: a dialog for creating API keys - `OAuth2FlowWaitingModal`: a dialog to indicate that the application is waiting for the user to log in to the 3rd party service in the provided pop-up window - `NodeCredentialsInput`: wrapper for `CredentialsInput` with the "usual" interface of node input components ### Data model - `CredentialsProvider`: introduces the app-level `CredentialsProvidersContext`, which acts as an application-wide store and cache for credentials metadata. - `useCredentials` for use on `CustomNode`: uses `CredentialsProvidersContext` and provides node-specific credential data and provider-specific data/functions - `/auth/integrations/oauth_callback` route to close the loop to the `CredentialsInput` after a user completes sign-in to the external service ### API client - Add `oAuthLogin`, `oAuthCallback` methods - Add types `BlockIOCredentialsSubSchema`, `CredentialsMetaInput` - Add `isAuthenticated` method ## REST API - Tweak output of OAuth `/login` endpoint: add `state_token` separately in response ## Back end - Add `required_scopes` param to `CredentialsField` (`credentials_scopes` in schema) - Add scope requirement (`repo` in all cases) to all GitHub blocks - Add `FRONTEND_BASE_URL` config option, mainly for local development use. --------- Co-authored-by: Reinier van der Leer --- .../auth/integrations/oauth_callback/route.ts | 38 ++ rnd/autogpt_builder/src/app/providers.tsx | 5 +- .../src/components/CustomNode.tsx | 20 +- .../integrations/credentials-input.tsx | 422 ++++++++++++++++++ .../integrations/credentials-provider.tsx | 164 +++++++ .../src/components/node-input-components.tsx | 36 ++ .../src/components/ui/icons.tsx | 144 ++++++ .../src/hooks/useCredentials.ts | 68 +++ .../src/lib/autogpt-server-api/baseClient.ts | 57 ++- .../src/lib/autogpt-server-api/types.ts | 21 +- .../autogpt_libs/auth/middleware.py | 5 +- .../autogpt_server/blocks/github.py | 64 +-- .../autogpt_server/data/model.py | 4 +- .../server/routers/integrations.py | 10 +- .../autogpt_server/util/service.py | 3 - .../autogpt_server/util/settings.py | 6 + 16 files changed, 1007 insertions(+), 60 deletions(-) create mode 100644 rnd/autogpt_builder/src/app/auth/integrations/oauth_callback/route.ts create mode 100644 rnd/autogpt_builder/src/components/integrations/credentials-input.tsx create mode 100644 rnd/autogpt_builder/src/components/integrations/credentials-provider.tsx create mode 100644 rnd/autogpt_builder/src/hooks/useCredentials.ts diff --git a/rnd/autogpt_builder/src/app/auth/integrations/oauth_callback/route.ts b/rnd/autogpt_builder/src/app/auth/integrations/oauth_callback/route.ts new file mode 100644 index 000000000000..6ec06a46ef28 --- /dev/null +++ b/rnd/autogpt_builder/src/app/auth/integrations/oauth_callback/route.ts @@ -0,0 +1,38 @@ +import { OAuthPopupResultMessage } from "@/components/integrations/credentials-input"; +import { NextResponse } from "next/server"; + +// This route is intended to be used as the callback for integration OAuth flows, +// controlled by the CredentialsInput component. The CredentialsInput opens the login +// page in a pop-up window, which then redirects to this route to close the loop. +export async function GET(request: Request) { + const { searchParams, origin } = new URL(request.url); + const code = searchParams.get("code"); + const state = searchParams.get("state"); + + // Send message from popup window to host window + const message: OAuthPopupResultMessage = + code && state + ? { message_type: "oauth_popup_result", success: true, code, state } + : { + message_type: "oauth_popup_result", + success: false, + message: `Incomplete query: ${searchParams.toString()}`, + }; + + // Return a response with the message as JSON and a script to close the window + return new NextResponse( + ` + + + + + + `, + { + headers: { "Content-Type": "text/html" }, + }, + ); +} diff --git a/rnd/autogpt_builder/src/app/providers.tsx b/rnd/autogpt_builder/src/app/providers.tsx index ce2731227566..93db52de71c1 100644 --- a/rnd/autogpt_builder/src/app/providers.tsx +++ b/rnd/autogpt_builder/src/app/providers.tsx @@ -5,12 +5,15 @@ import { ThemeProvider as NextThemesProvider } from "next-themes"; import { ThemeProviderProps } from "next-themes/dist/types"; import { TooltipProvider } from "@/components/ui/tooltip"; import SupabaseProvider from "@/components/SupabaseProvider"; +import CredentialsProvider from "@/components/integrations/credentials-provider"; export function Providers({ children, ...props }: ThemeProviderProps) { return ( - {children} + + {children} + ); diff --git a/rnd/autogpt_builder/src/components/CustomNode.tsx b/rnd/autogpt_builder/src/components/CustomNode.tsx index 3cd58c7a2801..f3ae0232d88c 100644 --- a/rnd/autogpt_builder/src/components/CustomNode.tsx +++ b/rnd/autogpt_builder/src/components/CustomNode.tsx @@ -254,13 +254,19 @@ export function CustomNode({ data, id, width, height }: NodeProps) { return ( (isRequired || isAdvancedOpen || isConnected || !isAdvanced) && (
{}}> - + {"credentials_provider" in propSchema ? ( + + Credentials + + ) : ( + + )} {!isConnected && ( > = { + github: FaGithub, + google: FaGoogle, + notion: NotionLogoIcon, +}; + +export type OAuthPopupResultMessage = { message_type: "oauth_popup_result" } & ( + | { + success: true; + code: string; + state: string; + } + | { + success: false; + message: string; + } +); + +export const CredentialsInput: FC<{ + className?: string; + selectedCredentials?: CredentialsMetaInput; + onSelectCredentials: (newValue: CredentialsMetaInput) => void; +}> = ({ className, selectedCredentials, onSelectCredentials }) => { + const api = useMemo(() => new AutoGPTServerAPI(), []); + const credentials = useCredentials(); + const [isAPICredentialsModalOpen, setAPICredentialsModalOpen] = + useState(false); + const [isOAuth2FlowInProgress, setOAuth2FlowInProgress] = useState(false); + const [oAuthPopupController, setOAuthPopupController] = + useState(null); + + if (!credentials) { + return null; + } + + if (credentials.isLoading) { + return
Loading...
; + } + + const { + schema, + provider, + providerName, + supportsApiKey, + supportsOAuth2, + savedApiKeys, + savedOAuthCredentials, + oAuthCallback, + } = credentials; + + async function handleOAuthLogin() { + const { login_url, state_token } = await api.oAuthLogin( + provider, + schema.credentials_scopes, + ); + setOAuth2FlowInProgress(true); + const popup = window.open(login_url, "_blank", "popup=true"); + + const controller = new AbortController(); + setOAuthPopupController(controller); + controller.signal.onabort = () => { + setOAuth2FlowInProgress(false); + popup?.close(); + }; + popup?.addEventListener( + "message", + async (e: MessageEvent) => { + if ( + typeof e.data != "object" || + !( + "message_type" in e.data && + e.data.message_type == "oauth_popup_result" + ) + ) + return; + + if (!e.data.success) { + console.error("OAuth flow failed:", e.data.message); + return; + } + + if (e.data.state !== state_token) return; + + const credentials = await oAuthCallback(e.data.code, e.data.state); + onSelectCredentials({ + id: credentials.id, + type: "oauth2", + title: credentials.title, + provider, + }); + controller.abort("success"); + }, + { signal: controller.signal }, + ); + + setTimeout( + () => { + controller.abort("timeout"); + }, + 5 * 60 * 1000, + ); + } + + const ProviderIcon = providerIcons[provider]; + const modals = ( + <> + {supportsApiKey && ( + setAPICredentialsModalOpen(false)} + onCredentialsCreate={(credsMeta) => { + onSelectCredentials(credsMeta); + setAPICredentialsModalOpen(false); + }} + /> + )} + {supportsOAuth2 && ( + oAuthPopupController?.abort("canceled")} + providerName={providerName} + /> + )} + + ); + + // No saved credentials yet + if (savedApiKeys.length === 0 && savedOAuthCredentials.length === 0) { + return ( + <> +
+ {supportsOAuth2 && ( + + )} + {supportsApiKey && ( + + )} +
+ {modals} + + ); + } + + function handleValueChange(newValue: string) { + if (newValue === "sign-in") { + // Trigger OAuth2 sign in flow + handleOAuthLogin(); + } else if (newValue === "add-api-key") { + // Open API key dialog + setAPICredentialsModalOpen(true); + } else { + const selectedCreds = savedApiKeys + .concat(savedOAuthCredentials) + .find((c) => c.id == newValue)!; + + onSelectCredentials({ + id: selectedCreds.id, + type: selectedCreds.type, + provider: schema.credentials_provider, + // title: customTitle, // TODO: add input for title + }); + } + } + + // Saved credentials exist + return ( + <> + + {modals} + + ); +}; + +export const APIKeyCredentialsModal: FC<{ + open: boolean; + onClose: () => void; + onCredentialsCreate: (creds: CredentialsMetaInput) => void; +}> = ({ open, onClose, onCredentialsCreate }) => { + const credentials = useCredentials(); + + const formSchema = z.object({ + apiKey: z.string().min(1, "API Key is required"), + title: z.string().min(1, "Name is required"), + expiresAt: z.string().optional(), + }); + + const form = useForm>({ + resolver: zodResolver(formSchema), + defaultValues: { + apiKey: "", + title: "", + expiresAt: "", + }, + }); + + if (!credentials || credentials.isLoading || !credentials.supportsApiKey) { + return null; + } + + const { schema, provider, providerName, createAPIKeyCredentials } = + credentials; + + async function onSubmit(values: z.infer) { + const expiresAt = values.expiresAt + ? new Date(values.expiresAt).getTime() / 1000 + : undefined; + const newCredentials = await createAPIKeyCredentials({ + api_key: values.apiKey, + title: values.title, + expires_at: expiresAt, + }); + onCredentialsCreate({ + provider, + id: newCredentials.id, + type: "api_key", + title: newCredentials.title, + }); + } + + return ( + { + if (!open) onClose(); + }} + > + + + Add new API key for {providerName} + {schema.description && ( + {schema.description} + )} + + +
+ + ( + + API Key + {schema.credentials_scopes && ( + + Required scope(s) for this block:{" "} + {schema.credentials_scopes?.map((s, i, a) => ( + + {s} + {i < a.length - 1 && ", "} + + ))} + + )} + + + + + + )} + /> + ( + + Name (Optional) + + + + + + )} + /> + ( + + Expiration Date (Optional) + + + + + + )} + /> + + + +
+
+ ); +}; + +export const OAuth2FlowWaitingModal: FC<{ + open: boolean; + onClose: () => void; + providerName: string; +}> = ({ open, onClose, providerName }) => { + return ( + { + if (!open) onClose(); + }} + > + + + + Waiting on {providerName} sign-in process... + + + Complete the sign-in process in the pop-up window. +
+ Closing this dialog will cancel the sign-in process. +
+
+
+
+ ); +}; diff --git a/rnd/autogpt_builder/src/components/integrations/credentials-provider.tsx b/rnd/autogpt_builder/src/components/integrations/credentials-provider.tsx new file mode 100644 index 000000000000..96a37e40cdc5 --- /dev/null +++ b/rnd/autogpt_builder/src/components/integrations/credentials-provider.tsx @@ -0,0 +1,164 @@ +import AutoGPTServerAPI, { + APIKeyCredentials, + CredentialsMetaResponse, +} from "@/lib/autogpt-server-api"; +import { + createContext, + useCallback, + useEffect, + useMemo, + useState, +} from "react"; + +const CREDENTIALS_PROVIDER_NAMES = ["github", "google", "notion"] as const; + +type CredentialsProviderName = (typeof CREDENTIALS_PROVIDER_NAMES)[number]; + +const providerDisplayNames: Record = { + github: "GitHub", + google: "Google", + notion: "Notion", +}; + +type APIKeyCredentialsCreatable = Omit< + APIKeyCredentials, + "id" | "provider" | "type" +>; + +export type CredentialsProviderData = { + provider: string; + providerName: string; + savedApiKeys: CredentialsMetaResponse[]; + savedOAuthCredentials: CredentialsMetaResponse[]; + oAuthCallback: ( + code: string, + state_token: string, + ) => Promise; + createAPIKeyCredentials: ( + credentials: APIKeyCredentialsCreatable, + ) => Promise; +}; + +export type CredentialsProvidersContextType = { + [key in CredentialsProviderName]?: CredentialsProviderData; +}; + +export const CredentialsProvidersContext = + createContext(null); + +export default function CredentialsProvider({ + children, +}: { + children: React.ReactNode; +}) { + const [providers, setProviders] = + useState(null); + const api = useMemo(() => new AutoGPTServerAPI(), []); + + const addCredentials = useCallback( + ( + provider: CredentialsProviderName, + credentials: CredentialsMetaResponse, + ) => { + setProviders((prev) => { + if (!prev || !prev[provider]) return prev; + + const updatedProvider = { ...prev[provider] }; + + if (credentials.type === "api_key") { + updatedProvider.savedApiKeys = [ + ...updatedProvider.savedApiKeys, + credentials, + ]; + } else if (credentials.type === "oauth2") { + updatedProvider.savedOAuthCredentials = [ + ...updatedProvider.savedOAuthCredentials, + credentials, + ]; + } + + return { + ...prev, + [provider]: updatedProvider, + }; + }); + }, + [setProviders], + ); + + /** Wraps `AutoGPTServerAPI.oAuthCallback`, and adds the result to the internal credentials store. */ + const oAuthCallback = useCallback( + async ( + provider: CredentialsProviderName, + code: string, + state_token: string, + ): Promise => { + const credsMeta = await api.oAuthCallback(provider, code, state_token); + addCredentials(provider, credsMeta); + return credsMeta; + }, + [api, addCredentials], + ); + + /** Wraps `AutoGPTServerAPI.createAPIKeyCredentials`, and adds the result to the internal credentials store. */ + const createAPIKeyCredentials = useCallback( + async ( + provider: CredentialsProviderName, + credentials: APIKeyCredentialsCreatable, + ): Promise => { + const credsMeta = await api.createAPIKeyCredentials({ + provider, + ...credentials, + }); + addCredentials(provider, credsMeta); + return credsMeta; + }, + [api, addCredentials], + ); + + useEffect(() => { + api.isAuthenticated().then((isAuthenticated) => { + if (!isAuthenticated) return; + + CREDENTIALS_PROVIDER_NAMES.forEach((provider) => { + api.listCredentials(provider).then((response) => { + const { oauthCreds, apiKeys } = response.reduce<{ + oauthCreds: CredentialsMetaResponse[]; + apiKeys: CredentialsMetaResponse[]; + }>( + (acc, cred) => { + if (cred.type === "oauth2") { + acc.oauthCreds.push(cred); + } else if (cred.type === "api_key") { + acc.apiKeys.push(cred); + } + return acc; + }, + { oauthCreds: [], apiKeys: [] }, + ); + + setProviders((prev) => ({ + ...prev, + [provider]: { + provider, + providerName: providerDisplayNames[provider], + savedApiKeys: apiKeys, + savedOAuthCredentials: oauthCreds, + oAuthCallback: (code: string, state_token: string) => + oAuthCallback(provider, code, state_token), + createAPIKeyCredentials: ( + credentials: APIKeyCredentialsCreatable, + ) => createAPIKeyCredentials(provider, credentials), + }, + })); + }); + }); + }); + }, [api, createAPIKeyCredentials, oAuthCallback]); + + return ( + + {children} + + ); +} diff --git a/rnd/autogpt_builder/src/components/node-input-components.tsx b/rnd/autogpt_builder/src/components/node-input-components.tsx index 9799876756cf..4841e3358a2e 100644 --- a/rnd/autogpt_builder/src/components/node-input-components.tsx +++ b/rnd/autogpt_builder/src/components/node-input-components.tsx @@ -9,6 +9,7 @@ import { BlockIOStringSubSchema, BlockIONumberSubSchema, BlockIOBooleanSubSchema, + BlockIOCredentialsSubSchema, } from "@/lib/autogpt-server-api/types"; import React, { FC, useCallback, useEffect, useState } from "react"; import { Button } from "./ui/button"; @@ -23,6 +24,7 @@ import { import { Input } from "./ui/input"; import NodeHandle from "./NodeHandle"; import { ConnectionData } from "./CustomNode"; +import { CredentialsInput } from "./integrations/credentials-input"; type NodeObjectInputTreeProps = { selfKey?: string; @@ -114,6 +116,18 @@ export const NodeGenericInputField: FC<{ console.warn(`Unsupported 'allOf' in schema for '${propKey}'!`, propSchema); } + if ("credentials_provider" in propSchema) { + return ( + + ); + } + if ("properties" in propSchema) { return ( = ({ selfKey, value, errors, handleInputChange, className }) => { + return ( +
+ + handleInputChange(selfKey, credsMeta) + } + selectedCredentials={value} + /> + {errors[selfKey] && ( + {errors[selfKey]} + )} +
+ ); +}; + const NodeKeyValueInput: FC<{ selfKey: string; schema: BlockIOKVSubSchema; diff --git a/rnd/autogpt_builder/src/components/ui/icons.tsx b/rnd/autogpt_builder/src/components/ui/icons.tsx index 4f9c85d3bd09..b9f51d0eabaf 100644 --- a/rnd/autogpt_builder/src/components/ui/icons.tsx +++ b/rnd/autogpt_builder/src/components/ui/icons.tsx @@ -547,4 +547,148 @@ export const IconMegaphone = createIcon((props) => ( )); +/** + * Key icon component. + * + * @component IconKey + * @param {IconProps} props - The props object containing additional attributes and event handlers for the icon. + * @returns {JSX.Element} - The key icon. + * + * @example + * // Default usage + * + * + * @example + * // With custom color and size + * + * + * @example + * // With custom size and onClick handler + * + */ +export const IconKey = createIcon((props) => ( + + + + +)); + +/** + * Key(+) icon component. + * + * @component IconKeyPlus + * @param {IconProps} props - The props object containing additional attributes and event handlers for the icon. + * @returns {JSX.Element} - The key(+) icon. + * + * @example + * // Default usage + * + * + * @example + * // With custom color and size + * + * + * @example + * // With custom size and onClick handler + * + */ +export const IconKeyPlus = createIcon((props) => ( + + + {/* */} + + + +)); + +/** + * User icon component. + * + * @component IconUser + * @param {IconProps} props - The props object containing additional attributes and event handlers for the icon. + * @returns {JSX.Element} - The user icon. + * + * @example + * // Default usage + * + * + * @example + * // With custom color and size + * + * + * @example + * // With custom size and onClick handler + * + */ +export const IconUser = createIcon((props) => ( + + + + +)); + +/** + * User(+) icon component. + * + * @component IconUserPlus + * @param {IconProps} props - The props object containing additional attributes and event handlers for the icon. + * @returns {JSX.Element} - The user plus icon. + * + * @example + * // Default usage + * + * + * @example + * // With custom color and size + * + * + * @example + * // With custom size and onClick handler + * + */ +export const IconUserPlus = createIcon((props) => ( + + + + + + +)); + export { iconVariants }; diff --git a/rnd/autogpt_builder/src/hooks/useCredentials.ts b/rnd/autogpt_builder/src/hooks/useCredentials.ts new file mode 100644 index 000000000000..2072b83da204 --- /dev/null +++ b/rnd/autogpt_builder/src/hooks/useCredentials.ts @@ -0,0 +1,68 @@ +import { useContext } from "react"; +import { CustomNodeData } from "@/components/CustomNode"; +import { BlockIOCredentialsSubSchema } from "@/lib/autogpt-server-api"; +import { Node, useNodeId, useNodesData } from "@xyflow/react"; +import { + CredentialsProviderData, + CredentialsProvidersContext, +} from "@/components/integrations/credentials-provider"; + +export type CredentialsData = + | { + provider: string; + schema: BlockIOCredentialsSubSchema; + supportsApiKey: boolean; + supportsOAuth2: boolean; + isLoading: true; + } + | (CredentialsProviderData & { + schema: BlockIOCredentialsSubSchema; + supportsApiKey: boolean; + supportsOAuth2: boolean; + isLoading: false; + }); + +export default function useCredentials(): CredentialsData | null { + const nodeId = useNodeId(); + const allProviders = useContext(CredentialsProvidersContext); + + if (!nodeId) { + throw new Error("useCredentials must be within a CustomNode"); + } + + const data = useNodesData>(nodeId)!.data; + const credentialsSchema = data.inputSchema.properties + .credentials as BlockIOCredentialsSubSchema; + + // If block input schema doesn't have credentials, return null + if (!credentialsSchema) { + return null; + } + + const provider = allProviders + ? allProviders[credentialsSchema?.credentials_provider] + : null; + + const supportsApiKey = + credentialsSchema.credentials_types.includes("api_key"); + const supportsOAuth2 = credentialsSchema.credentials_types.includes("oauth2"); + + // No provider means maybe it's still loading + if (!provider) { + return { + provider: credentialsSchema.credentials_provider, + schema: credentialsSchema, + supportsApiKey, + supportsOAuth2, + isLoading: true, + }; + } + + return { + ...provider, + schema: credentialsSchema, + supportsApiKey, + supportsOAuth2, + isLoading: false, + }; +} diff --git a/rnd/autogpt_builder/src/lib/autogpt-server-api/baseClient.ts b/rnd/autogpt_builder/src/lib/autogpt-server-api/baseClient.ts index 878606f9c481..72026f1ee9ab 100644 --- a/rnd/autogpt_builder/src/lib/autogpt-server-api/baseClient.ts +++ b/rnd/autogpt_builder/src/lib/autogpt-server-api/baseClient.ts @@ -35,6 +35,14 @@ export default class BaseAutoGPTServerAPI { this.supabaseClient = supabaseClient; } + async isAuthenticated(): Promise { + if (!this.supabaseClient) return false; + const { + data: { session }, + } = await this.supabaseClient?.auth.getSession(); + return session != null; + } + async createUser(): Promise { return this._request("POST", "/auth/user", {}); } @@ -152,6 +160,25 @@ export default class BaseAutoGPTServerAPI { ).map(parseNodeExecutionResultTimestamps); } + async oAuthLogin( + provider: string, + scopes?: string[], + ): Promise<{ login_url: string; state_token: string }> { + const query = scopes ? { scopes: scopes.join(",") } : undefined; + return await this._get(`/integrations/${provider}/login`, query); + } + + async oAuthCallback( + provider: string, + code: string, + state_token: string, + ): Promise { + return this._request("POST", `/integrations/${provider}/callback`, { + code, + state_token, + }); + } + async createAPIKeyCredentials( credentials: Omit, ): Promise { @@ -188,8 +215,8 @@ export default class BaseAutoGPTServerAPI { return this._request("POST", "/analytics/log_raw_analytics", analytic); } - private async _get(path: string) { - return this._request("GET", path); + private async _get(path: string, query?: Record) { + return this._request("GET", path, query); } private async _request( @@ -205,18 +232,24 @@ export default class BaseAutoGPTServerAPI { (await this.supabaseClient?.auth.getSession())?.data.session ?.access_token || ""; + let url = this.baseUrl + path; + if (method === "GET" && payload) { + // For GET requests, use payload as query + const queryParams = new URLSearchParams(payload); + url += `?${queryParams.toString()}`; + } + const hasRequestBody = method !== "GET" && payload !== undefined; - const response = await fetch(this.baseUrl + path, { + const response = await fetch(url, { method, - headers: - hasRequestBody - ? { - "Content-Type": "application/json", - Authorization: token ? `Bearer ${token}` : "", - } - : { - Authorization: token ? `Bearer ${token}` : "", - }, + headers: hasRequestBody + ? { + "Content-Type": "application/json", + Authorization: token ? `Bearer ${token}` : "", + } + : { + Authorization: token ? `Bearer ${token}` : "", + }, body: hasRequestBody ? JSON.stringify(payload) : undefined, }); const response_data = await response.json(); diff --git a/rnd/autogpt_builder/src/lib/autogpt-server-api/types.ts b/rnd/autogpt_builder/src/lib/autogpt-server-api/types.ts index f9c568718c75..f5db3d8f81fa 100644 --- a/rnd/autogpt_builder/src/lib/autogpt-server-api/types.ts +++ b/rnd/autogpt_builder/src/lib/autogpt-server-api/types.ts @@ -42,6 +42,7 @@ export type BlockIOSubSchema = type BlockIOSimpleTypeSubSchema = | BlockIOObjectSubSchema + | BlockIOCredentialsSubSchema | BlockIOKVSubSchema | BlockIOArraySubSchema | BlockIOStringSubSchema @@ -94,6 +95,12 @@ export type BlockIOBooleanSubSchema = BlockIOSubSchemaMeta & { export type CredentialsType = "api_key" | "oauth2"; +export type BlockIOCredentialsSubSchema = BlockIOSubSchemaMeta & { + credentials_provider: "github" | "google" | "notion"; + credentials_scopes?: string[]; + credentials_types: Array; +}; + export type BlockIONullSubSchema = BlockIOSubSchemaMeta & { type: "null"; }; @@ -203,13 +210,21 @@ export type CredentialsMetaResponse = { username?: string; }; +/* Mirror of autogpt_server/data/model.py:CredentialsMetaInput */ +export type CredentialsMetaInput = { + id: string; + type: CredentialsType; + title?: string; + provider: string; +}; + /* Mirror of autogpt_libs/supabase_integration_credentials_store/types.py:_BaseCredentials */ type BaseCredentials = { id: string; type: CredentialsType; title?: string; provider: string; -} +}; /* Mirror of autogpt_libs/supabase_integration_credentials_store/types.py:OAuth2Credentials */ export type OAuth2Credentials = BaseCredentials & { @@ -221,7 +236,7 @@ export type OAuth2Credentials = BaseCredentials & { refresh_token?: string; refresh_token_expires_at?: number; metadata: Record; -} +}; /* Mirror of autogpt_libs/supabase_integration_credentials_store/types.py:APIKeyCredentials */ export type APIKeyCredentials = BaseCredentials & { @@ -229,7 +244,7 @@ export type APIKeyCredentials = BaseCredentials & { title: string; api_key: string; expires_at?: number; -} +}; export type User = { id: string; diff --git a/rnd/autogpt_libs/autogpt_libs/auth/middleware.py b/rnd/autogpt_libs/autogpt_libs/auth/middleware.py index 71a2268fa03d..783e1b35beab 100644 --- a/rnd/autogpt_libs/autogpt_libs/auth/middleware.py +++ b/rnd/autogpt_libs/autogpt_libs/auth/middleware.py @@ -7,12 +7,13 @@ from .jwt_utils import parse_jwt_token security = HTTPBearer() +logger = logging.getLogger(__name__) async def auth_middleware(request: Request): if not settings.ENABLE_AUTH: # If authentication is disabled, allow the request to proceed - logging.warn("Auth disabled") + logger.warn("Auth disabled") return {} security = HTTPBearer() @@ -24,7 +25,7 @@ async def auth_middleware(request: Request): try: payload = parse_jwt_token(credentials.credentials) request.state.user = payload - logging.info("Token decoded successfully") + logger.debug("Token decoded successfully") except ValueError as e: raise HTTPException(status_code=401, detail=str(e)) return payload diff --git a/rnd/autogpt_server/autogpt_server/blocks/github.py b/rnd/autogpt_server/autogpt_server/blocks/github.py index 4800e578b14b..6a0ca8d185e7 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/github.py +++ b/rnd/autogpt_server/autogpt_server/blocks/github.py @@ -19,16 +19,26 @@ Literal["github"], Literal["api_key", "oauth2"] ] -GITHUB_CREDS_FIELD: GithubCredentialsInput = CredentialsField( - provider="github", - supported_credential_types={"api_key", "oauth2"}, - description="GitHub OAuth credentials", -) + +def GithubCredentialsField(scope: str) -> GithubCredentialsInput: + """ + Creates a GitHub credentials input on a block. + + Params: + scope: The authorization scope needed for the block to work. ([list of available scopes](https://docs.github.com/en/apps/oauth-apps/building-oauth-apps/scopes-for-oauth-apps#available-scopes)) + """ # noqa + return CredentialsField( + provider="github", + supported_credential_types={"api_key", "oauth2"}, + required_scopes={scope}, + description="The GitHub integration can be used with OAuth, " + "or any API key with sufficient permissions for the blocks it is used on.", + ) class GithubCommentBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") issue_url: str = SchemaField( description="URL of the GitHub issue or pull request", placeholder="https://github.com/owner/repo/issues/1", @@ -115,7 +125,7 @@ def run( class GithubMakeIssueBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -194,7 +204,7 @@ def run( class GithubMakePRBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -301,7 +311,7 @@ def run( class GithubReadIssueBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") issue_url: str = SchemaField( description="URL of the GitHub issue", placeholder="https://github.com/owner/repo/issues/1", @@ -387,7 +397,7 @@ def run( class GithubReadPRBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") pr_url: str = SchemaField( description="URL of the GitHub pull request", placeholder="https://github.com/owner/repo/pull/1", @@ -523,7 +533,7 @@ def run( class GithubListIssuesBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -611,7 +621,7 @@ def run( class GithubReadTagsBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -704,7 +714,7 @@ def run( class GithubReadBranchesBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -795,7 +805,7 @@ def run( class GithubReadDiscussionsBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -907,7 +917,7 @@ def run( class GithubReadReleasesBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -997,7 +1007,7 @@ def run( class GithubAddLabelBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") issue_url: str = SchemaField( description="URL of the GitHub issue or pull request", placeholder="https://github.com/owner/repo/issues/1", @@ -1085,7 +1095,7 @@ def run( class GithubRemoveLabelBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") issue_url: str = SchemaField( description="URL of the GitHub issue or pull request", placeholder="https://github.com/owner/repo/issues/1", @@ -1175,7 +1185,7 @@ def run( class GithubAssignReviewerBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") pr_url: str = SchemaField( description="URL of the GitHub pull request", placeholder="https://github.com/owner/repo/pull/1", @@ -1270,7 +1280,7 @@ def run( class GithubUnassignReviewerBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") pr_url: str = SchemaField( description="URL of the GitHub pull request", placeholder="https://github.com/owner/repo/pull/1", @@ -1352,7 +1362,7 @@ def run( class GithubListReviewersBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") pr_url: str = SchemaField( description="URL of the GitHub pull request", placeholder="https://github.com/owner/repo/pull/1", @@ -1448,7 +1458,7 @@ def run( class GithubAssignIssueBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") issue_url: str = SchemaField( description="URL of the GitHub issue", placeholder="https://github.com/owner/repo/issues/1", @@ -1534,7 +1544,7 @@ def run( class GithubUnassignIssueBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") issue_url: str = SchemaField( description="URL of the GitHub issue", placeholder="https://github.com/owner/repo/issues/1", @@ -1620,7 +1630,7 @@ def run( class GithubReadCodeownersFileBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -1690,7 +1700,7 @@ def run( class GithubReadFileFromMasterBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -1764,7 +1774,7 @@ def run( class GithubReadFileFolderRepoBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -1854,7 +1864,7 @@ def run( class GithubMakeBranchBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", @@ -1948,7 +1958,7 @@ def run( class GithubDeleteBranchBlock(Block): class Input(BlockSchema): - credentials: GithubCredentialsInput = GITHUB_CREDS_FIELD + credentials: GithubCredentialsInput = GithubCredentialsField("repo") repo_url: str = SchemaField( description="URL of the GitHub repository", placeholder="https://github.com/owner/repo", diff --git a/rnd/autogpt_server/autogpt_server/data/model.py b/rnd/autogpt_server/autogpt_server/data/model.py index 08c9622d5672..06875ead4f81 100644 --- a/rnd/autogpt_server/autogpt_server/data/model.py +++ b/rnd/autogpt_server/autogpt_server/data/model.py @@ -146,7 +146,7 @@ def SchemaField( class CredentialsMetaInput(BaseModel, Generic[CP, CT]): id: str - title: str + title: Optional[str] = None provider: CP type: CT @@ -154,6 +154,7 @@ class CredentialsMetaInput(BaseModel, Generic[CP, CT]): def CredentialsField( provider: CP, supported_credential_types: set[CT], + required_scopes: set[str] = set(), *, title: Optional[str] = None, description: Optional[str] = None, @@ -167,6 +168,7 @@ def CredentialsField( k: v for k, v in { "credentials_provider": provider, + "credentials_scopes": list(required_scopes) or None, # omit if empty "credentials_types": list(supported_credential_types), }.items() if v is not None diff --git a/rnd/autogpt_server/autogpt_server/server/routers/integrations.py b/rnd/autogpt_server/autogpt_server/server/routers/integrations.py index d1e5a30928af..63593b5ecb09 100644 --- a/rnd/autogpt_server/autogpt_server/server/routers/integrations.py +++ b/rnd/autogpt_server/autogpt_server/server/routers/integrations.py @@ -39,6 +39,7 @@ def get_store(supabase: Client = Depends(get_supabase)): class LoginResponse(BaseModel): login_url: str + state_token: str @router.get("/{provider}/login") @@ -54,12 +55,12 @@ async def login( handler = _get_provider_oauth_handler(request, provider) # Generate and store a secure random state token - state = await store.store_state_token(user_id, provider) + state_token = await store.store_state_token(user_id, provider) requested_scopes = scopes.split(",") if scopes else [] - login_url = handler.get_login_url(requested_scopes, state) + login_url = handler.get_login_url(requested_scopes, state_token) - return LoginResponse(login_url=login_url) + return LoginResponse(login_url=login_url, state_token=state_token) class CredentialsMetaResponse(BaseModel): @@ -202,8 +203,9 @@ def _get_provider_oauth_handler(req: Request, provider_name: str) -> BaseOAuthHa ) handler_class = HANDLERS_BY_NAME[provider_name] + frontend_base_url = settings.config.frontend_base_url or str(req.base_url) return handler_class( client_id=client_id, client_secret=client_secret, - redirect_uri=str(req.url_for("callback", provider=provider_name)), + redirect_uri=f"{frontend_base_url}/auth/integrations/oauth_callback", ) diff --git a/rnd/autogpt_server/autogpt_server/util/service.py b/rnd/autogpt_server/autogpt_server/util/service.py index db63d14c02d2..aa9b9f21abcc 100644 --- a/rnd/autogpt_server/autogpt_server/util/service.py +++ b/rnd/autogpt_server/autogpt_server/util/service.py @@ -103,9 +103,6 @@ def cleanup(self): if self.use_redis: logger.info(f"[{self.__class__.__name__}] ⏳ Disconnecting Redis...") self.run_and_wait(self.event_queue.close()) - if self.use_supabase: - logger.info(f"[{self.__class__.__name__}] ⏳ Disconnecting Supabase...") - self.supabase.realtime.remove_all_channels() @conn_retry def __start_pyro(self): diff --git a/rnd/autogpt_server/autogpt_server/util/settings.py b/rnd/autogpt_server/autogpt_server/util/settings.py index 15f7e6cc43c3..bbcc51045e93 100644 --- a/rnd/autogpt_server/autogpt_server/util/settings.py +++ b/rnd/autogpt_server/autogpt_server/util/settings.py @@ -105,6 +105,12 @@ class Config(UpdateTrackingModel["Config"], BaseSettings): description="The port for agent server API to run on", ) + frontend_base_url: str = Field( + default="", + description="Can be used to explicitly set the base URL for the frontend. " + "This value is then used to generate redirect URLs for OAuth flows.", + ) + @classmethod def settings_customise_sources( cls, From d9b2ccc660d9b2923cc1995a887b788b2cb29cfb Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 19 Sep 2024 04:28:24 +0200 Subject: [PATCH 14/43] fix field description on API key dialog --- .../src/components/integrations/credentials-input.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/rnd/autogpt_builder/src/components/integrations/credentials-input.tsx b/rnd/autogpt_builder/src/components/integrations/credentials-input.tsx index 0eeedb3d5db0..c4376314d87b 100644 --- a/rnd/autogpt_builder/src/components/integrations/credentials-input.tsx +++ b/rnd/autogpt_builder/src/components/integrations/credentials-input.tsx @@ -354,7 +354,7 @@ export const APIKeyCredentialsModal: FC<{ name="title" render={({ field }) => ( - Name (Optional) + Name Date: Thu, 19 Sep 2024 05:02:41 +0200 Subject: [PATCH 15/43] parametrize `test_available_blocks` --- rnd/autogpt_server/test/block/test_block.py | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/rnd/autogpt_server/test/block/test_block.py b/rnd/autogpt_server/test/block/test_block.py index e4a8e4d4a381..4c7d68317db4 100644 --- a/rnd/autogpt_server/test/block/test_block.py +++ b/rnd/autogpt_server/test/block/test_block.py @@ -1,7 +1,9 @@ -from autogpt_server.data.block import get_blocks +import pytest + +from autogpt_server.data.block import Block, get_blocks from autogpt_server.util.test import execute_block_test -def test_available_blocks(): - for block in get_blocks().values(): - execute_block_test(type(block)()) +@pytest.mark.parametrize("block", get_blocks().values(), ids=lambda b: b.name) +def test_available_blocks(block: Block): + execute_block_test(type(block)()) From a5264d6bef3cc43b066f7c3a9dee842d08d4397b Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 19 Sep 2024 16:02:47 +0200 Subject: [PATCH 16/43] Update .env.example and docker-compose.yml with new config variables --- rnd/autogpt_server/.env.example | 14 ++++++++++++-- rnd/autogpt_server/autogpt_server/server/utils.py | 4 +++- rnd/autogpt_server/autogpt_server/util/service.py | 4 +++- rnd/autogpt_server/autogpt_server/util/settings.py | 4 +++- rnd/docker-compose.yml | 8 +++----- 5 files changed, 24 insertions(+), 10 deletions(-) diff --git a/rnd/autogpt_server/.env.example b/rnd/autogpt_server/.env.example index ab536498a700..9e2e5b15cc6b 100644 --- a/rnd/autogpt_server/.env.example +++ b/rnd/autogpt_server/.env.example @@ -9,14 +9,24 @@ REDIS_HOST=localhost REDIS_PORT=6379 REDIS_PASSWORD=password -ENABLE_AUTH=false ENABLE_CREDIT=false APP_ENV="local" PYRO_HOST=localhost SENTRY_DSN= -# This is needed when ENABLE_AUTH is true + +## User auth with Supabase is required for any of the 3rd party integrations with auth to work. +ENABLE_AUTH=false +SUPABASE_URL= +SUPABASE_SERVICE_KEY= SUPABASE_JWT_SECRET= +## == INTEGRATION CREDENTIALS == ## +# Each set of server side credentials is required for the corresponding 3rd party +# integration to work. + +GITHUB_CLIENT_ID= +GITHUB_CLIENT_SECRET= + ## ===== OPTIONAL API KEYS ===== ## # LLM diff --git a/rnd/autogpt_server/autogpt_server/server/utils.py b/rnd/autogpt_server/autogpt_server/server/utils.py index 07ccab4a6fb8..07a282a40dfb 100644 --- a/rnd/autogpt_server/autogpt_server/server/utils.py +++ b/rnd/autogpt_server/autogpt_server/server/utils.py @@ -20,4 +20,6 @@ def get_user_id(payload: dict = Depends(auth_middleware)) -> str: def get_supabase() -> Client: - return create_client(settings.secrets.supabase_url, settings.secrets.supabase_key) + return create_client( + settings.secrets.supabase_url, settings.secrets.supabase_service_key + ) diff --git a/rnd/autogpt_server/autogpt_server/util/service.py b/rnd/autogpt_server/autogpt_server/util/service.py index aa9b9f21abcc..39b968c848f6 100644 --- a/rnd/autogpt_server/autogpt_server/util/service.py +++ b/rnd/autogpt_server/autogpt_server/util/service.py @@ -81,7 +81,9 @@ def run(self): from supabase import create_client secrets = Secrets() - self.supabase = create_client(secrets.supabase_url, secrets.supabase_key) + self.supabase = create_client( + secrets.supabase_url, secrets.supabase_service_key + ) # Initialize the async loop. async_thread = threading.Thread(target=self.__start_async_loop) diff --git a/rnd/autogpt_server/autogpt_server/util/settings.py b/rnd/autogpt_server/autogpt_server/util/settings.py index bbcc51045e93..19e37cfa9ab3 100644 --- a/rnd/autogpt_server/autogpt_server/util/settings.py +++ b/rnd/autogpt_server/autogpt_server/util/settings.py @@ -133,7 +133,9 @@ class Secrets(UpdateTrackingModel["Secrets"], BaseSettings): """Secrets for the server.""" supabase_url: str = Field(default="", description="Supabase URL") - supabase_key: str = Field(default="", description="Supabase key") + supabase_service_key: str = Field( + default="", description="Supabase service role key" + ) # OAuth server credentials for integrations github_client_id: str = Field(default="", description="GitHub OAuth client ID") diff --git a/rnd/docker-compose.yml b/rnd/docker-compose.yml index 0e98131ab9db..0a397e55c4e9 100644 --- a/rnd/docker-compose.yml +++ b/rnd/docker-compose.yml @@ -74,7 +74,7 @@ services: environment: - SUPABASE_URL=http://kong:8000 - SUPABASE_JWT_SECRET=your-super-secret-jwt-token-with-at-least-32-characters-long - - SUPABASE_ANON_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyAgCiAgICAicm9sZSI6ICJhbm9uIiwKICAgICJpc3MiOiAic3VwYWJhc2UtZGVtbyIsCiAgICAiaWF0IjogMTY0MTc2OTIwMCwKICAgICJleHAiOiAxNzk5NTM1NjAwCn0.dc_X5iR_VP_qT0zsiyj_I_OZ2T9FtRU2BBNWN8Bu4GE + - SUPABASE_SERVICE_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyAgCiAgICAicm9sZSI6ICJzZXJ2aWNlX3JvbGUiLAogICAgImlzcyI6ICJzdXBhYmFzZS1kZW1vIiwKICAgICJpYXQiOiAxNjQxNzY5MjAwLAogICAgImV4cCI6IDE3OTk1MzU2MDAKfQ.DaYlNEoUrrEn2Ig7tqibS-PHK5vgusbcbo7X36XVt4Q - DATABASE_URL=postgresql://agpt_user:pass123@postgres:5432/agpt_local?connect_timeout=60 - REDIS_HOST=redis - REDIS_PORT=6379 @@ -107,9 +107,9 @@ services: migrate: condition: service_completed_successfully environment: - - NEXT_PUBLIC_SUPABASE_URL=http://kong:8000 + - SUPABASE_URL=http://kong:8000 - SUPABASE_JWT_SECRET=your-super-secret-jwt-token-with-at-least-32-characters-long - - SUPABASE_ANON_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyAgCiAgICAicm9sZSI6ICJhbm9uIiwKICAgICJpc3MiOiAic3VwYWJhc2UtZGVtbyIsCiAgICAiaWF0IjogMTY0MTc2OTIwMCwKICAgICJleHAiOiAxNzk5NTM1NjAwCn0.dc_X5iR_VP_qT0zsiyj_I_OZ2T9FtRU2BBNWN8Bu4GE + - SUPABASE_SERVICE_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyAgCiAgICAicm9sZSI6ICJzZXJ2aWNlX3JvbGUiLAogICAgImlzcyI6ICJzdXBhYmFzZS1kZW1vIiwKICAgICJpYXQiOiAxNjQxNzY5MjAwLAogICAgImV4cCI6IDE3OTk1MzU2MDAKfQ.DaYlNEoUrrEn2Ig7tqibS-PHK5vgusbcbo7X36XVt4Q - DATABASE_URL=postgresql://agpt_user:pass123@postgres:5432/agpt_local?connect_timeout=60 - REDIS_HOST=redis - REDIS_PORT=6379 @@ -141,9 +141,7 @@ services: migrate: condition: service_completed_successfully environment: - - SUPABASE_URL=http://kong:8000 - SUPABASE_JWT_SECRET=your-super-secret-jwt-token-with-at-least-32-characters-long - - SUPABASE_ANON_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyAgCiAgICAicm9sZSI6ICJhbm9uIiwKICAgICJpc3MiOiAic3VwYWJhc2UtZGVtbyIsCiAgICAiaWF0IjogMTY0MTc2OTIwMCwKICAgICJleHAiOiAxNzk5NTM1NjAwCn0.dc_X5iR_VP_qT0zsiyj_I_OZ2T9FtRU2BBNWN8Bu4GE - DATABASE_URL=postgresql://agpt_user:pass123@postgres:5432/agpt_local?connect_timeout=60 - REDIS_HOST=redis - REDIS_PORT=6379 From 74138d7b2d2ae97641e08b177f5e10625d4dbab2 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 19 Sep 2024 17:25:09 +0200 Subject: [PATCH 17/43] add docs for adding authenticated blocks --- docs/content/server/new_blocks.md | 138 ++++++++++++++++++++++++++++++ 1 file changed, 138 insertions(+) diff --git a/docs/content/server/new_blocks.md b/docs/content/server/new_blocks.md index 41341d39635f..69e138b33534 100644 --- a/docs/content/server/new_blocks.md +++ b/docs/content/server/new_blocks.md @@ -105,6 +105,144 @@ Follow these steps to create and test a new block: - **Error handling**: Handle various exceptions that might occur during the API request and data processing. - **Yield**: Use `yield` to output the results. +### Blocks with authentication + +Our system supports auth offloading for API keys and OAuth2 authorization flows. +Adding a block with API key authentication is straight-forward, as is adding a block +for a service that we already have OAuth2 support for. + +Implementing the block itself is relatively simple. On top of the instructions above, +you're going to add a `credentials` parameter to the `Input` model and the `run` method: +```python +from autogpt_libs.supabase_integration_credentials_store.types import ( + APIKeyCredentials, + OAuth2Credentials, + Credentials, +) + +from autogpt_server.data.block import Block, BlockOutput, BlockSchema +from autogpt_server.data.model import CredentialsField + + +class BlockWithAPIKeyAuth(Block): + class Input(BlockSchema): + credentials = CredentialsField( + provider="github", + supported_credential_types={"api_key"}, + required_scopes={"repo"}, + description="The GitHub integration can be used with " + "any API key with sufficient permissions for the blocks it is used on.", + ) + + # ... + + def run( + self, + input_data: Input, + *, + credentials: APIKeyCredentials, + **kwargs, + ) -> BlockOutput: + ... + + +class BlockWithOAuth(Block): + class Input(BlockSchema): + credentials = CredentialsField( + provider="github", + supported_credential_types={"oauth2"}, + required_scopes={"repo"}, + description="The GitHub integration can be used with OAuth.", + ) + + # ... + + def run( + self, + input_data: Input, + *, + credentials: OAuth2Credentials, + **kwargs, + ) -> BlockOutput: + ... + + +class BlockWithAPIKeyAndOAuth(Block): + class Input(BlockSchema): + credentials = CredentialsField( + provider="github", + supported_credential_types={"api_key", "oauth2"}, + required_scopes={"repo"}, + description="The GitHub integration can be used with OAuth, " + "or any API key with sufficient permissions for the blocks it is used on.", + ) + + # ... + + def run( + self, + input_data: Input, + *, + credentials: Credentials, + **kwargs, + ) -> BlockOutput: + ... +``` +The credentials will be automagically injected by the executor in the back end. + +The `APIKeyCredentials` and `OAuth2Credentials` models are defined [here](https://github.com/Significant-Gravitas/AutoGPT/blob/master/rnd/autogpt_libs/autogpt_libs/supabase_integration_credentials_store/types.py). +To use them in e.g. an API request, you can either access the token directly: +```python +# credentials: APIKeyCredentials +response = requests.post( + url, + headers={ + "Authorization": f"Bearer {credentials.api_key.get_secret_value()})", + }, +) + +# credentials: OAuth2Credentials +response = requests.post( + url, + headers={ + "Authorization": f"Bearer {credentials.access_token.get_secret_value()})", + }, +) +``` +or use the shortcut `credentials.bearer()`: +```python +# credentials: APIKeyCredentials | OAuth2Credentials +response = requests.post( + url, + headers={"Authorization": credentials.bearer()}, +) +``` + +#### Adding an OAuth2 service integration + +To add support for a new OAuth2-authenticated service, you'll need to add an `OAuthHandler`. +All our existing handlers and the base class can be found [here][OAuth2 handlers]. + +Every handler must implement the following parts of the [`BaseOAuthHandler`] interface: +- `PROVIDER_NAME` +- `__init__(client_id, client_secret, redirect_uri)` +- `get_login_url(scopes, state)` +- `exchange_code_for_tokens(code)` +- `_refresh_tokens(credentials)` + +As you can see, this is modeled after the standard OAuth2 flow. + +Aside from implementing the `OAuthHandler` itself, adding a handler into the system requires two more things: +- Adding the handler class to `HANDLERS_BY_NAME` [here](https://github.com/Significant-Gravitas/AutoGPT/blob/master/rnd/autogpt_server/autogpt_server/integrations/oauth/__init__.py) +- Adding `{provider}_client_id` and `{provider}_client_secret` to the application's `Secrets` [here](https://github.com/Significant-Gravitas/AutoGPT/blob/e3f35d79c7e9fc6ee0cabefcb73e0fad15a0ce2d/rnd/autogpt_server/autogpt_server/util/settings.py#L132) + +[OAuth2 handlers]: https://github.com/Significant-Gravitas/AutoGPT/tree/master/rnd/autogpt_server/autogpt_server/integrations/oauth +[`BaseOAuthHandler`]: https://github.com/Significant-Gravitas/AutoGPT/blob/master/rnd/autogpt_server/autogpt_server/integrations/oauth/base.py + +#### Example: GitHub integration +- GitHub blocks with API key + OAuth2 support: [`blocks/github.py`](https://github.com/Significant-Gravitas/AutoGPT/blob/master/rnd/autogpt_server/autogpt_server/blocks/github.py) +- GitHub OAuth2 handler: [`integrations/oauth/github.py`](https://github.com/Significant-Gravitas/AutoGPT/blob/master/rnd/autogpt_server/autogpt_server/integrations/oauth/github.py) + ## Key Points to Remember - **Unique ID**: Give your block a unique ID in the **init** method. From 57afc85ad904f1fdef95292d05af41d46b9e14d0 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 19 Sep 2024 18:27:45 +0200 Subject: [PATCH 18/43] add comment to .env.example about GitHub app type --- rnd/autogpt_server/.env.example | 1 + 1 file changed, 1 insertion(+) diff --git a/rnd/autogpt_server/.env.example b/rnd/autogpt_server/.env.example index 9e2e5b15cc6b..b3768ceaea19 100644 --- a/rnd/autogpt_server/.env.example +++ b/rnd/autogpt_server/.env.example @@ -24,6 +24,7 @@ SUPABASE_JWT_SECRET= # Each set of server side credentials is required for the corresponding 3rd party # integration to work. +# GitHub OAuth App server credentials - https://github.com/settings/developers GITHUB_CLIENT_ID= GITHUB_CLIENT_SECRET= From 6aea79b52d4b7d3d98fd043fc2082ea2e45846ca Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Thu, 19 Sep 2024 19:17:37 +0200 Subject: [PATCH 19/43] fix block tests with credentials --- docs/content/server/new_blocks.md | 8 +- .../autogpt_server/blocks/github.py | 126 +++++++++--------- .../autogpt_server/data/block.py | 15 ++- .../autogpt_server/util/test.py | 12 +- 4 files changed, 90 insertions(+), 71 deletions(-) diff --git a/docs/content/server/new_blocks.md b/docs/content/server/new_blocks.md index 69e138b33534..812afdc48a9a 100644 --- a/docs/content/server/new_blocks.md +++ b/docs/content/server/new_blocks.md @@ -124,6 +124,7 @@ from autogpt_server.data.block import Block, BlockOutput, BlockSchema from autogpt_server.data.model import CredentialsField +# API Key auth: class BlockWithAPIKeyAuth(Block): class Input(BlockSchema): credentials = CredentialsField( @@ -145,7 +146,7 @@ class BlockWithAPIKeyAuth(Block): ) -> BlockOutput: ... - +# OAuth: class BlockWithOAuth(Block): class Input(BlockSchema): credentials = CredentialsField( @@ -166,7 +167,7 @@ class BlockWithOAuth(Block): ) -> BlockOutput: ... - +# API Key auth + OAuth: class BlockWithAPIKeyAndOAuth(Block): class Input(BlockSchema): credentials = CredentialsField( @@ -255,7 +256,8 @@ Aside from implementing the `OAuthHandler` itself, adding a handler into the sys The testing of blocks is handled by `test_block.py`, which does the following: -1. It calls the block with the provided `test_input`. +1. It calls the block with the provided `test_input`. + If the block has a `credentials` field, `test_credentials` is passed in as well. 2. If a `test_mock` is provided, it temporarily replaces the specified methods with the mock functions. 3. It then asserts that the output matches the `test_output`. diff --git a/rnd/autogpt_server/autogpt_server/blocks/github.py b/rnd/autogpt_server/autogpt_server/blocks/github.py index 6a0ca8d185e7..f1118e9225e1 100644 --- a/rnd/autogpt_server/autogpt_server/blocks/github.py +++ b/rnd/autogpt_server/autogpt_server/blocks/github.py @@ -6,6 +6,7 @@ APIKeyCredentials, OAuth2Credentials, ) +from pydantic import SecretStr from autogpt_server.data.block import Block, BlockCategory, BlockOutput, BlockSchema from autogpt_server.data.model import ( @@ -36,6 +37,21 @@ def GithubCredentialsField(scope: str) -> GithubCredentialsInput: ) +TEST_CREDENTIALS = APIKeyCredentials( + id="01234567-89ab-cdef-0123-456789abcdef", + provider="github", + api_key=SecretStr("mock-github-api-key"), + title="Mock GitHub API key", + expires_at=None, +) +TEST_CREDENTIALS_INPUT = { + "provider": TEST_CREDENTIALS.provider, + "id": TEST_CREDENTIALS.id, + "type": TEST_CREDENTIALS.type, + "title": TEST_CREDENTIALS.type, +} + + class GithubCommentBlock(Block): class Input(BlockSchema): credentials: GithubCredentialsInput = GithubCredentialsField("repo") @@ -64,10 +80,9 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "comment": "This is a test comment.", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Comment posted successfully")], test_mock={ "post_comment": lambda *args, **kwargs: "Comment posted successfully" @@ -154,10 +169,9 @@ def __init__(self): "repo_url": "https://github.com/owner/repo", "title": "Test Issue", "body": "This is a test issue.", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Issue created successfully")], test_mock={ "create_issue": lambda *args, **kwargs: "Issue created successfully" @@ -247,10 +261,9 @@ def __init__(self): "body": "This is a test pull request.", "head": "feature-branch", "base": "main", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Pull request created successfully")], test_mock={ "create_pr": lambda *args, **kwargs: "Pull request created successfully" @@ -334,10 +347,9 @@ def __init__(self): output_schema=GithubReadIssueBlock.Output, test_input={ "issue_url": "https://github.com/owner/repo/issues/1", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[ ("title", "Title of the issue"), ("body", "This is the body of the issue."), @@ -425,11 +437,10 @@ def __init__(self): output_schema=GithubReadPRBlock.Output, test_input={ "pr_url": "https://github.com/owner/repo/pull/1", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, "include_pr_changes": True, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[ ("title", "Title of the pull request"), ("body", "This is the body of the pull request."), @@ -554,10 +565,9 @@ def __init__(self): output_schema=GithubListIssuesBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[ ( "issues", @@ -642,10 +652,9 @@ def __init__(self): output_schema=GithubReadTagsBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[ ( "tags", @@ -735,10 +744,9 @@ def __init__(self): output_schema=GithubReadBranchesBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[ ( "branches", @@ -831,11 +839,10 @@ def __init__(self): output_schema=GithubReadDiscussionsBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, "num_discussions": 3, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[ ( "discussions", @@ -938,10 +945,9 @@ def __init__(self): output_schema=GithubReadReleasesBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[ ( "releases", @@ -1033,10 +1039,9 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "label": "bug", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Label added successfully")], test_mock={"add_label": lambda *args, **kwargs: "Label added successfully"}, ) @@ -1121,10 +1126,9 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "label": "bug", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Label removed successfully")], test_mock={ "remove_label": lambda *args, **kwargs: "Label removed successfully" @@ -1213,10 +1217,9 @@ def __init__(self): test_input={ "pr_url": "https://github.com/owner/repo/pull/1", "reviewer": "reviewer_username", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Reviewer assigned successfully")], test_mock={ "assign_reviewer": lambda *args, **kwargs: "Reviewer assigned successfully" @@ -1308,10 +1311,9 @@ def __init__(self): test_input={ "pr_url": "https://github.com/owner/repo/pull/1", "reviewer": "reviewer_username", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Reviewer unassigned successfully")], test_mock={ "unassign_reviewer": lambda *args, **kwargs: "Reviewer unassigned successfully" @@ -1385,10 +1387,9 @@ def __init__(self): output_schema=GithubListReviewersBlock.Output, test_input={ "pr_url": "https://github.com/owner/repo/pull/1", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[ ( "reviewers", @@ -1486,10 +1487,9 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "assignee": "username1", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Issue assigned successfully")], test_mock={ "assign_issue": lambda *args, **kwargs: "Issue assigned successfully" @@ -1572,10 +1572,9 @@ def __init__(self): test_input={ "issue_url": "https://github.com/owner/repo/issues/1", "assignee": "username1", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Issue unassigned successfully")], test_mock={ "unassign_issue": lambda *args, **kwargs: "Issue unassigned successfully" @@ -1651,10 +1650,9 @@ def __init__(self): output_schema=GithubReadCodeownersFileBlock.Output, test_input={ "repo_url": "https://github.com/owner/repo", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("codeowners_content", "# CODEOWNERS content")], test_mock={ "read_codeowners": lambda *args, **kwargs: "# CODEOWNERS content" @@ -1726,10 +1724,9 @@ def __init__(self): test_input={ "repo_url": "https://github.com/owner/repo", "file_path": "path/to/file", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("file_content", "File content")], test_mock={"read_file": lambda *args, **kwargs: "File content"}, ) @@ -1805,10 +1802,9 @@ def __init__(self): "repo_url": "https://github.com/owner/repo", "path": "path/to/file_or_folder", "branch": "branch_name", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("content", "File or folder content")], test_mock={ "read_content": lambda *args, **kwargs: "File or folder content" @@ -1895,10 +1891,9 @@ def __init__(self): "repo_url": "https://github.com/owner/repo", "new_branch": "new_branch_name", "source_branch": "source_branch_name", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Branch created successfully")], test_mock={ "create_branch": lambda *args, **kwargs: "Branch created successfully" @@ -1984,10 +1979,9 @@ def __init__(self): test_input={ "repo_url": "https://github.com/owner/repo", "branch": "branch_name", - "credentials": { - "github_oauth_token": "your-github-oauth-token", - }, + "credentials": TEST_CREDENTIALS_INPUT, }, + test_credentials=TEST_CREDENTIALS, test_output=[("status", "Branch deleted successfully")], test_mock={ "delete_branch": lambda *args, **kwargs: "Branch deleted successfully" diff --git a/rnd/autogpt_server/autogpt_server/data/block.py b/rnd/autogpt_server/autogpt_server/data/block.py index 87b896fdadd0..c5796874afd4 100644 --- a/rnd/autogpt_server/autogpt_server/data/block.py +++ b/rnd/autogpt_server/autogpt_server/data/block.py @@ -1,10 +1,21 @@ import inspect from abc import ABC, abstractmethod from enum import Enum -from typing import Any, ClassVar, Generator, Generic, Type, TypeVar, cast, get_origin +from typing import ( + Any, + ClassVar, + Generator, + Generic, + Optional, + Type, + TypeVar, + cast, + get_origin, +) import jsonref import jsonschema +from autogpt_libs.supabase_integration_credentials_store.types import Credentials from prisma.models import AgentBlock from pydantic import BaseModel @@ -186,6 +197,7 @@ def __init__( test_input: BlockInput | list[BlockInput] | None = None, test_output: BlockData | list[BlockData] | None = None, test_mock: dict[str, Any] | None = None, + test_credentials: Optional[Credentials] = None, disabled: bool = False, static_output: bool = False, ui_type: BlockUIType = BlockUIType.STANDARD, @@ -213,6 +225,7 @@ def __init__( self.test_input = test_input self.test_output = test_output self.test_mock = test_mock + self.test_credentials = test_credentials self.description = description self.categories = categories or set() self.contributors = contributors or set() diff --git a/rnd/autogpt_server/autogpt_server/util/test.py b/rnd/autogpt_server/autogpt_server/util/test.py index 7b938b62511b..df427bf75176 100644 --- a/rnd/autogpt_server/autogpt_server/util/test.py +++ b/rnd/autogpt_server/autogpt_server/util/test.py @@ -4,6 +4,7 @@ from autogpt_server.data import db from autogpt_server.data.block import Block, initialize_blocks from autogpt_server.data.execution import ExecutionResult, ExecutionStatus +from autogpt_server.data.model import CREDENTIALS_FIELD_NAME from autogpt_server.data.queue import AsyncEventQueue from autogpt_server.data.user import create_default_user from autogpt_server.executor import ExecutionManager, ExecutionScheduler @@ -130,10 +131,19 @@ def execute_block_test(block: Block): else: log(f"{prefix} mock {mock_name} not found in block") + extra_exec_kwargs = {} + + if CREDENTIALS_FIELD_NAME in block.input_schema.model_fields: + if not block.test_credentials: + raise ValueError( + f"{prefix} requires credentials but has no test_credentials" + ) + extra_exec_kwargs[CREDENTIALS_FIELD_NAME] = block.test_credentials + for input_data in block.test_input: log(f"{prefix} in: {input_data}") - for output_name, output_data in block.execute(input_data): + for output_name, output_data in block.execute(input_data, **extra_exec_kwargs): if output_index >= len(block.test_output): raise ValueError(f"{prefix} produced output more than expected") ex_output_name, ex_output_data = block.test_output[output_index] From 34a3dd9764eaa7af974642e492312e07695a3707 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Fri, 20 Sep 2024 22:48:12 +0200 Subject: [PATCH 20/43] filter OAuth credentials in dropdown by required scopes for block --- autogpt_platform/frontend/src/hooks/useCredentials.ts | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/autogpt_platform/frontend/src/hooks/useCredentials.ts b/autogpt_platform/frontend/src/hooks/useCredentials.ts index 2072b83da204..ac0720d6cd13 100644 --- a/autogpt_platform/frontend/src/hooks/useCredentials.ts +++ b/autogpt_platform/frontend/src/hooks/useCredentials.ts @@ -58,11 +58,20 @@ export default function useCredentials(): CredentialsData | null { }; } + // Filter by OAuth credentials that have sufficient scopes for this block + const requiredScopes = credentialsSchema.credentials_scopes; + const savedOAuthCredentials = requiredScopes + ? provider.savedOAuthCredentials.filter((c) => + new Set(c.scopes).isSupersetOf(new Set(requiredScopes)), + ) + : provider.savedOAuthCredentials; + return { ...provider, schema: credentialsSchema, supportsApiKey, supportsOAuth2, + savedOAuthCredentials, isLoading: false, }; } From dd731f01aa9297c388834cc3323bbd31d2dc5b9a Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Mon, 23 Sep 2024 12:21:45 +0200 Subject: [PATCH 21/43] resolve block category comment --- autogpt_platform/backend/backend/data/block.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/autogpt_platform/backend/backend/data/block.py b/autogpt_platform/backend/backend/data/block.py index aa9cd26ba5bb..f4d05d648a54 100644 --- a/autogpt_platform/backend/backend/data/block.py +++ b/autogpt_platform/backend/backend/data/block.py @@ -49,7 +49,7 @@ class BlockCategory(Enum): INPUT = "Block that interacts with input of the graph." OUTPUT = "Block that interacts with output of the graph." LOGIC = "Programming logic to control the flow of your agent" - DEVELOPER_TOOLS = "Developer tools like Github." # added this so all the github blocks are in the same category for now, this is to be changed + DEVELOPER_TOOLS = "Developer tools such as GitHub blocks." def dict(self) -> dict[str, str]: return {"category": self.name, "description": self.value} From a64055b6754d62efffd54c38e64ba1df77d83136 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Mon, 23 Sep 2024 14:53:18 +0200 Subject: [PATCH 22/43] Break up `blocks/github.py` --- .../backend/backend/blocks/__init__.py | 15 +- .../backend/backend/blocks/github.py | 2025 ----------------- .../backend/backend/blocks/github/_auth.py | 45 + .../backend/backend/blocks/github/issues.py | 694 ++++++ .../backend/blocks/github/pull_requests.py | 523 +++++ .../backend/backend/blocks/github/repo.py | 798 +++++++ docs/content/server/new_blocks.md | 2 +- 7 files changed, 2068 insertions(+), 2034 deletions(-) delete mode 100644 autogpt_platform/backend/backend/blocks/github.py create mode 100644 autogpt_platform/backend/backend/blocks/github/_auth.py create mode 100644 autogpt_platform/backend/backend/blocks/github/issues.py create mode 100644 autogpt_platform/backend/backend/blocks/github/pull_requests.py create mode 100644 autogpt_platform/backend/backend/blocks/github/repo.py diff --git a/autogpt_platform/backend/backend/blocks/__init__.py b/autogpt_platform/backend/backend/blocks/__init__.py index 84364bdae354..d090aa41be01 100644 --- a/autogpt_platform/backend/backend/blocks/__init__.py +++ b/autogpt_platform/backend/backend/blocks/__init__.py @@ -1,4 +1,3 @@ -import glob import importlib import os import re @@ -8,17 +7,17 @@ # Dynamically load all modules under backend.blocks AVAILABLE_MODULES = [] -current_dir = os.path.dirname(__file__) -modules = glob.glob(os.path.join(current_dir, "*.py")) +current_dir = Path(__file__).parent modules = [ - Path(f).stem - for f in modules - if os.path.isfile(f) and f.endswith(".py") and not f.endswith("__init__.py") + str(f.relative_to(current_dir))[:-3].replace(os.path.sep, ".") + for f in current_dir.rglob("*.py") + if f.is_file() and f.name != "__init__.py" ] for module in modules: - if not re.match("^[a-z_]+$", module): + if not re.match("^[a-z_.]+$", module): raise ValueError( - f"Block module {module} error: module name must be lowercase, separated by underscores, and contain only alphabet characters" + f"Block module {module} error: module name must be lowercase, " + "separated by underscores, and contain only alphabet characters" ) importlib.import_module(f".{module}", package=__name__) diff --git a/autogpt_platform/backend/backend/blocks/github.py b/autogpt_platform/backend/backend/blocks/github.py deleted file mode 100644 index e991e7ff0bb6..000000000000 --- a/autogpt_platform/backend/backend/blocks/github.py +++ /dev/null @@ -1,2025 +0,0 @@ -import base64 -from typing import Literal - -import requests -from autogpt_libs.supabase_integration_credentials_store.types import ( - APIKeyCredentials, - OAuth2Credentials, -) -from pydantic import SecretStr - -from backend.data.block import Block, BlockCategory, BlockOutput, BlockSchema -from backend.data.model import CredentialsField, CredentialsMetaInput, SchemaField - -GithubCredentials = APIKeyCredentials | OAuth2Credentials -GithubCredentialsInput = CredentialsMetaInput[ - Literal["github"], Literal["api_key", "oauth2"] -] - - -def GithubCredentialsField(scope: str) -> GithubCredentialsInput: - """ - Creates a GitHub credentials input on a block. - - Params: - scope: The authorization scope needed for the block to work. ([list of available scopes](https://docs.github.com/en/apps/oauth-apps/building-oauth-apps/scopes-for-oauth-apps#available-scopes)) - """ # noqa - return CredentialsField( - provider="github", - supported_credential_types={"api_key", "oauth2"}, - required_scopes={scope}, - description="The GitHub integration can be used with OAuth, " - "or any API key with sufficient permissions for the blocks it is used on.", - ) - - -TEST_CREDENTIALS = APIKeyCredentials( - id="01234567-89ab-cdef-0123-456789abcdef", - provider="github", - api_key=SecretStr("mock-github-api-key"), - title="Mock GitHub API key", - expires_at=None, -) -TEST_CREDENTIALS_INPUT = { - "provider": TEST_CREDENTIALS.provider, - "id": TEST_CREDENTIALS.id, - "type": TEST_CREDENTIALS.type, - "title": TEST_CREDENTIALS.type, -} - - -class GithubCommentBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - issue_url: str = SchemaField( - description="URL of the GitHub issue or pull request", - placeholder="https://github.com/owner/repo/issues/1", - ) - comment: str = SchemaField( - description="Comment to post on the issue or pull request", - placeholder="Enter your comment", - ) - - class Output(BlockSchema): - status: str = SchemaField(description="Status of the comment posting operation") - error: str = SchemaField( - description="Error message if the comment posting failed" - ) - - def __init__(self): - super().__init__( - id="0001c3d4-5678-90ef-1234-567890abcdef", - description="This block posts a comment on a specified GitHub issue or pull request using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubCommentBlock.Input, - output_schema=GithubCommentBlock.Output, - test_input={ - "issue_url": "https://github.com/owner/repo/issues/1", - "comment": "This is a test comment.", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Comment posted successfully")], - test_mock={ - "post_comment": lambda *args, **kwargs: "Comment posted successfully" - }, - ) - - @staticmethod - def post_comment( - credentials: GithubCredentials, issue_url: str, comment: str - ) -> str: - try: - if "/pull/" in issue_url: - api_url = ( - issue_url.replace("github.com", "api.github.com/repos").replace( - "/pull/", "/issues/" - ) - + "/comments" - ) - else: - api_url = ( - issue_url.replace("github.com", "api.github.com/repos") - + "/comments" - ) - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - data = {"body": comment} - - response = requests.post(api_url, headers=headers, json=data) - response.raise_for_status() - - return "Comment posted successfully" - except Exception as e: - return f"Failed to post comment: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.post_comment( - credentials, - input_data.issue_url, - input_data.comment, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubMakeIssueBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - title: str = SchemaField( - description="Title of the issue", placeholder="Enter the issue title" - ) - body: str = SchemaField( - description="Body of the issue", placeholder="Enter the issue body" - ) - - class Output(BlockSchema): - status: str = SchemaField(description="Status of the issue creation operation") - error: str = SchemaField( - description="Error message if the issue creation failed" - ) - - def __init__(self): - super().__init__( - id="0002d3e4-5678-90ab-1234-567890abcdef", - description="This block creates a new issue on a specified GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubMakeIssueBlock.Input, - output_schema=GithubMakeIssueBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "title": "Test Issue", - "body": "This is a test issue.", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Issue created successfully")], - test_mock={ - "create_issue": lambda *args, **kwargs: "Issue created successfully" - }, - ) - - @staticmethod - def create_issue( - credentials: GithubCredentials, repo_url: str, title: str, body: str - ) -> str: - try: - api_url = repo_url.replace("github.com", "api.github.com/repos") + "/issues" - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - data = {"title": title, "body": body} - - response = requests.post(api_url, headers=headers, json=data) - response.raise_for_status() - - return "Issue created successfully" - except Exception as e: - return f"Failed to create issue: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.create_issue( - credentials, - input_data.repo_url, - input_data.title, - input_data.body, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubMakePRBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - title: str = SchemaField( - description="Title of the pull request", - placeholder="Enter the pull request title", - ) - body: str = SchemaField( - description="Body of the pull request", - placeholder="Enter the pull request body", - ) - head: str = SchemaField( - description="The name of the branch where your changes are implemented. For cross-repository pull requests in the same network, namespace head with a user like this: username:branch.", - placeholder="Enter the head branch", - ) - base: str = SchemaField( - description="The name of the branch you want the changes pulled into.", - placeholder="Enter the base branch", - ) - - class Output(BlockSchema): - status: str = SchemaField( - description="Status of the pull request creation operation" - ) - error: str = SchemaField( - description="Error message if the pull request creation failed" - ) - - def __init__(self): - super().__init__( - id="0003q3r4-5678-90ab-1234-567890abcdef", - description="This block creates a new pull request on a specified GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubMakePRBlock.Input, - output_schema=GithubMakePRBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "title": "Test Pull Request", - "body": "This is a test pull request.", - "head": "feature-branch", - "base": "main", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Pull request created successfully")], - test_mock={ - "create_pr": lambda *args, **kwargs: "Pull request created successfully" - }, - ) - - @staticmethod - def create_pr( - credentials: GithubCredentials, - repo_url: str, - title: str, - body: str, - head: str, - base: str, - ) -> str: - response = None - try: - repo_path = repo_url.replace("https://github.com/", "") - api_url = f"https://api.github.com/repos/{repo_path}/pulls" - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - data = {"title": title, "body": body, "head": head, "base": base} - - response = requests.post(api_url, headers=headers, json=data) - response.raise_for_status() - - return "Pull request created successfully" - except requests.exceptions.HTTPError as http_err: - if response and response.status_code == 422: - error_details = response.json() - return f"Failed to create pull request: {error_details.get('message', 'Unknown error')}" - return f"Failed to create pull request: {str(http_err)}" - except Exception as e: - return f"Failed to create pull request: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.create_pr( - credentials, - input_data.repo_url, - input_data.title, - input_data.body, - input_data.head, - input_data.base, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubReadIssueBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - issue_url: str = SchemaField( - description="URL of the GitHub issue", - placeholder="https://github.com/owner/repo/issues/1", - ) - - class Output(BlockSchema): - title: str = SchemaField(description="Title of the issue") - body: str = SchemaField(description="Body of the issue") - user: str = SchemaField(description="User who created the issue") - error: str = SchemaField( - description="Error message if reading the issue failed" - ) - - def __init__(self): - super().__init__( - id="0004e3f4-5678-90ab-1234-567890abcdef", - description="This block reads the body, title, and user of a specified GitHub issue using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadIssueBlock.Input, - output_schema=GithubReadIssueBlock.Output, - test_input={ - "issue_url": "https://github.com/owner/repo/issues/1", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[ - ("title", "Title of the issue"), - ("body", "This is the body of the issue."), - ("user", "username"), - ], - test_mock={ - "read_issue": lambda *args, **kwargs: ( - "Title of the issue", - "This is the body of the issue.", - "username", - ) - }, - ) - - @staticmethod - def read_issue( - credentials: GithubCredentials, issue_url: str - ) -> tuple[str, str, str]: - try: - api_url = issue_url.replace("github.com", "api.github.com/repos") - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - data = response.json() - title = data.get("title", "No title found") - body = data.get("body", "No body content found") - user = data.get("user", {}).get("login", "No user found") - - return title, body, user - except Exception as e: - return f"Failed to read issue: {str(e)}", "", "" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - title, body, user = self.read_issue( - credentials, - input_data.issue_url, - ) - if "Failed" in title: - yield "error", title - else: - yield "title", title - yield "body", body - yield "user", user - - -class GithubReadPRBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - pr_url: str = SchemaField( - description="URL of the GitHub pull request", - placeholder="https://github.com/owner/repo/pull/1", - ) - include_pr_changes: bool = SchemaField( - description="Whether to include the changes made in the pull request", - default=False, - ) - - class Output(BlockSchema): - title: str = SchemaField(description="Title of the pull request") - body: str = SchemaField(description="Body of the pull request") - user: str = SchemaField(description="User who created the pull request") - changes: str = SchemaField(description="Changes made in the pull request") - error: str = SchemaField( - description="Error message if reading the pull request failed" - ) - - def __init__(self): - super().__init__( - id="0005g3h4-5678-90ab-1234-567890abcdeg", - description="This block reads the body, title, user, and changes of a specified GitHub pull request using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadPRBlock.Input, - output_schema=GithubReadPRBlock.Output, - test_input={ - "pr_url": "https://github.com/owner/repo/pull/1", - "include_pr_changes": True, - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[ - ("title", "Title of the pull request"), - ("body", "This is the body of the pull request."), - ("user", "username"), - ("changes", "List of changes made in the pull request."), - ], - test_mock={ - "read_pr": lambda *args, **kwargs: ( - "Title of the pull request", - "This is the body of the pull request.", - "username", - ), - "read_pr_changes": lambda *args, **kwargs: "List of changes made in the pull request.", - }, - ) - - @staticmethod - def read_pr(credentials: GithubCredentials, pr_url: str) -> tuple[str, str, str]: - try: - api_url = pr_url.replace("github.com", "api.github.com/repos").replace( - "/pull/", "/issues/" - ) - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - data = response.json() - title = data.get("title", "No title found") - body = data.get("body", "No body content found") - user = data.get("user", {}).get("login", "No user found") - - return title, body, user - except Exception as e: - return f"Failed to read pull request: {str(e)}", "", "" - - @staticmethod - def read_pr_changes(credentials: GithubCredentials, pr_url: str) -> str: - try: - api_url = ( - pr_url.replace("github.com", "api.github.com/repos").replace( - "/pull/", "/pulls/" - ) - + "/files" - ) - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - files = response.json() - changes = [] - for file in files: - filename = file.get("filename") - patch = file.get("patch") - if filename and patch: - changes.append(f"File: {filename}\n{patch}") - - return "\n\n".join(changes) - except Exception as e: - return f"Failed to read PR changes: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - title, body, user = self.read_pr( - credentials, - input_data.pr_url, - ) - if "Failed" in title: - yield "error", title - else: - yield "title", title - yield "body", body - yield "user", user - - if input_data.include_pr_changes: - changes = self.read_pr_changes( - credentials, - input_data.pr_url, - ) - if "Failed" in changes: - yield "error", changes - else: - yield "changes", changes - else: - yield "changes", "Changes not included" - - -class GithubListIssuesBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - - class Output(BlockSchema): - issues: list[dict[str, str]] = SchemaField( - description="List of issues with their URLs" - ) - error: str = SchemaField(description="Error message if listing issues failed") - - def __init__(self): - super().__init__( - id="0006h3i4-5678-90ab-1234-567890abcdef", - description="This block lists all issues for a specified GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubListIssuesBlock.Input, - output_schema=GithubListIssuesBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[ - ( - "issues", - [ - { - "title": "Issue 1", - "url": "https://github.com/owner/repo/issues/1", - } - ], - ) - ], - test_mock={ - "list_issues": lambda *args, **kwargs: [ - { - "title": "Issue 1", - "url": "https://github.com/owner/repo/issues/1", - } - ] - }, - ) - - @staticmethod - def list_issues( - credentials: GithubCredentials, repo_url: str - ) -> list[dict[str, str]]: - try: - api_url = repo_url.replace("github.com", "api.github.com/repos") + "/issues" - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - data = response.json() - issues = [ - {"title": issue["title"], "url": issue["html_url"]} for issue in data - ] - - return issues - except Exception as e: - return [{"title": "Error", "url": f"Failed to list issues: {str(e)}"}] - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - issues = self.list_issues( - credentials, - input_data.repo_url, - ) - if any("Failed" in issue["url"] for issue in issues): - yield "error", issues[0]["url"] - else: - yield "issues", issues - - -class GithubReadTagsBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - - class Output(BlockSchema): - tags: list[dict[str, str]] = SchemaField( - description="List of tags with their names and URLs" - ) - error: str = SchemaField(description="Error message if listing tags failed") - - def __init__(self): - super().__init__( - id="0007g3h4-5678-90ab-1234-567890abcdef", - description="This block lists all tags for a specified GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadTagsBlock.Input, - output_schema=GithubReadTagsBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[ - ( - "tags", - [ - { - "name": "v1.0.0", - "url": "https://github.com/owner/repo/tree/v1.0.0", - } - ], - ) - ], - test_mock={ - "list_tags": lambda *args, **kwargs: [ - { - "name": "v1.0.0", - "url": "https://github.com/owner/repo/tree/v1.0.0", - } - ] - }, - ) - - @staticmethod - def list_tags( - credentials: GithubCredentials, repo_url: str - ) -> list[dict[str, str]]: - try: - repo_path = repo_url.replace("https://github.com/", "") - api_url = f"https://api.github.com/repos/{repo_path}/tags" - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - data = response.json() - tags = [ - { - "name": tag["name"], - "url": f"https://github.com/{repo_path}/tree/{tag['name']}", - } - for tag in data - ] - - return tags - except Exception as e: - return [{"name": "Error", "url": f"Failed to list tags: {str(e)}"}] - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - tags = self.list_tags( - credentials, - input_data.repo_url, - ) - if any("Failed" in tag["url"] for tag in tags): - yield "error", tags[0]["url"] - else: - yield "tags", tags - - -class GithubReadBranchesBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - - class Output(BlockSchema): - branches: list[dict[str, str]] = SchemaField( - description="List of branches with their names and URLs" - ) - error: str = SchemaField(description="Error message if listing branches failed") - - def __init__(self): - super().__init__( - id="0008i3j4-5678-90ab-1234-567890abcdef", - description="This block lists all branches for a specified GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadBranchesBlock.Input, - output_schema=GithubReadBranchesBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[ - ( - "branches", - [ - { - "name": "main", - "url": "https://github.com/owner/repo/tree/main", - } - ], - ) - ], - test_mock={ - "list_branches": lambda *args, **kwargs: [ - { - "name": "main", - "url": "https://github.com/owner/repo/tree/main", - } - ] - }, - ) - - @staticmethod - def list_branches( - credentials: GithubCredentials, repo_url: str - ) -> list[dict[str, str]]: - try: - api_url = ( - repo_url.replace("github.com", "api.github.com/repos") + "/branches" - ) - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - data = response.json() - branches = [ - {"name": branch["name"], "url": branch["commit"]["url"]} - for branch in data - ] - - return branches - except Exception as e: - return [{"name": "Error", "url": f"Failed to list branches: {str(e)}"}] - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - branches = self.list_branches( - credentials, - input_data.repo_url, - ) - if any("Failed" in branch["url"] for branch in branches): - yield "error", branches[0]["url"] - else: - yield "branches", branches - - -class GithubReadDiscussionsBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - num_discussions: int = SchemaField( - description="Number of discussions to fetch", default=5 - ) - - class Output(BlockSchema): - discussions: list[dict[str, str]] = SchemaField( - description="List of discussions with their titles and URLs" - ) - error: str = SchemaField( - description="Error message if listing discussions failed" - ) - - def __init__(self): - super().__init__( - id="0009j3k4-5678-90ab-1234-567890abcdef", - description="This block lists recent discussions for a specified GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadDiscussionsBlock.Input, - output_schema=GithubReadDiscussionsBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "num_discussions": 3, - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[ - ( - "discussions", - [ - { - "title": "Discussion 1", - "url": "https://github.com/owner/repo/discussions/1", - } - ], - ) - ], - test_mock={ - "list_discussions": lambda *args, **kwargs: [ - { - "title": "Discussion 1", - "url": "https://github.com/owner/repo/discussions/1", - } - ] - }, - ) - - @staticmethod - def list_discussions( - credentials: GithubCredentials, repo_url: str, num_discussions: int - ) -> list[dict[str, str]]: - try: - repo_path = repo_url.replace("https://github.com/", "") - owner, repo = repo_path.split("/") - query = """ - query($owner: String!, $repo: String!, $num: Int!) { - repository(owner: $owner, name: $repo) { - discussions(first: $num) { - nodes { - title - url - } - } - } - } - """ - variables = {"owner": owner, "repo": repo, "num": num_discussions} - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.post( - "https://api.github.com/graphql", - json={"query": query, "variables": variables}, - headers=headers, - ) - response.raise_for_status() - - data = response.json() - discussions = [ - {"title": discussion["title"], "url": discussion["url"]} - for discussion in data["data"]["repository"]["discussions"]["nodes"] - ] - - return discussions - except Exception as e: - return [{"title": "Error", "url": f"Failed to list discussions: {str(e)}"}] - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - discussions = self.list_discussions( - credentials, input_data.repo_url, input_data.num_discussions - ) - if any("Failed" in discussion["url"] for discussion in discussions): - yield "error", discussions[0]["url"] - else: - yield "discussions", discussions - - -class GithubReadReleasesBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - - class Output(BlockSchema): - releases: list[dict[str, str]] = SchemaField( - description="List of releases with their names and URLs" - ) - error: str = SchemaField(description="Error message if listing releases failed") - - def __init__(self): - super().__init__( - id="0010k3l4-5678-90ab-1234-567890abcdef", - description="This block lists all releases for a specified GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadReleasesBlock.Input, - output_schema=GithubReadReleasesBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[ - ( - "releases", - [ - { - "name": "v1.0.0", - "url": "https://github.com/owner/repo/releases/tag/v1.0.0", - } - ], - ) - ], - test_mock={ - "list_releases": lambda *args, **kwargs: [ - { - "name": "v1.0.0", - "url": "https://github.com/owner/repo/releases/tag/v1.0.0", - } - ] - }, - ) - - @staticmethod - def list_releases( - credentials: GithubCredentials, repo_url: str - ) -> list[dict[str, str]]: - try: - repo_path = repo_url.replace("https://github.com/", "") - api_url = f"https://api.github.com/repos/{repo_path}/releases" - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - data = response.json() - releases = [ - {"name": release["name"], "url": release["html_url"]} - for release in data - ] - - return releases - except Exception as e: - return [{"name": "Error", "url": f"Failed to list releases: {str(e)}"}] - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - releases = self.list_releases( - credentials, - input_data.repo_url, - ) - if any("Failed" in release["url"] for release in releases): - yield "error", releases[0]["url"] - else: - yield "releases", releases - - -class GithubAddLabelBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - issue_url: str = SchemaField( - description="URL of the GitHub issue or pull request", - placeholder="https://github.com/owner/repo/issues/1", - ) - label: str = SchemaField( - description="Label to add to the issue or pull request", - placeholder="Enter the label", - ) - - class Output(BlockSchema): - status: str = SchemaField(description="Status of the label addition operation") - error: str = SchemaField( - description="Error message if the label addition failed" - ) - - def __init__(self): - super().__init__( - id="0011l3m4-5678-90ab-1234-567890abcdef", - description="This block adds a label to a specified GitHub issue or pull request using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubAddLabelBlock.Input, - output_schema=GithubAddLabelBlock.Output, - test_input={ - "issue_url": "https://github.com/owner/repo/issues/1", - "label": "bug", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Label added successfully")], - test_mock={"add_label": lambda *args, **kwargs: "Label added successfully"}, - ) - - @staticmethod - def add_label(credentials: GithubCredentials, issue_url: str, label: str) -> str: - try: - # Convert the provided GitHub URL to the API URL - if "/pull/" in issue_url: - api_url = ( - issue_url.replace("github.com", "api.github.com/repos").replace( - "/pull/", "/issues/" - ) - + "/labels" - ) - else: - api_url = ( - issue_url.replace("github.com", "api.github.com/repos") + "/labels" - ) - - # Log the constructed API URL for debugging - print(f"Constructed API URL: {api_url}") - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - data = {"labels": [label]} - - response = requests.post(api_url, headers=headers, json=data) - response.raise_for_status() - - return "Label added successfully" - except requests.exceptions.HTTPError as http_err: - return f"HTTP error occurred: {http_err} - {http_err.response.text}" - except Exception as e: - return f"Failed to add label: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.add_label( - credentials, - input_data.issue_url, - input_data.label, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubRemoveLabelBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - issue_url: str = SchemaField( - description="URL of the GitHub issue or pull request", - placeholder="https://github.com/owner/repo/issues/1", - ) - label: str = SchemaField( - description="Label to remove from the issue or pull request", - placeholder="Enter the label", - ) - - class Output(BlockSchema): - status: str = SchemaField(description="Status of the label removal operation") - error: str = SchemaField( - description="Error message if the label removal failed" - ) - - def __init__(self): - super().__init__( - id="0012m3n4-5678-90ab-1234-567890abcdef", - description="This block removes a label from a specified GitHub issue or pull request using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubRemoveLabelBlock.Input, - output_schema=GithubRemoveLabelBlock.Output, - test_input={ - "issue_url": "https://github.com/owner/repo/issues/1", - "label": "bug", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Label removed successfully")], - test_mock={ - "remove_label": lambda *args, **kwargs: "Label removed successfully" - }, - ) - - @staticmethod - def remove_label(credentials: GithubCredentials, issue_url: str, label: str) -> str: - try: - # Convert the provided GitHub URL to the API URL - if "/pull/" in issue_url: - api_url = ( - issue_url.replace("github.com", "api.github.com/repos").replace( - "/pull/", "/issues/" - ) - + f"/labels/{label}" - ) - else: - api_url = ( - issue_url.replace("github.com", "api.github.com/repos") - + f"/labels/{label}" - ) - - # Log the constructed API URL for debugging - print(f"Constructed API URL: {api_url}") - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.delete(api_url, headers=headers) - response.raise_for_status() - - return "Label removed successfully" - except requests.exceptions.HTTPError as http_err: - return f"HTTP error occurred: {http_err} - {http_err.response.text}" - except Exception as e: - return f"Failed to remove label: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.remove_label( - credentials, - input_data.issue_url, - input_data.label, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubAssignReviewerBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - pr_url: str = SchemaField( - description="URL of the GitHub pull request", - placeholder="https://github.com/owner/repo/pull/1", - ) - reviewer: str = SchemaField( - description="Username of the reviewer to assign", - placeholder="Enter the reviewer's username", - ) - - class Output(BlockSchema): - status: str = SchemaField( - description="Status of the reviewer assignment operation" - ) - error: str = SchemaField( - description="Error message if the reviewer assignment failed" - ) - - def __init__(self): - super().__init__( - id="0014o3p4-5678-90ab-1234-567890abcdef", - description="This block assigns a reviewer to a specified GitHub pull request using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubAssignReviewerBlock.Input, - output_schema=GithubAssignReviewerBlock.Output, - test_input={ - "pr_url": "https://github.com/owner/repo/pull/1", - "reviewer": "reviewer_username", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Reviewer assigned successfully")], - test_mock={ - "assign_reviewer": lambda *args, **kwargs: "Reviewer assigned successfully" - }, - ) - - @staticmethod - def assign_reviewer( - credentials: GithubCredentials, pr_url: str, reviewer: str - ) -> str: - try: - # Convert the PR URL to the appropriate API endpoint - api_url = ( - pr_url.replace("github.com", "api.github.com/repos").replace( - "/pull/", "/pulls/" - ) - + "/requested_reviewers" - ) - - # Log the constructed API URL for debugging - print(f"Constructed API URL: {api_url}") - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - data = {"reviewers": [reviewer]} - - # Log the request data for debugging - print(f"Request data: {data}") - - response = requests.post(api_url, headers=headers, json=data) - response.raise_for_status() - - return "Reviewer assigned successfully" - except requests.exceptions.HTTPError as http_err: - if http_err.response.status_code == 422: - return f"Failed to assign reviewer: The reviewer '{reviewer}' may not have permission or the pull request is not in a valid state. Detailed error: {http_err.response.text}" - else: - return f"HTTP error occurred: {http_err} - {http_err.response.text}" - except Exception as e: - return f"Failed to assign reviewer: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.assign_reviewer( - credentials, - input_data.pr_url, - input_data.reviewer, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubUnassignReviewerBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - pr_url: str = SchemaField( - description="URL of the GitHub pull request", - placeholder="https://github.com/owner/repo/pull/1", - ) - reviewer: str = SchemaField( - description="Username of the reviewer to unassign", - placeholder="Enter the reviewer's username", - ) - - class Output(BlockSchema): - status: str = SchemaField( - description="Status of the reviewer unassignment operation" - ) - error: str = SchemaField( - description="Error message if the reviewer unassignment failed" - ) - - def __init__(self): - super().__init__( - id="0015p3q4-5678-90ab-1234-567890abcdef", - description="This block unassigns a reviewer from a specified GitHub pull request using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubUnassignReviewerBlock.Input, - output_schema=GithubUnassignReviewerBlock.Output, - test_input={ - "pr_url": "https://github.com/owner/repo/pull/1", - "reviewer": "reviewer_username", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Reviewer unassigned successfully")], - test_mock={ - "unassign_reviewer": lambda *args, **kwargs: "Reviewer unassigned successfully" - }, - ) - - @staticmethod - def unassign_reviewer( - credentials: GithubCredentials, pr_url: str, reviewer: str - ) -> str: - try: - api_url = ( - pr_url.replace("github.com", "api.github.com/repos").replace( - "/pull/", "/pulls/" - ) - + "/requested_reviewers" - ) - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - data = {"reviewers": [reviewer]} - - response = requests.delete(api_url, headers=headers, json=data) - response.raise_for_status() - - return "Reviewer unassigned successfully" - except Exception as e: - return f"Failed to unassign reviewer: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.unassign_reviewer( - credentials, - input_data.pr_url, - input_data.reviewer, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubListReviewersBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - pr_url: str = SchemaField( - description="URL of the GitHub pull request", - placeholder="https://github.com/owner/repo/pull/1", - ) - - class Output(BlockSchema): - reviewers: list[dict[str, str]] = SchemaField( - description="List of reviewers with their usernames and URLs" - ) - error: str = SchemaField( - description="Error message if listing reviewers failed" - ) - - def __init__(self): - super().__init__( - id="0016q3r4-5678-90ab-1234-567890abcdef", - description="This block lists all reviewers for a specified GitHub pull request using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubListReviewersBlock.Input, - output_schema=GithubListReviewersBlock.Output, - test_input={ - "pr_url": "https://github.com/owner/repo/pull/1", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[ - ( - "reviewers", - [ - { - "username": "reviewer1", - "url": "https://github.com/reviewer1", - } - ], - ) - ], - test_mock={ - "list_reviewers": lambda *args, **kwargs: [ - { - "username": "reviewer1", - "url": "https://github.com/reviewer1", - } - ] - }, - ) - - @staticmethod - def list_reviewers( - credentials: GithubCredentials, pr_url: str - ) -> list[dict[str, str]]: - try: - api_url = ( - pr_url.replace("github.com", "api.github.com/repos").replace( - "/pull/", "/pulls/" - ) - + "/requested_reviewers" - ) - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - data = response.json() - reviewers = [ - {"username": reviewer["login"], "url": reviewer["html_url"]} - for reviewer in data.get("users", []) - ] - - return reviewers - except Exception as e: - return [{"username": "Error", "url": f"Failed to list reviewers: {str(e)}"}] - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - reviewers = self.list_reviewers( - credentials, - input_data.pr_url, - ) - if any("Failed" in reviewer["url"] for reviewer in reviewers): - yield "error", reviewers[0]["url"] - else: - yield "reviewers", reviewers - - -class GithubAssignIssueBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - issue_url: str = SchemaField( - description="URL of the GitHub issue", - placeholder="https://github.com/owner/repo/issues/1", - ) - assignee: str = SchemaField( - description="Username to assign to the issue", - placeholder="Enter the username", - ) - - class Output(BlockSchema): - status: str = SchemaField( - description="Status of the issue assignment operation" - ) - error: str = SchemaField( - description="Error message if the issue assignment failed" - ) - - def __init__(self): - super().__init__( - id="0004r3s5-6789-01bc-2345-678901bcdefg", - description="This block assigns a user to a specified GitHub issue using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubAssignIssueBlock.Input, - output_schema=GithubAssignIssueBlock.Output, - test_input={ - "issue_url": "https://github.com/owner/repo/issues/1", - "assignee": "username1", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Issue assigned successfully")], - test_mock={ - "assign_issue": lambda *args, **kwargs: "Issue assigned successfully" - }, - ) - - @staticmethod - def assign_issue( - credentials: GithubCredentials, - issue_url: str, - assignee: str, - ) -> str: - try: - # Extracting repo path and issue number from the issue URL - repo_path, issue_number = issue_url.replace( - "https://github.com/", "" - ).split("/issues/") - api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - data = {"assignees": [assignee]} - - response = requests.post(api_url, headers=headers, json=data) - response.raise_for_status() - - return "Issue assigned successfully" - except requests.exceptions.HTTPError as http_err: - return f"Failed to assign issue: {str(http_err)}" - except Exception as e: - return f"Failed to assign issue: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.assign_issue( - credentials, - input_data.issue_url, - input_data.assignee, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubUnassignIssueBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - issue_url: str = SchemaField( - description="URL of the GitHub issue", - placeholder="https://github.com/owner/repo/issues/1", - ) - assignee: str = SchemaField( - description="Username to unassign from the issue", - placeholder="Enter the username", - ) - - class Output(BlockSchema): - status: str = SchemaField( - description="Status of the issue unassignment operation" - ) - error: str = SchemaField( - description="Error message if the issue unassignment failed" - ) - - def __init__(self): - super().__init__( - id="0005r3s6-7890-12cd-3456-789012cdefgh", - description="This block unassigns a user from a specified GitHub issue using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubUnassignIssueBlock.Input, - output_schema=GithubUnassignIssueBlock.Output, - test_input={ - "issue_url": "https://github.com/owner/repo/issues/1", - "assignee": "username1", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Issue unassigned successfully")], - test_mock={ - "unassign_issue": lambda *args, **kwargs: "Issue unassigned successfully" - }, - ) - - @staticmethod - def unassign_issue( - credentials: GithubCredentials, - issue_url: str, - assignee: str, - ) -> str: - try: - # Extracting repo path and issue number from the issue URL - repo_path, issue_number = issue_url.replace( - "https://github.com/", "" - ).split("/issues/") - api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" - - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - data = {"assignees": [assignee]} - - response = requests.delete(api_url, headers=headers, json=data) - response.raise_for_status() - - return "Issue unassigned successfully" - except requests.exceptions.HTTPError as http_err: - return f"Failed to unassign issue: {str(http_err)}" - except Exception as e: - return f"Failed to unassign issue: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.unassign_issue( - credentials, - input_data.issue_url, - input_data.assignee, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubReadCodeownersFileBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - - class Output(BlockSchema): - codeowners_content: str = SchemaField( - description="Content of the CODEOWNERS file" - ) - error: str = SchemaField(description="Error message if the file reading failed") - - def __init__(self): - super().__init__( - id="0006r3s7-8901-23de-4567-890123defghi", - description="This block reads the CODEOWNERS file from the master branch of a specified GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadCodeownersFileBlock.Input, - output_schema=GithubReadCodeownersFileBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("codeowners_content", "# CODEOWNERS content")], - test_mock={ - "read_codeowners": lambda *args, **kwargs: "# CODEOWNERS content" - }, - ) - - @staticmethod - def read_codeowners(credentials: GithubCredentials, repo_url: str) -> str: - try: - repo_path = repo_url.replace("https://github.com/", "") - api_url = f"https://api.github.com/repos/{repo_path}/contents/.github/CODEOWNERS?ref=master" - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - content = response.json() - return base64.b64decode(content["content"]).decode("utf-8") - except requests.exceptions.HTTPError as http_err: - return f"Failed to read CODEOWNERS file: {str(http_err)}" - except Exception as e: - return f"Failed to read CODEOWNERS file: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - content = self.read_codeowners( - credentials, - input_data.repo_url, - ) - if "Failed" not in content: - yield "codeowners_content", content - else: - yield "error", content - - -class GithubReadFileFromMasterBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - file_path: str = SchemaField( - description="Path to the file in the repository", - placeholder="path/to/file", - ) - - class Output(BlockSchema): - file_content: str = SchemaField( - description="Content of the file from the master branch" - ) - error: str = SchemaField(description="Error message if the file reading failed") - - def __init__(self): - super().__init__( - id="0007r3s8-9012-34ef-5678-901234efghij", - description="This block reads the content of a specified file from the master branch of a GitHub repository using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadFileFromMasterBlock.Input, - output_schema=GithubReadFileFromMasterBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "file_path": "path/to/file", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("file_content", "File content")], - test_mock={"read_file": lambda *args, **kwargs: "File content"}, - ) - - @staticmethod - def read_file(credentials: GithubCredentials, repo_url: str, file_path: str) -> str: - try: - repo_path = repo_url.replace("https://github.com/", "") - api_url = f"https://api.github.com/repos/{repo_path}/contents/{file_path}?ref=master" - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - content = response.json() - return base64.b64decode(content["content"]).decode("utf-8") - except requests.exceptions.HTTPError as http_err: - return f"Failed to read file: {str(http_err)}" - except Exception as e: - return f"Failed to read file: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - content = self.read_file( - credentials, - input_data.repo_url, - input_data.file_path, - ) - if "Failed" not in content: - yield "file_content", content - else: - yield "error", content - - -class GithubReadFileFolderRepoBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - path: str = SchemaField( - description="Path to the file/folder in the repository", - placeholder="path/to/file_or_folder", - ) - branch: str = SchemaField( - description="Branch name to read from", - placeholder="branch_name", - ) - - class Output(BlockSchema): - content: str = SchemaField( - description="Content of the file/folder/repo from the specified branch" - ) - error: str = SchemaField(description="Error message if the reading failed") - - def __init__(self): - super().__init__( - id="0008r3s9-0123-45fg-6789-012345fghijk", - description="This block reads the content of a specified file, folder, or repository from a specified branch using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubReadFileFolderRepoBlock.Input, - output_schema=GithubReadFileFolderRepoBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "path": "path/to/file_or_folder", - "branch": "branch_name", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("content", "File or folder content")], - test_mock={ - "read_content": lambda *args, **kwargs: "File or folder content" - }, - ) - - @staticmethod - def read_content( - credentials: GithubCredentials, repo_url: str, path: str, branch: str - ) -> str: - try: - repo_path = repo_url.replace("https://github.com/", "") - api_url = ( - f"https://api.github.com/repos/{repo_path}/contents/{path}?ref={branch}" - ) - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(api_url, headers=headers) - response.raise_for_status() - - content = response.json() - if "content" in content: - return base64.b64decode(content["content"]).decode("utf-8") - else: - return content # Return the directory content as JSON - - except requests.exceptions.HTTPError as http_err: - return f"Failed to read content: {str(http_err)}" - except Exception as e: - return f"Failed to read content: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - content = self.read_content( - credentials, - input_data.repo_url, - input_data.path, - input_data.branch, - ) - if "Failed" not in content: - yield "content", content - else: - yield "error", content - - -class GithubMakeBranchBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - new_branch: str = SchemaField( - description="Name of the new branch", - placeholder="new_branch_name", - ) - source_branch: str = SchemaField( - description="Name of the source branch", - placeholder="source_branch_name", - ) - - class Output(BlockSchema): - status: str = SchemaField(description="Status of the branch creation operation") - error: str = SchemaField( - description="Error message if the branch creation failed" - ) - - def __init__(self): - super().__init__( - id="0008r3s9-0123-45fg-6789-012345fghijp", - description="This block creates a new branch from a specified source branch using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubMakeBranchBlock.Input, - output_schema=GithubMakeBranchBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "new_branch": "new_branch_name", - "source_branch": "source_branch_name", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Branch created successfully")], - test_mock={ - "create_branch": lambda *args, **kwargs: "Branch created successfully" - }, - ) - - @staticmethod - def create_branch( - credentials: GithubCredentials, - repo_url: str, - new_branch: str, - source_branch: str, - ) -> str: - try: - repo_path = repo_url.replace("https://github.com/", "") - ref_api_url = f"https://api.github.com/repos/{repo_path}/git/refs/heads/{source_branch}" - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.get(ref_api_url, headers=headers) - response.raise_for_status() - - sha = response.json()["object"]["sha"] - - create_branch_api_url = f"https://api.github.com/repos/{repo_path}/git/refs" - data = {"ref": f"refs/heads/{new_branch}", "sha": sha} - - response = requests.post(create_branch_api_url, headers=headers, json=data) - response.raise_for_status() - - return "Branch created successfully" - except requests.exceptions.HTTPError as http_err: - return f"Failed to create branch: {str(http_err)}" - except Exception as e: - return f"Failed to create branch: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.create_branch( - credentials, - input_data.repo_url, - input_data.new_branch, - input_data.source_branch, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status - - -class GithubDeleteBranchBlock(Block): - class Input(BlockSchema): - credentials: GithubCredentialsInput = GithubCredentialsField("repo") - repo_url: str = SchemaField( - description="URL of the GitHub repository", - placeholder="https://github.com/owner/repo", - ) - branch: str = SchemaField( - description="Name of the branch to delete", - placeholder="branch_name", - ) - - class Output(BlockSchema): - status: str = SchemaField(description="Status of the branch deletion operation") - error: str = SchemaField( - description="Error message if the branch deletion failed" - ) - - def __init__(self): - super().__init__( - id="0008r3s9-0123-45fg-6789-012345fghijq", - description="This block deletes a specified branch using OAuth credentials.", - categories={BlockCategory.DEVELOPER_TOOLS}, - input_schema=GithubDeleteBranchBlock.Input, - output_schema=GithubDeleteBranchBlock.Output, - test_input={ - "repo_url": "https://github.com/owner/repo", - "branch": "branch_name", - "credentials": TEST_CREDENTIALS_INPUT, - }, - test_credentials=TEST_CREDENTIALS, - test_output=[("status", "Branch deleted successfully")], - test_mock={ - "delete_branch": lambda *args, **kwargs: "Branch deleted successfully" - }, - ) - - @staticmethod - def delete_branch( - credentials: GithubCredentials, repo_url: str, branch: str - ) -> str: - try: - repo_path = repo_url.replace("https://github.com/", "") - api_url = ( - f"https://api.github.com/repos/{repo_path}/git/refs/heads/{branch}" - ) - headers = { - "Authorization": credentials.bearer(), - "Accept": "application/vnd.github.v3+json", - } - - response = requests.delete(api_url, headers=headers) - response.raise_for_status() - - return "Branch deleted successfully" - except requests.exceptions.HTTPError as http_err: - return f"Failed to delete branch: {str(http_err)}" - except Exception as e: - return f"Failed to delete branch: {str(e)}" - - def run( - self, - input_data: Input, - *, - credentials: GithubCredentials, - **kwargs, - ) -> BlockOutput: - status = self.delete_branch( - credentials, - input_data.repo_url, - input_data.branch, - ) - if "successfully" in status: - yield "status", status - else: - yield "error", status diff --git a/autogpt_platform/backend/backend/blocks/github/_auth.py b/autogpt_platform/backend/backend/blocks/github/_auth.py new file mode 100644 index 000000000000..6946124360af --- /dev/null +++ b/autogpt_platform/backend/backend/blocks/github/_auth.py @@ -0,0 +1,45 @@ +from typing import Literal + +from autogpt_libs.supabase_integration_credentials_store.types import ( + APIKeyCredentials, + OAuth2Credentials, +) +from pydantic import SecretStr + +from backend.data.model import CredentialsField, CredentialsMetaInput + +GithubCredentials = APIKeyCredentials | OAuth2Credentials +GithubCredentialsInput = CredentialsMetaInput[ + Literal["github"], Literal["api_key", "oauth2"] +] + + +def GithubCredentialsField(scope: str) -> GithubCredentialsInput: + """ + Creates a GitHub credentials input on a block. + + Params: + scope: The authorization scope needed for the block to work. ([list of available scopes](https://docs.github.com/en/apps/oauth-apps/building-oauth-apps/scopes-for-oauth-apps#available-scopes)) + """ # noqa + return CredentialsField( + provider="github", + supported_credential_types={"api_key", "oauth2"}, + required_scopes={scope}, + description="The GitHub integration can be used with OAuth, " + "or any API key with sufficient permissions for the blocks it is used on.", + ) + + +TEST_CREDENTIALS = APIKeyCredentials( + id="01234567-89ab-cdef-0123-456789abcdef", + provider="github", + api_key=SecretStr("mock-github-api-key"), + title="Mock GitHub API key", + expires_at=None, +) +TEST_CREDENTIALS_INPUT = { + "provider": TEST_CREDENTIALS.provider, + "id": TEST_CREDENTIALS.id, + "type": TEST_CREDENTIALS.type, + "title": TEST_CREDENTIALS.type, +} diff --git a/autogpt_platform/backend/backend/blocks/github/issues.py b/autogpt_platform/backend/backend/blocks/github/issues.py new file mode 100644 index 000000000000..97552a307287 --- /dev/null +++ b/autogpt_platform/backend/backend/blocks/github/issues.py @@ -0,0 +1,694 @@ +import requests + +from backend.data.block import Block, BlockCategory, BlockOutput, BlockSchema +from backend.data.model import SchemaField + +from ._auth import ( + TEST_CREDENTIALS, + TEST_CREDENTIALS_INPUT, + GithubCredentials, + GithubCredentialsField, + GithubCredentialsInput, +) + + +class GithubCommentBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + issue_url: str = SchemaField( + description="URL of the GitHub issue or pull request", + placeholder="https://github.com/owner/repo/issues/1", + ) + comment: str = SchemaField( + description="Comment to post on the issue or pull request", + placeholder="Enter your comment", + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the comment posting operation") + error: str = SchemaField( + description="Error message if the comment posting failed" + ) + + def __init__(self): + super().__init__( + id="0001c3d4-5678-90ef-1234-567890abcdef", + description="This block posts a comment on a specified GitHub issue or pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubCommentBlock.Input, + output_schema=GithubCommentBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "comment": "This is a test comment.", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Comment posted successfully")], + test_mock={ + "post_comment": lambda *args, **kwargs: "Comment posted successfully" + }, + ) + + @staticmethod + def post_comment( + credentials: GithubCredentials, issue_url: str, comment: str + ) -> str: + try: + if "/pull/" in issue_url: + api_url = ( + issue_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/issues/" + ) + + "/comments" + ) + else: + api_url = ( + issue_url.replace("github.com", "api.github.com/repos") + + "/comments" + ) + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + data = {"body": comment} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Comment posted successfully" + except Exception as e: + return f"Failed to post comment: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.post_comment( + credentials, + input_data.issue_url, + input_data.comment, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubMakeIssueBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + title: str = SchemaField( + description="Title of the issue", placeholder="Enter the issue title" + ) + body: str = SchemaField( + description="Body of the issue", placeholder="Enter the issue body" + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the issue creation operation") + error: str = SchemaField( + description="Error message if the issue creation failed" + ) + + def __init__(self): + super().__init__( + id="0002d3e4-5678-90ab-1234-567890abcdef", + description="This block creates a new issue on a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubMakeIssueBlock.Input, + output_schema=GithubMakeIssueBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "title": "Test Issue", + "body": "This is a test issue.", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Issue created successfully")], + test_mock={ + "create_issue": lambda *args, **kwargs: "Issue created successfully" + }, + ) + + @staticmethod + def create_issue( + credentials: GithubCredentials, repo_url: str, title: str, body: str + ) -> str: + try: + api_url = repo_url.replace("github.com", "api.github.com/repos") + "/issues" + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + data = {"title": title, "body": body} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Issue created successfully" + except Exception as e: + return f"Failed to create issue: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.create_issue( + credentials, + input_data.repo_url, + input_data.title, + input_data.body, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubReadIssueBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + issue_url: str = SchemaField( + description="URL of the GitHub issue", + placeholder="https://github.com/owner/repo/issues/1", + ) + + class Output(BlockSchema): + title: str = SchemaField(description="Title of the issue") + body: str = SchemaField(description="Body of the issue") + user: str = SchemaField(description="User who created the issue") + error: str = SchemaField( + description="Error message if reading the issue failed" + ) + + def __init__(self): + super().__init__( + id="0004e3f4-5678-90ab-1234-567890abcdef", + description="This block reads the body, title, and user of a specified GitHub issue using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadIssueBlock.Input, + output_schema=GithubReadIssueBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[ + ("title", "Title of the issue"), + ("body", "This is the body of the issue."), + ("user", "username"), + ], + test_mock={ + "read_issue": lambda *args, **kwargs: ( + "Title of the issue", + "This is the body of the issue.", + "username", + ) + }, + ) + + @staticmethod + def read_issue( + credentials: GithubCredentials, issue_url: str + ) -> tuple[str, str, str]: + try: + api_url = issue_url.replace("github.com", "api.github.com/repos") + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + title = data.get("title", "No title found") + body = data.get("body", "No body content found") + user = data.get("user", {}).get("login", "No user found") + + return title, body, user + except Exception as e: + return f"Failed to read issue: {str(e)}", "", "" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + title, body, user = self.read_issue( + credentials, + input_data.issue_url, + ) + if "Failed" in title: + yield "error", title + else: + yield "title", title + yield "body", body + yield "user", user + + +class GithubListIssuesBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + + class Output(BlockSchema): + issues: list[dict[str, str]] = SchemaField( + description="List of issues with their URLs" + ) + error: str = SchemaField(description="Error message if listing issues failed") + + def __init__(self): + super().__init__( + id="0006h3i4-5678-90ab-1234-567890abcdef", + description="This block lists all issues for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubListIssuesBlock.Input, + output_schema=GithubListIssuesBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[ + ( + "issues", + [ + { + "title": "Issue 1", + "url": "https://github.com/owner/repo/issues/1", + } + ], + ) + ], + test_mock={ + "list_issues": lambda *args, **kwargs: [ + { + "title": "Issue 1", + "url": "https://github.com/owner/repo/issues/1", + } + ] + }, + ) + + @staticmethod + def list_issues( + credentials: GithubCredentials, repo_url: str + ) -> list[dict[str, str]]: + try: + api_url = repo_url.replace("github.com", "api.github.com/repos") + "/issues" + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + issues = [ + {"title": issue["title"], "url": issue["html_url"]} for issue in data + ] + + return issues + except Exception as e: + return [{"title": "Error", "url": f"Failed to list issues: {str(e)}"}] + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + issues = self.list_issues( + credentials, + input_data.repo_url, + ) + if any("Failed" in issue["url"] for issue in issues): + yield "error", issues[0]["url"] + else: + yield "issues", issues + + +class GithubAddLabelBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + issue_url: str = SchemaField( + description="URL of the GitHub issue or pull request", + placeholder="https://github.com/owner/repo/issues/1", + ) + label: str = SchemaField( + description="Label to add to the issue or pull request", + placeholder="Enter the label", + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the label addition operation") + error: str = SchemaField( + description="Error message if the label addition failed" + ) + + def __init__(self): + super().__init__( + id="0011l3m4-5678-90ab-1234-567890abcdef", + description="This block adds a label to a specified GitHub issue or pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubAddLabelBlock.Input, + output_schema=GithubAddLabelBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "label": "bug", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Label added successfully")], + test_mock={"add_label": lambda *args, **kwargs: "Label added successfully"}, + ) + + @staticmethod + def add_label(credentials: GithubCredentials, issue_url: str, label: str) -> str: + try: + # Convert the provided GitHub URL to the API URL + if "/pull/" in issue_url: + api_url = ( + issue_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/issues/" + ) + + "/labels" + ) + else: + api_url = ( + issue_url.replace("github.com", "api.github.com/repos") + "/labels" + ) + + # Log the constructed API URL for debugging + print(f"Constructed API URL: {api_url}") + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + data = {"labels": [label]} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Label added successfully" + except requests.exceptions.HTTPError as http_err: + return f"HTTP error occurred: {http_err} - {http_err.response.text}" + except Exception as e: + return f"Failed to add label: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.add_label( + credentials, + input_data.issue_url, + input_data.label, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubRemoveLabelBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + issue_url: str = SchemaField( + description="URL of the GitHub issue or pull request", + placeholder="https://github.com/owner/repo/issues/1", + ) + label: str = SchemaField( + description="Label to remove from the issue or pull request", + placeholder="Enter the label", + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the label removal operation") + error: str = SchemaField( + description="Error message if the label removal failed" + ) + + def __init__(self): + super().__init__( + id="0012m3n4-5678-90ab-1234-567890abcdef", + description="This block removes a label from a specified GitHub issue or pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubRemoveLabelBlock.Input, + output_schema=GithubRemoveLabelBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "label": "bug", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Label removed successfully")], + test_mock={ + "remove_label": lambda *args, **kwargs: "Label removed successfully" + }, + ) + + @staticmethod + def remove_label(credentials: GithubCredentials, issue_url: str, label: str) -> str: + try: + # Convert the provided GitHub URL to the API URL + if "/pull/" in issue_url: + api_url = ( + issue_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/issues/" + ) + + f"/labels/{label}" + ) + else: + api_url = ( + issue_url.replace("github.com", "api.github.com/repos") + + f"/labels/{label}" + ) + + # Log the constructed API URL for debugging + print(f"Constructed API URL: {api_url}") + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.delete(api_url, headers=headers) + response.raise_for_status() + + return "Label removed successfully" + except requests.exceptions.HTTPError as http_err: + return f"HTTP error occurred: {http_err} - {http_err.response.text}" + except Exception as e: + return f"Failed to remove label: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.remove_label( + credentials, + input_data.issue_url, + input_data.label, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubAssignIssueBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + issue_url: str = SchemaField( + description="URL of the GitHub issue", + placeholder="https://github.com/owner/repo/issues/1", + ) + assignee: str = SchemaField( + description="Username to assign to the issue", + placeholder="Enter the username", + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the issue assignment operation" + ) + error: str = SchemaField( + description="Error message if the issue assignment failed" + ) + + def __init__(self): + super().__init__( + id="0004r3s5-6789-01bc-2345-678901bcdefg", + description="This block assigns a user to a specified GitHub issue using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubAssignIssueBlock.Input, + output_schema=GithubAssignIssueBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "assignee": "username1", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Issue assigned successfully")], + test_mock={ + "assign_issue": lambda *args, **kwargs: "Issue assigned successfully" + }, + ) + + @staticmethod + def assign_issue( + credentials: GithubCredentials, + issue_url: str, + assignee: str, + ) -> str: + try: + # Extracting repo path and issue number from the issue URL + repo_path, issue_number = issue_url.replace( + "https://github.com/", "" + ).split("/issues/") + api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + data = {"assignees": [assignee]} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Issue assigned successfully" + except requests.exceptions.HTTPError as http_err: + return f"Failed to assign issue: {str(http_err)}" + except Exception as e: + return f"Failed to assign issue: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.assign_issue( + credentials, + input_data.issue_url, + input_data.assignee, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubUnassignIssueBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + issue_url: str = SchemaField( + description="URL of the GitHub issue", + placeholder="https://github.com/owner/repo/issues/1", + ) + assignee: str = SchemaField( + description="Username to unassign from the issue", + placeholder="Enter the username", + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the issue unassignment operation" + ) + error: str = SchemaField( + description="Error message if the issue unassignment failed" + ) + + def __init__(self): + super().__init__( + id="0005r3s6-7890-12cd-3456-789012cdefgh", + description="This block unassigns a user from a specified GitHub issue using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubUnassignIssueBlock.Input, + output_schema=GithubUnassignIssueBlock.Output, + test_input={ + "issue_url": "https://github.com/owner/repo/issues/1", + "assignee": "username1", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Issue unassigned successfully")], + test_mock={ + "unassign_issue": lambda *args, **kwargs: "Issue unassigned successfully" + }, + ) + + @staticmethod + def unassign_issue( + credentials: GithubCredentials, + issue_url: str, + assignee: str, + ) -> str: + try: + # Extracting repo path and issue number from the issue URL + repo_path, issue_number = issue_url.replace( + "https://github.com/", "" + ).split("/issues/") + api_url = f"https://api.github.com/repos/{repo_path}/issues/{issue_number}/assignees" + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + data = {"assignees": [assignee]} + + response = requests.delete(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Issue unassigned successfully" + except requests.exceptions.HTTPError as http_err: + return f"Failed to unassign issue: {str(http_err)}" + except Exception as e: + return f"Failed to unassign issue: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.unassign_issue( + credentials, + input_data.issue_url, + input_data.assignee, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status diff --git a/autogpt_platform/backend/backend/blocks/github/pull_requests.py b/autogpt_platform/backend/backend/blocks/github/pull_requests.py new file mode 100644 index 000000000000..f4448ba2ded1 --- /dev/null +++ b/autogpt_platform/backend/backend/blocks/github/pull_requests.py @@ -0,0 +1,523 @@ +import requests + +from backend.data.block import Block, BlockCategory, BlockOutput, BlockSchema +from backend.data.model import SchemaField + +from ._auth import ( + TEST_CREDENTIALS, + TEST_CREDENTIALS_INPUT, + GithubCredentials, + GithubCredentialsField, + GithubCredentialsInput, +) + + +class GithubMakePRBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + title: str = SchemaField( + description="Title of the pull request", + placeholder="Enter the pull request title", + ) + body: str = SchemaField( + description="Body of the pull request", + placeholder="Enter the pull request body", + ) + head: str = SchemaField( + description="The name of the branch where your changes are implemented. For cross-repository pull requests in the same network, namespace head with a user like this: username:branch.", + placeholder="Enter the head branch", + ) + base: str = SchemaField( + description="The name of the branch you want the changes pulled into.", + placeholder="Enter the base branch", + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the pull request creation operation" + ) + error: str = SchemaField( + description="Error message if the pull request creation failed" + ) + + def __init__(self): + super().__init__( + id="0003q3r4-5678-90ab-1234-567890abcdef", + description="This block creates a new pull request on a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubMakePRBlock.Input, + output_schema=GithubMakePRBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "title": "Test Pull Request", + "body": "This is a test pull request.", + "head": "feature-branch", + "base": "main", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Pull request created successfully")], + test_mock={ + "create_pr": lambda *args, **kwargs: "Pull request created successfully" + }, + ) + + @staticmethod + def create_pr( + credentials: GithubCredentials, + repo_url: str, + title: str, + body: str, + head: str, + base: str, + ) -> str: + response = None + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/pulls" + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + data = {"title": title, "body": body, "head": head, "base": base} + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Pull request created successfully" + except requests.exceptions.HTTPError as http_err: + if response and response.status_code == 422: + error_details = response.json() + return f"Failed to create pull request: {error_details.get('message', 'Unknown error')}" + return f"Failed to create pull request: {str(http_err)}" + except Exception as e: + return f"Failed to create pull request: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.create_pr( + credentials, + input_data.repo_url, + input_data.title, + input_data.body, + input_data.head, + input_data.base, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubReadPRBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + pr_url: str = SchemaField( + description="URL of the GitHub pull request", + placeholder="https://github.com/owner/repo/pull/1", + ) + include_pr_changes: bool = SchemaField( + description="Whether to include the changes made in the pull request", + default=False, + ) + + class Output(BlockSchema): + title: str = SchemaField(description="Title of the pull request") + body: str = SchemaField(description="Body of the pull request") + user: str = SchemaField(description="User who created the pull request") + changes: str = SchemaField(description="Changes made in the pull request") + error: str = SchemaField( + description="Error message if reading the pull request failed" + ) + + def __init__(self): + super().__init__( + id="0005g3h4-5678-90ab-1234-567890abcdeg", + description="This block reads the body, title, user, and changes of a specified GitHub pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadPRBlock.Input, + output_schema=GithubReadPRBlock.Output, + test_input={ + "pr_url": "https://github.com/owner/repo/pull/1", + "include_pr_changes": True, + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[ + ("title", "Title of the pull request"), + ("body", "This is the body of the pull request."), + ("user", "username"), + ("changes", "List of changes made in the pull request."), + ], + test_mock={ + "read_pr": lambda *args, **kwargs: ( + "Title of the pull request", + "This is the body of the pull request.", + "username", + ), + "read_pr_changes": lambda *args, **kwargs: "List of changes made in the pull request.", + }, + ) + + @staticmethod + def read_pr(credentials: GithubCredentials, pr_url: str) -> tuple[str, str, str]: + try: + api_url = pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/issues/" + ) + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + title = data.get("title", "No title found") + body = data.get("body", "No body content found") + user = data.get("user", {}).get("login", "No user found") + + return title, body, user + except Exception as e: + return f"Failed to read pull request: {str(e)}", "", "" + + @staticmethod + def read_pr_changes(credentials: GithubCredentials, pr_url: str) -> str: + try: + api_url = ( + pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/pulls/" + ) + + "/files" + ) + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + files = response.json() + changes = [] + for file in files: + filename = file.get("filename") + patch = file.get("patch") + if filename and patch: + changes.append(f"File: {filename}\n{patch}") + + return "\n\n".join(changes) + except Exception as e: + return f"Failed to read PR changes: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + title, body, user = self.read_pr( + credentials, + input_data.pr_url, + ) + if "Failed" in title: + yield "error", title + else: + yield "title", title + yield "body", body + yield "user", user + + if input_data.include_pr_changes: + changes = self.read_pr_changes( + credentials, + input_data.pr_url, + ) + if "Failed" in changes: + yield "error", changes + else: + yield "changes", changes + else: + yield "changes", "Changes not included" + + +class GithubAssignReviewerBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + pr_url: str = SchemaField( + description="URL of the GitHub pull request", + placeholder="https://github.com/owner/repo/pull/1", + ) + reviewer: str = SchemaField( + description="Username of the reviewer to assign", + placeholder="Enter the reviewer's username", + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the reviewer assignment operation" + ) + error: str = SchemaField( + description="Error message if the reviewer assignment failed" + ) + + def __init__(self): + super().__init__( + id="0014o3p4-5678-90ab-1234-567890abcdef", + description="This block assigns a reviewer to a specified GitHub pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubAssignReviewerBlock.Input, + output_schema=GithubAssignReviewerBlock.Output, + test_input={ + "pr_url": "https://github.com/owner/repo/pull/1", + "reviewer": "reviewer_username", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Reviewer assigned successfully")], + test_mock={ + "assign_reviewer": lambda *args, **kwargs: "Reviewer assigned successfully" + }, + ) + + @staticmethod + def assign_reviewer( + credentials: GithubCredentials, pr_url: str, reviewer: str + ) -> str: + try: + # Convert the PR URL to the appropriate API endpoint + api_url = ( + pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/pulls/" + ) + + "/requested_reviewers" + ) + + # Log the constructed API URL for debugging + print(f"Constructed API URL: {api_url}") + + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + data = {"reviewers": [reviewer]} + + # Log the request data for debugging + print(f"Request data: {data}") + + response = requests.post(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Reviewer assigned successfully" + except requests.exceptions.HTTPError as http_err: + if http_err.response.status_code == 422: + return f"Failed to assign reviewer: The reviewer '{reviewer}' may not have permission or the pull request is not in a valid state. Detailed error: {http_err.response.text}" + else: + return f"HTTP error occurred: {http_err} - {http_err.response.text}" + except Exception as e: + return f"Failed to assign reviewer: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.assign_reviewer( + credentials, + input_data.pr_url, + input_data.reviewer, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubUnassignReviewerBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + pr_url: str = SchemaField( + description="URL of the GitHub pull request", + placeholder="https://github.com/owner/repo/pull/1", + ) + reviewer: str = SchemaField( + description="Username of the reviewer to unassign", + placeholder="Enter the reviewer's username", + ) + + class Output(BlockSchema): + status: str = SchemaField( + description="Status of the reviewer unassignment operation" + ) + error: str = SchemaField( + description="Error message if the reviewer unassignment failed" + ) + + def __init__(self): + super().__init__( + id="0015p3q4-5678-90ab-1234-567890abcdef", + description="This block unassigns a reviewer from a specified GitHub pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubUnassignReviewerBlock.Input, + output_schema=GithubUnassignReviewerBlock.Output, + test_input={ + "pr_url": "https://github.com/owner/repo/pull/1", + "reviewer": "reviewer_username", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Reviewer unassigned successfully")], + test_mock={ + "unassign_reviewer": lambda *args, **kwargs: "Reviewer unassigned successfully" + }, + ) + + @staticmethod + def unassign_reviewer( + credentials: GithubCredentials, pr_url: str, reviewer: str + ) -> str: + try: + api_url = ( + pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/pulls/" + ) + + "/requested_reviewers" + ) + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + data = {"reviewers": [reviewer]} + + response = requests.delete(api_url, headers=headers, json=data) + response.raise_for_status() + + return "Reviewer unassigned successfully" + except Exception as e: + return f"Failed to unassign reviewer: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.unassign_reviewer( + credentials, + input_data.pr_url, + input_data.reviewer, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubListReviewersBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + pr_url: str = SchemaField( + description="URL of the GitHub pull request", + placeholder="https://github.com/owner/repo/pull/1", + ) + + class Output(BlockSchema): + reviewers: list[dict[str, str]] = SchemaField( + description="List of reviewers with their usernames and URLs" + ) + error: str = SchemaField( + description="Error message if listing reviewers failed" + ) + + def __init__(self): + super().__init__( + id="0016q3r4-5678-90ab-1234-567890abcdef", + description="This block lists all reviewers for a specified GitHub pull request using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubListReviewersBlock.Input, + output_schema=GithubListReviewersBlock.Output, + test_input={ + "pr_url": "https://github.com/owner/repo/pull/1", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[ + ( + "reviewers", + [ + { + "username": "reviewer1", + "url": "https://github.com/reviewer1", + } + ], + ) + ], + test_mock={ + "list_reviewers": lambda *args, **kwargs: [ + { + "username": "reviewer1", + "url": "https://github.com/reviewer1", + } + ] + }, + ) + + @staticmethod + def list_reviewers( + credentials: GithubCredentials, pr_url: str + ) -> list[dict[str, str]]: + try: + api_url = ( + pr_url.replace("github.com", "api.github.com/repos").replace( + "/pull/", "/pulls/" + ) + + "/requested_reviewers" + ) + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + reviewers = [ + {"username": reviewer["login"], "url": reviewer["html_url"]} + for reviewer in data.get("users", []) + ] + + return reviewers + except Exception as e: + return [{"username": "Error", "url": f"Failed to list reviewers: {str(e)}"}] + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + reviewers = self.list_reviewers( + credentials, + input_data.pr_url, + ) + if any("Failed" in reviewer["url"] for reviewer in reviewers): + yield "error", reviewers[0]["url"] + else: + yield "reviewers", reviewers diff --git a/autogpt_platform/backend/backend/blocks/github/repo.py b/autogpt_platform/backend/backend/blocks/github/repo.py new file mode 100644 index 000000000000..3679849e731b --- /dev/null +++ b/autogpt_platform/backend/backend/blocks/github/repo.py @@ -0,0 +1,798 @@ +import base64 + +import requests + +from backend.data.block import Block, BlockCategory, BlockOutput, BlockSchema +from backend.data.model import SchemaField + +from ._auth import ( + TEST_CREDENTIALS, + TEST_CREDENTIALS_INPUT, + GithubCredentials, + GithubCredentialsField, + GithubCredentialsInput, +) + + +class GithubReadTagsBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + + class Output(BlockSchema): + tags: list[dict[str, str]] = SchemaField( + description="List of tags with their names and URLs" + ) + error: str = SchemaField(description="Error message if listing tags failed") + + def __init__(self): + super().__init__( + id="0007g3h4-5678-90ab-1234-567890abcdef", + description="This block lists all tags for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadTagsBlock.Input, + output_schema=GithubReadTagsBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[ + ( + "tags", + [ + { + "name": "v1.0.0", + "url": "https://github.com/owner/repo/tree/v1.0.0", + } + ], + ) + ], + test_mock={ + "list_tags": lambda *args, **kwargs: [ + { + "name": "v1.0.0", + "url": "https://github.com/owner/repo/tree/v1.0.0", + } + ] + }, + ) + + @staticmethod + def list_tags( + credentials: GithubCredentials, repo_url: str + ) -> list[dict[str, str]]: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/tags" + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + tags = [ + { + "name": tag["name"], + "url": f"https://github.com/{repo_path}/tree/{tag['name']}", + } + for tag in data + ] + + return tags + except Exception as e: + return [{"name": "Error", "url": f"Failed to list tags: {str(e)}"}] + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + tags = self.list_tags( + credentials, + input_data.repo_url, + ) + if any("Failed" in tag["url"] for tag in tags): + yield "error", tags[0]["url"] + else: + yield "tags", tags + + +class GithubReadBranchesBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + + class Output(BlockSchema): + branches: list[dict[str, str]] = SchemaField( + description="List of branches with their names and URLs" + ) + error: str = SchemaField(description="Error message if listing branches failed") + + def __init__(self): + super().__init__( + id="0008i3j4-5678-90ab-1234-567890abcdef", + description="This block lists all branches for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadBranchesBlock.Input, + output_schema=GithubReadBranchesBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[ + ( + "branches", + [ + { + "name": "main", + "url": "https://github.com/owner/repo/tree/main", + } + ], + ) + ], + test_mock={ + "list_branches": lambda *args, **kwargs: [ + { + "name": "main", + "url": "https://github.com/owner/repo/tree/main", + } + ] + }, + ) + + @staticmethod + def list_branches( + credentials: GithubCredentials, repo_url: str + ) -> list[dict[str, str]]: + try: + api_url = ( + repo_url.replace("github.com", "api.github.com/repos") + "/branches" + ) + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + branches = [ + {"name": branch["name"], "url": branch["commit"]["url"]} + for branch in data + ] + + return branches + except Exception as e: + return [{"name": "Error", "url": f"Failed to list branches: {str(e)}"}] + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + branches = self.list_branches( + credentials, + input_data.repo_url, + ) + if any("Failed" in branch["url"] for branch in branches): + yield "error", branches[0]["url"] + else: + yield "branches", branches + + +class GithubReadDiscussionsBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + num_discussions: int = SchemaField( + description="Number of discussions to fetch", default=5 + ) + + class Output(BlockSchema): + discussions: list[dict[str, str]] = SchemaField( + description="List of discussions with their titles and URLs" + ) + error: str = SchemaField( + description="Error message if listing discussions failed" + ) + + def __init__(self): + super().__init__( + id="0009j3k4-5678-90ab-1234-567890abcdef", + description="This block lists recent discussions for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadDiscussionsBlock.Input, + output_schema=GithubReadDiscussionsBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "num_discussions": 3, + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[ + ( + "discussions", + [ + { + "title": "Discussion 1", + "url": "https://github.com/owner/repo/discussions/1", + } + ], + ) + ], + test_mock={ + "list_discussions": lambda *args, **kwargs: [ + { + "title": "Discussion 1", + "url": "https://github.com/owner/repo/discussions/1", + } + ] + }, + ) + + @staticmethod + def list_discussions( + credentials: GithubCredentials, repo_url: str, num_discussions: int + ) -> list[dict[str, str]]: + try: + repo_path = repo_url.replace("https://github.com/", "") + owner, repo = repo_path.split("/") + query = """ + query($owner: String!, $repo: String!, $num: Int!) { + repository(owner: $owner, name: $repo) { + discussions(first: $num) { + nodes { + title + url + } + } + } + } + """ + variables = {"owner": owner, "repo": repo, "num": num_discussions} + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.post( + "https://api.github.com/graphql", + json={"query": query, "variables": variables}, + headers=headers, + ) + response.raise_for_status() + + data = response.json() + discussions = [ + {"title": discussion["title"], "url": discussion["url"]} + for discussion in data["data"]["repository"]["discussions"]["nodes"] + ] + + return discussions + except Exception as e: + return [{"title": "Error", "url": f"Failed to list discussions: {str(e)}"}] + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + discussions = self.list_discussions( + credentials, input_data.repo_url, input_data.num_discussions + ) + if any("Failed" in discussion["url"] for discussion in discussions): + yield "error", discussions[0]["url"] + else: + yield "discussions", discussions + + +class GithubReadReleasesBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + + class Output(BlockSchema): + releases: list[dict[str, str]] = SchemaField( + description="List of releases with their names and URLs" + ) + error: str = SchemaField(description="Error message if listing releases failed") + + def __init__(self): + super().__init__( + id="0010k3l4-5678-90ab-1234-567890abcdef", + description="This block lists all releases for a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadReleasesBlock.Input, + output_schema=GithubReadReleasesBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[ + ( + "releases", + [ + { + "name": "v1.0.0", + "url": "https://github.com/owner/repo/releases/tag/v1.0.0", + } + ], + ) + ], + test_mock={ + "list_releases": lambda *args, **kwargs: [ + { + "name": "v1.0.0", + "url": "https://github.com/owner/repo/releases/tag/v1.0.0", + } + ] + }, + ) + + @staticmethod + def list_releases( + credentials: GithubCredentials, repo_url: str + ) -> list[dict[str, str]]: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/releases" + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + data = response.json() + releases = [ + {"name": release["name"], "url": release["html_url"]} + for release in data + ] + + return releases + except Exception as e: + return [{"name": "Error", "url": f"Failed to list releases: {str(e)}"}] + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + releases = self.list_releases( + credentials, + input_data.repo_url, + ) + if any("Failed" in release["url"] for release in releases): + yield "error", releases[0]["url"] + else: + yield "releases", releases + + +class GithubReadCodeownersFileBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + + class Output(BlockSchema): + codeowners_content: str = SchemaField( + description="Content of the CODEOWNERS file" + ) + error: str = SchemaField(description="Error message if the file reading failed") + + def __init__(self): + super().__init__( + id="0006r3s7-8901-23de-4567-890123defghi", + description="This block reads the CODEOWNERS file from the master branch of a specified GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadCodeownersFileBlock.Input, + output_schema=GithubReadCodeownersFileBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("codeowners_content", "# CODEOWNERS content")], + test_mock={ + "read_codeowners": lambda *args, **kwargs: "# CODEOWNERS content" + }, + ) + + @staticmethod + def read_codeowners(credentials: GithubCredentials, repo_url: str) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/contents/.github/CODEOWNERS?ref=master" + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + content = response.json() + return base64.b64decode(content["content"]).decode("utf-8") + except requests.exceptions.HTTPError as http_err: + return f"Failed to read CODEOWNERS file: {str(http_err)}" + except Exception as e: + return f"Failed to read CODEOWNERS file: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + content = self.read_codeowners( + credentials, + input_data.repo_url, + ) + if "Failed" not in content: + yield "codeowners_content", content + else: + yield "error", content + + +class GithubReadFileFromMasterBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + file_path: str = SchemaField( + description="Path to the file in the repository", + placeholder="path/to/file", + ) + + class Output(BlockSchema): + file_content: str = SchemaField( + description="Content of the file from the master branch" + ) + error: str = SchemaField(description="Error message if the file reading failed") + + def __init__(self): + super().__init__( + id="0007r3s8-9012-34ef-5678-901234efghij", + description="This block reads the content of a specified file from the master branch of a GitHub repository using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadFileFromMasterBlock.Input, + output_schema=GithubReadFileFromMasterBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "file_path": "path/to/file", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("file_content", "File content")], + test_mock={"read_file": lambda *args, **kwargs: "File content"}, + ) + + @staticmethod + def read_file(credentials: GithubCredentials, repo_url: str, file_path: str) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = f"https://api.github.com/repos/{repo_path}/contents/{file_path}?ref=master" + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + content = response.json() + return base64.b64decode(content["content"]).decode("utf-8") + except requests.exceptions.HTTPError as http_err: + return f"Failed to read file: {str(http_err)}" + except Exception as e: + return f"Failed to read file: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + content = self.read_file( + credentials, + input_data.repo_url, + input_data.file_path, + ) + if "Failed" not in content: + yield "file_content", content + else: + yield "error", content + + +class GithubReadFileFolderRepoBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + path: str = SchemaField( + description="Path to the file/folder in the repository", + placeholder="path/to/file_or_folder", + ) + branch: str = SchemaField( + description="Branch name to read from", + placeholder="branch_name", + ) + + class Output(BlockSchema): + content: str = SchemaField( + description="Content of the file/folder/repo from the specified branch" + ) + error: str = SchemaField(description="Error message if the reading failed") + + def __init__(self): + super().__init__( + id="0008r3s9-0123-45fg-6789-012345fghijk", + description="This block reads the content of a specified file, folder, or repository from a specified branch using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubReadFileFolderRepoBlock.Input, + output_schema=GithubReadFileFolderRepoBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "path": "path/to/file_or_folder", + "branch": "branch_name", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("content", "File or folder content")], + test_mock={ + "read_content": lambda *args, **kwargs: "File or folder content" + }, + ) + + @staticmethod + def read_content( + credentials: GithubCredentials, repo_url: str, path: str, branch: str + ) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = ( + f"https://api.github.com/repos/{repo_path}/contents/{path}?ref={branch}" + ) + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(api_url, headers=headers) + response.raise_for_status() + + content = response.json() + if "content" in content: + return base64.b64decode(content["content"]).decode("utf-8") + else: + return content # Return the directory content as JSON + + except requests.exceptions.HTTPError as http_err: + return f"Failed to read content: {str(http_err)}" + except Exception as e: + return f"Failed to read content: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + content = self.read_content( + credentials, + input_data.repo_url, + input_data.path, + input_data.branch, + ) + if "Failed" not in content: + yield "content", content + else: + yield "error", content + + +class GithubMakeBranchBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + new_branch: str = SchemaField( + description="Name of the new branch", + placeholder="new_branch_name", + ) + source_branch: str = SchemaField( + description="Name of the source branch", + placeholder="source_branch_name", + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the branch creation operation") + error: str = SchemaField( + description="Error message if the branch creation failed" + ) + + def __init__(self): + super().__init__( + id="0008r3s9-0123-45fg-6789-012345fghijp", + description="This block creates a new branch from a specified source branch using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubMakeBranchBlock.Input, + output_schema=GithubMakeBranchBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "new_branch": "new_branch_name", + "source_branch": "source_branch_name", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Branch created successfully")], + test_mock={ + "create_branch": lambda *args, **kwargs: "Branch created successfully" + }, + ) + + @staticmethod + def create_branch( + credentials: GithubCredentials, + repo_url: str, + new_branch: str, + source_branch: str, + ) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + ref_api_url = f"https://api.github.com/repos/{repo_path}/git/refs/heads/{source_branch}" + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.get(ref_api_url, headers=headers) + response.raise_for_status() + + sha = response.json()["object"]["sha"] + + create_branch_api_url = f"https://api.github.com/repos/{repo_path}/git/refs" + data = {"ref": f"refs/heads/{new_branch}", "sha": sha} + + response = requests.post(create_branch_api_url, headers=headers, json=data) + response.raise_for_status() + + return "Branch created successfully" + except requests.exceptions.HTTPError as http_err: + return f"Failed to create branch: {str(http_err)}" + except Exception as e: + return f"Failed to create branch: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.create_branch( + credentials, + input_data.repo_url, + input_data.new_branch, + input_data.source_branch, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status + + +class GithubDeleteBranchBlock(Block): + class Input(BlockSchema): + credentials: GithubCredentialsInput = GithubCredentialsField("repo") + repo_url: str = SchemaField( + description="URL of the GitHub repository", + placeholder="https://github.com/owner/repo", + ) + branch: str = SchemaField( + description="Name of the branch to delete", + placeholder="branch_name", + ) + + class Output(BlockSchema): + status: str = SchemaField(description="Status of the branch deletion operation") + error: str = SchemaField( + description="Error message if the branch deletion failed" + ) + + def __init__(self): + super().__init__( + id="0008r3s9-0123-45fg-6789-012345fghijq", + description="This block deletes a specified branch using OAuth credentials.", + categories={BlockCategory.DEVELOPER_TOOLS}, + input_schema=GithubDeleteBranchBlock.Input, + output_schema=GithubDeleteBranchBlock.Output, + test_input={ + "repo_url": "https://github.com/owner/repo", + "branch": "branch_name", + "credentials": TEST_CREDENTIALS_INPUT, + }, + test_credentials=TEST_CREDENTIALS, + test_output=[("status", "Branch deleted successfully")], + test_mock={ + "delete_branch": lambda *args, **kwargs: "Branch deleted successfully" + }, + ) + + @staticmethod + def delete_branch( + credentials: GithubCredentials, repo_url: str, branch: str + ) -> str: + try: + repo_path = repo_url.replace("https://github.com/", "") + api_url = ( + f"https://api.github.com/repos/{repo_path}/git/refs/heads/{branch}" + ) + headers = { + "Authorization": credentials.bearer(), + "Accept": "application/vnd.github.v3+json", + } + + response = requests.delete(api_url, headers=headers) + response.raise_for_status() + + return "Branch deleted successfully" + except requests.exceptions.HTTPError as http_err: + return f"Failed to delete branch: {str(http_err)}" + except Exception as e: + return f"Failed to delete branch: {str(e)}" + + def run( + self, + input_data: Input, + *, + credentials: GithubCredentials, + **kwargs, + ) -> BlockOutput: + status = self.delete_branch( + credentials, + input_data.repo_url, + input_data.branch, + ) + if "successfully" in status: + yield "status", status + else: + yield "error", status diff --git a/docs/content/server/new_blocks.md b/docs/content/server/new_blocks.md index 7b46dcbca797..956d241ed7da 100644 --- a/docs/content/server/new_blocks.md +++ b/docs/content/server/new_blocks.md @@ -241,7 +241,7 @@ Aside from implementing the `OAuthHandler` itself, adding a handler into the sys [`BaseOAuthHandler`]: https://github.com/Significant-Gravitas/AutoGPT/blob/master/autogpt_platform/backend/backend/integrations/oauth/base.py #### Example: GitHub integration -- GitHub blocks with API key + OAuth2 support: [`blocks/github.py`](https://github.com/Significant-Gravitas/AutoGPT/blob/master/autogpt_platform/backend/backend/blocks/github.py) +- GitHub blocks with API key + OAuth2 support: [`blocks/github`](https://github.com/Significant-Gravitas/AutoGPT/tree/master/autogpt_platform/backend/backend/blocks/github/) - GitHub OAuth2 handler: [`integrations/oauth/github.py`](https://github.com/Significant-Gravitas/AutoGPT/blob/master/autogpt_platform/backend/backend/integrations/oauth/github.py) ## Key Points to Remember From 3ee7939bf6d4311b6f26e3285f4844323020c80d Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Mon, 23 Sep 2024 15:42:18 +0200 Subject: [PATCH 23/43] regenerate GitHub block IDs --- .../backend/backend/blocks/github/issues.py | 16 ++++++++-------- .../backend/blocks/github/pull_requests.py | 10 +++++----- .../backend/backend/blocks/github/repo.py | 18 +++++++++--------- 3 files changed, 22 insertions(+), 22 deletions(-) diff --git a/autogpt_platform/backend/backend/blocks/github/issues.py b/autogpt_platform/backend/backend/blocks/github/issues.py index 97552a307287..255a1bee0f0c 100644 --- a/autogpt_platform/backend/backend/blocks/github/issues.py +++ b/autogpt_platform/backend/backend/blocks/github/issues.py @@ -32,7 +32,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0001c3d4-5678-90ef-1234-567890abcdef", + id="a8db4d8d-db1c-4a25-a1b0-416a8c33602b", description="This block posts a comment on a specified GitHub issue or pull request using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubCommentBlock.Input, @@ -120,7 +120,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0002d3e4-5678-90ab-1234-567890abcdef", + id="691dad47-f494-44c3-a1e8-05b7990f2dab", description="This block creates a new issue on a specified GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubMakeIssueBlock.Input, @@ -194,7 +194,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0004e3f4-5678-90ab-1234-567890abcdef", + id="6443c75d-032a-4772-9c08-230c707c8acc", description="This block reads the body, title, and user of a specified GitHub issue using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadIssueBlock.Input, @@ -277,7 +277,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0006h3i4-5678-90ab-1234-567890abcdef", + id="c215bfd7-0e57-4573-8f8c-f7d4963dcd74", description="This block lists all issues for a specified GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubListIssuesBlock.Input, @@ -368,7 +368,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0011l3m4-5678-90ab-1234-567890abcdef", + id="98bd6b77-9506-43d5-b669-6b9733c4b1f1", description="This block adds a label to a specified GitHub issue or pull request using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubAddLabelBlock.Input, @@ -455,7 +455,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0012m3n4-5678-90ab-1234-567890abcdef", + id="78f050c5-3e3a-48c0-9e5b-ef1ceca5589c", description="This block removes a label from a specified GitHub issue or pull request using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubRemoveLabelBlock.Input, @@ -546,7 +546,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0004r3s5-6789-01bc-2345-678901bcdefg", + id="90507c72-b0ff-413a-886a-23bbbd66f542", description="This block assigns a user to a specified GitHub issue using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubAssignIssueBlock.Input, @@ -631,7 +631,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0005r3s6-7890-12cd-3456-789012cdefgh", + id="d154002a-38f4-46c2-962d-2488f2b05ece", description="This block unassigns a user from a specified GitHub issue using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubUnassignIssueBlock.Input, diff --git a/autogpt_platform/backend/backend/blocks/github/pull_requests.py b/autogpt_platform/backend/backend/blocks/github/pull_requests.py index f4448ba2ded1..c6bcee001150 100644 --- a/autogpt_platform/backend/backend/blocks/github/pull_requests.py +++ b/autogpt_platform/backend/backend/blocks/github/pull_requests.py @@ -46,7 +46,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0003q3r4-5678-90ab-1234-567890abcdef", + id="dfb987f8-f197-4b2e-bf19-111812afd692", description="This block creates a new pull request on a specified GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubMakePRBlock.Input, @@ -141,7 +141,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0005g3h4-5678-90ab-1234-567890abcdeg", + id="bf94b2a4-1a30-4600-a783-a8a44ee31301", description="This block reads the body, title, user, and changes of a specified GitHub pull request using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadPRBlock.Input, @@ -275,7 +275,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0014o3p4-5678-90ab-1234-567890abcdef", + id="c0d22c5e-e688-43e3-ba43-d5faba7927fd", description="This block assigns a reviewer to a specified GitHub pull request using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubAssignReviewerBlock.Input, @@ -369,7 +369,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0015p3q4-5678-90ab-1234-567890abcdef", + id="9637945d-c602-4875-899a-9c22f8fd30de", description="This block unassigns a reviewer from a specified GitHub pull request using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubUnassignReviewerBlock.Input, @@ -446,7 +446,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0016q3r4-5678-90ab-1234-567890abcdef", + id="2646956e-96d5-4754-a3df-034017e7ed96", description="This block lists all reviewers for a specified GitHub pull request using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubListReviewersBlock.Input, diff --git a/autogpt_platform/backend/backend/blocks/github/repo.py b/autogpt_platform/backend/backend/blocks/github/repo.py index 3679849e731b..dd72cf8da2b4 100644 --- a/autogpt_platform/backend/backend/blocks/github/repo.py +++ b/autogpt_platform/backend/backend/blocks/github/repo.py @@ -30,7 +30,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0007g3h4-5678-90ab-1234-567890abcdef", + id="358924e7-9a11-4d1a-a0f2-13c67fe59e2e", description="This block lists all tags for a specified GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadTagsBlock.Input, @@ -122,7 +122,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0008i3j4-5678-90ab-1234-567890abcdef", + id="74243e49-2bec-4916-8bf4-db43d44aead5", description="This block lists all branches for a specified GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadBranchesBlock.Input, @@ -217,7 +217,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0009j3k4-5678-90ab-1234-567890abcdef", + id="3ef1a419-3d76-4e07-b761-de9dad4d51d7", description="This block lists recent discussions for a specified GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadDiscussionsBlock.Input, @@ -323,7 +323,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0010k3l4-5678-90ab-1234-567890abcdef", + id="3460367a-6ba7-4645-8ce6-47b05d040b92", description="This block lists all releases for a specified GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadReleasesBlock.Input, @@ -412,7 +412,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0006r3s7-8901-23de-4567-890123defghi", + id="ea64bb61-30c0-4693-9273-04081a8f955b", description="This block reads the CODEOWNERS file from the master branch of a specified GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadCodeownersFileBlock.Input, @@ -485,7 +485,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0007r3s8-9012-34ef-5678-901234efghij", + id="87ce6c27-5752-4bbc-8e26-6da40a3dcfd3", description="This block reads the content of a specified file from the master branch of a GitHub repository using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadFileFromMasterBlock.Input, @@ -562,7 +562,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0008r3s9-0123-45fg-6789-012345fghijk", + id="1355f863-2db3-4d75-9fba-f91e8a8ca400", description="This block reads the content of a specified file, folder, or repository from a specified branch using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubReadFileFolderRepoBlock.Input, @@ -651,7 +651,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0008r3s9-0123-45fg-6789-012345fghijp", + id="944cc076-95e7-4d1b-b6b6-b15d8ee5448d", description="This block creates a new branch from a specified source branch using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubMakeBranchBlock.Input, @@ -740,7 +740,7 @@ class Output(BlockSchema): def __init__(self): super().__init__( - id="0008r3s9-0123-45fg-6789-012345fghijq", + id="0d4130f7-e0ab-4d55-adc3-0a40225e80f4", description="This block deletes a specified branch using OAuth credentials.", categories={BlockCategory.DEVELOPER_TOOLS}, input_schema=GithubDeleteBranchBlock.Input, From a2c1f828c5be60cef4f0b19476de681ca36f8253 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Mon, 23 Sep 2024 16:05:10 +0200 Subject: [PATCH 24/43] fix API key creation issue --- .../backend/backend/server/routers/integrations.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/autogpt_platform/backend/backend/server/routers/integrations.py b/autogpt_platform/backend/backend/server/routers/integrations.py index ad97db1e3f6e..bcf03fcb4062 100644 --- a/autogpt_platform/backend/backend/server/routers/integrations.py +++ b/autogpt_platform/backend/backend/server/routers/integrations.py @@ -141,14 +141,14 @@ async def get_credential( @router.post("/{provider}/credentials", status_code=201) async def create_api_key_credentials( + store: Annotated[SupabaseIntegrationCredentialsStore, Depends(get_store)], + user_id: Annotated[str, Depends(get_user_id)], provider: Annotated[str, Path(title="The provider to create credentials for")], api_key: Annotated[str, Body(title="The API key to store")], title: Annotated[str, Body(title="Optional title for the credentials")], expires_at: Annotated[ int | None, Body(title="Unix timestamp when the key expires") - ], - user_id: Annotated[str, Depends(get_user_id)], - store: Annotated[SupabaseIntegrationCredentialsStore, Depends(get_store)], + ] = None, ): new_credentials = APIKeyCredentials( provider=provider, From 125b1f3f50c5db96330d881080b029b6655031ed Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Tue, 24 Sep 2024 15:48:06 +0200 Subject: [PATCH 25/43] fix credentials picker API key display issue --- .../frontend/src/components/integrations/credentials-input.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/autogpt_platform/frontend/src/components/integrations/credentials-input.tsx b/autogpt_platform/frontend/src/components/integrations/credentials-input.tsx index c4376314d87b..0cdfc0762562 100644 --- a/autogpt_platform/frontend/src/components/integrations/credentials-input.tsx +++ b/autogpt_platform/frontend/src/components/integrations/credentials-input.tsx @@ -235,7 +235,7 @@ export const CredentialsInput: FC<{ - {credentials.username} + {credentials.title} ))} From 2e3c14099636f779602e1425c49e6b2129d77588 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Tue, 24 Sep 2024 15:48:37 +0200 Subject: [PATCH 26/43] fix API key creation reponse type --- .../backend/backend/server/routers/integrations.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/autogpt_platform/backend/backend/server/routers/integrations.py b/autogpt_platform/backend/backend/server/routers/integrations.py index bcf03fcb4062..5f0fa411f032 100644 --- a/autogpt_platform/backend/backend/server/routers/integrations.py +++ b/autogpt_platform/backend/backend/server/routers/integrations.py @@ -149,7 +149,7 @@ async def create_api_key_credentials( expires_at: Annotated[ int | None, Body(title="Unix timestamp when the key expires") ] = None, -): +) -> APIKeyCredentials: new_credentials = APIKeyCredentials( provider=provider, api_key=SecretStr(api_key), @@ -163,7 +163,7 @@ async def create_api_key_credentials( raise HTTPException( status_code=500, detail=f"Failed to store credentials: {str(e)}" ) - return {"id": new_credentials.id} + return new_credentials @router.delete("/{provider}/credentials/{cred_id}", status_code=204) From b58f9a8710ca2789a6aa4705202b6b1392bdea74 Mon Sep 17 00:00:00 2001 From: Reinier van der Leer Date: Tue, 24 Sep 2024 15:57:03 +0200 Subject: [PATCH 27/43] fix credentials picker updating after creating API key --- .../frontend/src/components/integrations/credentials-input.tsx | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/autogpt_platform/frontend/src/components/integrations/credentials-input.tsx b/autogpt_platform/frontend/src/components/integrations/credentials-input.tsx index 0cdfc0762562..412dd5d0e72d 100644 --- a/autogpt_platform/frontend/src/components/integrations/credentials-input.tsx +++ b/autogpt_platform/frontend/src/components/integrations/credentials-input.tsx @@ -218,7 +218,7 @@ export const CredentialsInput: FC<{ return ( <> +