docs: Update adk docs update workflow to invoke the updater agent once per suggestion

Co-authored-by: Xuan Yang <xygoogle@google.com>
PiperOrigin-RevId: 842789476
This commit is contained in:
Xuan Yang
2025-12-10 10:44:30 -08:00
committed by Copybara-Service
parent 51a638b6b8
commit b23deeb9ab
2 changed files with 119 additions and 11 deletions
@@ -24,13 +24,14 @@ from adk_documentation.settings import DOC_OWNER
from adk_documentation.settings import DOC_REPO
from adk_documentation.tools import get_issue
from adk_documentation.utils import call_agent_async
from adk_documentation.utils import parse_suggestions
from google.adk.cli.utils import logs
from google.adk.runners import InMemoryRunner
APP_NAME = "adk_docs_updater"
USER_ID = "adk_docs_updater_user"
logs.setup_adk_logger(level=logging.DEBUG)
logs.setup_adk_logger(level=logging.INFO)
def process_arguments():
@@ -68,23 +69,84 @@ async def main():
print(f"Failed to get issue {issue_number}: {get_issue_response}\n")
return
issue = get_issue_response["issue"]
issue_title = issue.get("title", "")
issue_body = issue.get("body", "")
# Parse numbered suggestions from issue body
suggestions = parse_suggestions(issue_body)
if not suggestions:
print(f"No numbered suggestions found in issue #{issue_number}.")
print("Falling back to processing the entire issue as a single task.")
suggestions = [(1, issue_body)]
print(f"Found {len(suggestions)} suggestion(s) in issue #{issue_number}.")
print("=" * 80)
runner = InMemoryRunner(
agent=agent.root_agent,
app_name=APP_NAME,
)
session = await runner.session_service.create_session(
app_name=APP_NAME,
user_id=USER_ID,
)
response = await call_agent_async(
runner,
USER_ID,
session.id,
f"Please update the ADK docs according to the following issue:\n{issue}",
results = []
for suggestion_num, suggestion_text in suggestions:
print(f"\n>>> Processing suggestion #{suggestion_num}...")
print("-" * 80)
# Create a new session for each suggestion to avoid context interference
session = await runner.session_service.create_session(
app_name=APP_NAME,
user_id=USER_ID,
)
prompt = f"""
Please update the ADK docs according to suggestion #{suggestion_num} from issue #{issue_number}.
Issue title: {issue_title}
Suggestion to process:
{suggestion_text}
Note: Focus only on this specific suggestion. Create exactly one pull request for this suggestion.
"""
try:
response = await call_agent_async(
runner,
USER_ID,
session.id,
prompt,
)
results.append({
"suggestion_num": suggestion_num,
"status": "success",
"response": response,
})
print(f"<<<< Suggestion #{suggestion_num} completed.")
except Exception as e:
results.append({
"suggestion_num": suggestion_num,
"status": "error",
"error": str(e),
})
print(f"<<<< Suggestion #{suggestion_num} failed: {e}")
print("-" * 80)
# Print summary
print("\n" + "=" * 80)
print("SUMMARY")
print("=" * 80)
successful = [r for r in results if r["status"] == "success"]
failed = [r for r in results if r["status"] == "error"]
print(
f"Total: {len(results)}, Success: {len(successful)}, Failed:"
f" {len(failed)}"
)
print(f"<<<< Agent Final Output: {response}\n")
if failed:
print("\nFailed suggestions:")
for r in failed:
print(f" - Suggestion #{r['suggestion_num']}: {r['error']}")
if __name__ == "__main__":
@@ -12,9 +12,11 @@
# See the License for the specific language governing permissions and
# limitations under the License.
import re
from typing import Any
from typing import Dict
from typing import List
from typing import Tuple
from adk_documentation.settings import GITHUB_TOKEN
from google.adk.agents.run_config import RunConfig
@@ -96,3 +98,47 @@ async def call_agent_async(
final_response_text += text
return final_response_text
def parse_suggestions(issue_body: str) -> List[Tuple[int, str]]:
"""Parse numbered suggestions from issue body.
Supports multiple formats:
- Format A (markdown headers): "### 1. Title"
- Format B (numbered list with bold): "1. **Title**"
Args:
issue_body: The body text of the GitHub issue.
Returns:
A list of tuples, where each tuple contains:
- The suggestion number (1-based)
- The full text of that suggestion
"""
# Try different patterns in order of preference
patterns = [
# Format A: "### 1. Title" (markdown header with number)
(r"(?=^###\s+\d+\.)", r"^###\s+(\d+)\."),
# Format B: "1. **Title**" (numbered list with bold)
(r"(?=^\d+\.\s+\*\*)", r"^(\d+)\.\s+\*\*"),
]
for split_pattern, match_pattern in patterns:
parts = re.split(split_pattern, issue_body, flags=re.MULTILINE)
suggestions = []
for part in parts:
part = part.strip()
if not part:
continue
match = re.match(match_pattern, part)
if match:
suggestion_num = int(match.group(1))
suggestions.append((suggestion_num, part))
# If we found suggestions with this pattern, return them
if suggestions:
return suggestions
return []