3f06ba2ea6
PR Ready Gate / clear-labels (pull_request) Successful in 2s
CI / test (pull_request) Successful in 9m32s
CI / review (anthropic--claude-4.6-sonnet, sonnet, SONNET_REVIEW_TOKEN) (pull_request) Successful in 10m10s
CI / review (gpt-5, gpt, GPT_REVIEW_TOKEN) (pull_request) Successful in 10m51s
CI / review (gpt-5, security, SECURITY_REVIEW.md, SECURITY_REVIEW_TOKEN) (pull_request) Successful in 10m33s
Implements #60. - Add ParsePersonaBytes() for parsing personas from byte data - Add LoadRepoPersonas() to fetch personas from repo via Gitea API - Add MergePersonas() to combine built-in and repo personas - Add GetBuiltinPersonasMap() helper - Update main.go to load repo personas first, fall back to built-in - Add giteaClientAdapter to bridge gitea.Client to review.GiteaClient When --persona is specified, the bot now: 1. Attempts to fetch personas from .review-bot/personas/*.yaml 2. If the named persona exists in the repo, uses it 3. Otherwise falls back to built-in personas This allows repos to define domain-specific personas (e.g., trading experts for gargoyle, crypto experts for kms-lite) without modifying the review-bot codebase.
139 lines
4.1 KiB
Go
139 lines
4.1 KiB
Go
package review
|
|
|
|
import (
|
|
"context"
|
|
"log/slog"
|
|
"strings"
|
|
)
|
|
|
|
// RepoPersonaPath is the directory path where repo-specific personas are stored.
|
|
const RepoPersonaPath = ".review-bot/personas"
|
|
|
|
// GiteaClient defines the subset of gitea.Client methods needed for loading repo personas.
|
|
// This interface allows for easier testing and decouples the review package from gitea.
|
|
type GiteaClient interface {
|
|
ListContents(ctx context.Context, owner, repo, path string) ([]ContentEntry, error)
|
|
GetFileContent(ctx context.Context, owner, repo, filepath string) (string, error)
|
|
}
|
|
|
|
// ContentEntry represents a file or directory entry from the contents API.
|
|
// This mirrors gitea.ContentEntry to avoid import cycles.
|
|
type ContentEntry struct {
|
|
Name string `json:"name"`
|
|
Path string `json:"path"`
|
|
Type string `json:"type"` // "file" or "dir"
|
|
}
|
|
|
|
// LoadRepoPersonas fetches personas from a repository's .review-bot/personas/ directory.
|
|
// Returns an empty map (not nil) if the directory doesn't exist or is empty.
|
|
// Individual parse failures are logged and skipped; the remaining personas are still returned.
|
|
// Auth errors and other non-404 errors are propagated.
|
|
func LoadRepoPersonas(ctx context.Context, client GiteaClient, owner, repo string) (map[string]*Persona, error) {
|
|
result := make(map[string]*Persona)
|
|
|
|
entries, err := client.ListContents(ctx, owner, repo, RepoPersonaPath)
|
|
if err != nil {
|
|
// Check if this is a 404 (directory doesn't exist) - expected case
|
|
if isNotFoundError(err) {
|
|
slog.Debug("no repo personas directory found", "repo", owner+"/"+repo)
|
|
return result, nil
|
|
}
|
|
// Other errors (auth, server) should propagate
|
|
return nil, err
|
|
}
|
|
|
|
if len(entries) == 0 {
|
|
slog.Debug("repo personas directory is empty", "repo", owner+"/"+repo)
|
|
return result, nil
|
|
}
|
|
|
|
for _, entry := range entries {
|
|
if entry.Type != "file" {
|
|
continue
|
|
}
|
|
// Only process YAML files
|
|
if !isYAMLFile(entry.Name) {
|
|
continue
|
|
}
|
|
|
|
content, err := client.GetFileContent(ctx, owner, repo, entry.Path)
|
|
if err != nil {
|
|
slog.Warn("could not fetch repo persona file",
|
|
"file", entry.Path,
|
|
"repo", owner+"/"+repo,
|
|
"error", err)
|
|
continue
|
|
}
|
|
|
|
persona, err := ParsePersonaBytes([]byte(content), entry.Path)
|
|
if err != nil {
|
|
slog.Warn("could not parse repo persona file",
|
|
"file", entry.Path,
|
|
"repo", owner+"/"+repo,
|
|
"error", err)
|
|
continue
|
|
}
|
|
|
|
result[persona.Name] = persona
|
|
slog.Debug("loaded repo persona",
|
|
"name", persona.Name,
|
|
"file", entry.Path,
|
|
"repo", owner+"/"+repo)
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
// MergePersonas combines built-in personas with repo personas.
|
|
// Repo personas take precedence on name collision.
|
|
// Returns a new map; inputs are not modified.
|
|
func MergePersonas(builtin, repo map[string]*Persona) map[string]*Persona {
|
|
result := make(map[string]*Persona, len(builtin)+len(repo))
|
|
|
|
// Copy built-in personas first
|
|
for name, p := range builtin {
|
|
result[name] = p
|
|
}
|
|
|
|
// Overlay repo personas (override on collision)
|
|
for name, p := range repo {
|
|
if _, exists := result[name]; exists {
|
|
slog.Debug("repo persona overrides built-in", "name", name)
|
|
}
|
|
result[name] = p
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// GetBuiltinPersonasMap returns all built-in personas as a map keyed by name.
|
|
// Returns an empty map (not nil) if loading fails.
|
|
func GetBuiltinPersonasMap() map[string]*Persona {
|
|
result := make(map[string]*Persona)
|
|
for _, name := range ListBuiltinPersonas() {
|
|
p, err := LoadBuiltinPersona(name)
|
|
if err != nil {
|
|
slog.Warn("could not load built-in persona", "name", name, "error", err)
|
|
continue
|
|
}
|
|
result[name] = p
|
|
}
|
|
return result
|
|
}
|
|
|
|
// isYAMLFile checks if a filename has a YAML extension.
|
|
func isYAMLFile(name string) bool {
|
|
lower := strings.ToLower(name)
|
|
return strings.HasSuffix(lower, ".yaml") || strings.HasSuffix(lower, ".yml")
|
|
}
|
|
|
|
// isNotFoundError checks if an error represents a 404 response.
|
|
// This is a string-based heuristic since we don't have access to gitea.APIError here.
|
|
func isNotFoundError(err error) bool {
|
|
if err == nil {
|
|
return false
|
|
}
|
|
errStr := err.Error()
|
|
return strings.Contains(errStr, "HTTP 404") || strings.Contains(errStr, "not found")
|
|
}
|