+
+ Demande créée !
+ +
+ Votre demande a été créée et est disponible à l'adresse suivante:
+
+ { vmodel.IssueURL }
.
+ diff --git a/go.mod b/go.mod index a34bb35..c9f3493 100644 --- a/go.mod +++ b/go.mod @@ -6,7 +6,7 @@ toolchain go1.23.6 require ( code.gitea.io/sdk/gitea v0.20.0 - github.com/a-h/templ v0.3.819 + github.com/a-h/templ v0.3.833 github.com/bornholm/genai v0.0.0-20250222092500-1076426da67c github.com/caarlos0/env/v11 v11.2.2 github.com/davecgh/go-spew v1.1.1 @@ -45,7 +45,7 @@ require ( golang.org/x/mod v0.20.0 // indirect golang.org/x/net v0.35.0 // indirect golang.org/x/oauth2 v0.17.0 // indirect - golang.org/x/sync v0.10.0 // indirect + golang.org/x/sync v0.11.0 // indirect golang.org/x/sys v0.30.0 // indirect golang.org/x/tools v0.24.0 // indirect google.golang.org/appengine v1.6.8 // indirect diff --git a/go.sum b/go.sum index 73300e2..9f86101 100644 --- a/go.sum +++ b/go.sum @@ -12,6 +12,8 @@ github.com/RealAlexandreAI/json-repair v0.0.14 h1:4kTqotVonDVTio5n2yweRUELVcNe2x github.com/RealAlexandreAI/json-repair v0.0.14/go.mod h1:GKJi5borR78O8c7HCVbgqjhoiVibZ6hJldxbc6dGrAI= github.com/a-h/templ v0.3.819 h1:KDJ5jTFN15FyJnmSmo2gNirIqt7hfvBD2VXVDTySckM= github.com/a-h/templ v0.3.819/go.mod h1:iDJKJktpttVKdWoTkRNNLcllRI+BlpopJc+8au3gOUo= +github.com/a-h/templ v0.3.833 h1:L/KOk/0VvVTBegtE0fp2RJQiBm7/52Zxv5fqlEHiQUU= +github.com/a-h/templ v0.3.833/go.mod h1:cAu4AiZhtJfBjMY0HASlyzvkrtjnHWPeEsyGK2YYmfk= github.com/bornholm/genai v0.0.0-20250222092500-1076426da67c h1:bI0ebsgO1/7Jx6+ZQdDF/I6tTZxyB5hODYz7x/XxwK8= github.com/bornholm/genai v0.0.0-20250222092500-1076426da67c/go.mod h1:MnuvwSsBEWv/joeK/WgUyfZfOLcLTpd81NJdWoRpRfI= github.com/caarlos0/env/v11 v11.2.2 h1:95fApNrUyueipoZN/EhA8mMxiNxrBwDa+oAZrMWl3Kg= @@ -95,8 +97,8 @@ golang.org/x/oauth2 v0.17.0 h1:6m3ZPmLEFdVxKKWnKq4VqZ60gutO35zm+zrAHVmHyDQ= golang.org/x/oauth2 v0.17.0/go.mod h1:OzPDGQiuQMguemayvdylqddI7qcD9lnSDb+1FiwQ5HA= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ= -golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= +golang.org/x/sync v0.11.0 h1:GGz8+XQP4FvTTrjZPzNKTMFtSXH80RAzG+5ghFPgK9w= +golang.org/x/sync v0.11.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= diff --git a/internal/adapter/gitea/forge.go b/internal/adapter/gitea/forge.go index 269675b..39f3277 100644 --- a/internal/adapter/gitea/forge.go +++ b/internal/adapter/gitea/forge.go @@ -18,8 +18,26 @@ type Forge struct { } // CreateIssue implements port.Forge. -func (f *Forge) CreateIssue(ctx context.Context, projectID string, title string, content string) error { - return nil +func (f *Forge) CreateIssue(ctx context.Context, rawProjectID string, title string, body string) (string, error) { + projectID, err := strconv.ParseInt(rawProjectID, 10, 64) + if err != nil { + return "", errors.WithStack(err) + } + + project, _, err := f.client.GetRepoByID(projectID) + if err != nil { + return "", errors.WithStack(err) + } + + issue, _, err := f.client.CreateIssue(project.Owner.UserName, project.Name, gitea.CreateIssueOption{ + Title: title, + Body: body, + }) + if err != nil { + return "", errors.WithStack(err) + } + + return issue.HTMLURL, nil } // GetIssueTemplate implements port.Forge. diff --git a/internal/core/port/forge.go b/internal/core/port/forge.go index cd1c3d6..2660283 100644 --- a/internal/core/port/forge.go +++ b/internal/core/port/forge.go @@ -13,7 +13,7 @@ var ( type Forge interface { GetProjects(ctx context.Context) ([]*model.Project, error) - CreateIssue(ctx context.Context, projectID string, title string, content string) error + CreateIssue(ctx context.Context, projectID string, title string, body string) (string, error) GetIssues(ctx context.Context, projectID string, issueIDs ...string) ([]*model.Issue, error) GetIssueTemplate(ctx context.Context, projectID string) (string, error) } diff --git a/internal/core/service/issue_manager.go b/internal/core/service/issue_manager.go index 606fee6..97946e6 100644 --- a/internal/core/service/issue_manager.go +++ b/internal/core/service/issue_manager.go @@ -4,6 +4,7 @@ import ( "context" "fmt" "log/slog" + "strings" "time" _ "embed" @@ -39,6 +40,20 @@ type IssueManager struct { projectCache *cache.Cache[[]*model.Project] } +func (m *IssueManager) CreateIssue(ctx context.Context, user *model.User, projectID string, title string, body string) (string, error) { + forge, err := m.getUserForge(ctx, user) + if err != nil { + return "", errors.WithStack(err) + } + + issueURL, err := forge.CreateIssue(ctx, projectID, title, body) + if err != nil { + return "", errors.WithStack(err) + } + + return issueURL, nil +} + func (m *IssueManager) GetUserProjects(ctx context.Context, user *model.User) ([]*model.Project, error) { cacheKey := fmt.Sprintf("%s/%s", user.Provider, user.ID) @@ -62,15 +77,15 @@ func (m *IssueManager) GetUserProjects(ctx context.Context, user *model.User) ([ return projects, nil } -func (m *IssueManager) GenerateIssue(ctx context.Context, user *model.User, projectID string, issueSummary string) (string, error) { +func (m *IssueManager) GenerateIssue(ctx context.Context, user *model.User, projectID string, issueSummary string) (string, string, error) { systemPrompt, err := m.getIssueSystemPrompt(ctx, user, projectID) if err != nil { - return "", errors.WithStack(err) + return "", "", errors.WithStack(err) } userPrompt, err := m.getIssueUserPrompt(ctx, user, projectID, issueSummary) if err != nil { - return "", errors.WithStack(err) + return "", "", errors.WithStack(err) } messages := []llm.Message{ @@ -80,10 +95,22 @@ func (m *IssueManager) GenerateIssue(ctx context.Context, user *model.User, proj res, err := m.llmClient.ChatCompletion(ctx, llm.WithMessages(messages...)) if err != nil { - return "", errors.WithStack(err) + return "", "", errors.WithStack(err) } - return res.Message().Content(), nil + body := res.Message().Content() + + messages = append(messages, res.Message()) + messages = append(messages, llm.NewMessage(llm.RoleUser, "Generate a title for this issue. Keep it descriptive, simple and short. Do not write anything else.")) + + res, err = m.llmClient.ChatCompletion(ctx, llm.WithMessages(messages...)) + if err != nil { + return "", "", errors.WithStack(err) + } + + title := toTitle(res.Message().Content()) + + return title, body, nil } func (m *IssueManager) getIssueSystemPrompt(ctx context.Context, user *model.User, projectID string) (string, error) { @@ -156,3 +183,8 @@ func NewIssueManager(llmClient llm.Client, forgeFactories ...ForgeFactory) *Issu projectCache: cache.New[[]*model.Project](time.Minute*5, (time.Minute*5)/2), } } + +func toTitle(str string) string { + str = strings.ToLower(str) + return strings.ToUpper(string(str[0])) + str[1:] +} diff --git a/internal/core/service/issue_system_prompt.gotmpl b/internal/core/service/issue_system_prompt.gotmpl index 6f26502..0eb8269 100644 --- a/internal/core/service/issue_system_prompt.gotmpl +++ b/internal/core/service/issue_system_prompt.gotmpl @@ -9,6 +9,7 @@ You are an expert software developer with extensive experience in writing clear - Expected behavior. - Actual behavior. - Any relevant error messages or logs. + - Always use the user prompt context main language. 2. **Additional Context**: - Include any other relevant information that might help in understanding or resolving the issue/request. diff --git a/internal/http/handler/webui/issue/component/issue_page.templ b/internal/http/handler/webui/issue/component/issue_page.templ index 876e015..004342b 100644 --- a/internal/http/handler/webui/issue/component/issue_page.templ +++ b/internal/http/handler/webui/issue/component/issue_page.templ @@ -7,6 +7,7 @@ import ( ) type IssuePageVModel struct { + IssueURL string SummaryForm *form.Form IssueForm *form.Form Projects []*model.Project @@ -42,7 +43,7 @@ func NewIssueForm() *form.Form { form.NonEmpty("Ce champs ne doit pas être vide."), ), form.NewField( - "content", + "body", form.Attrs{ "type": "textarea", "rows": "20", @@ -59,11 +60,33 @@ templ IssuePage(vmodel IssuePageVModel) {
+ if vmodel.IssueURL != "" { + + + @templ.JSFuncCall("clearSummary", vmodel.SelectedProjectID) + @templ.JSFuncCall("openIssue", vmodel.IssueURL) + } +