mirror of
https://github.com/sourcegit-scm/sourcegit.git
synced 2025-01-02 22:27:21 -08:00
Compare commits
No commits in common. "c08c307c7aae24226dfd5481667deddfd86119d1" and "547c28adb852ce2eccf127c6991a86851f93bf67" have entirely different histories.
c08c307c7a
...
547c28adb8
15 changed files with 59 additions and 275 deletions
|
@ -32,36 +32,31 @@ namespace SourceGit.Commands
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var summarybuilder = new StringBuilder();
|
var summaries = new List<string>();
|
||||||
var bodyBuilder = new StringBuilder();
|
|
||||||
foreach (var change in _changes)
|
foreach (var change in _changes)
|
||||||
{
|
{
|
||||||
if (_cancelToken.IsCancellationRequested)
|
if (_cancelToken.IsCancellationRequested)
|
||||||
return "";
|
return "";
|
||||||
|
|
||||||
_onProgress?.Invoke($"Analyzing {change.Path}...");
|
_onProgress?.Invoke($"Analyzing {change.Path}...");
|
||||||
|
|
||||||
var summary = GenerateChangeSummary(change);
|
var summary = GenerateChangeSummary(change);
|
||||||
summarybuilder.Append("- ");
|
summaries.Add(summary);
|
||||||
summarybuilder.Append(summary);
|
|
||||||
summarybuilder.Append("(file: ");
|
|
||||||
summarybuilder.Append(change.Path);
|
|
||||||
summarybuilder.Append(")");
|
|
||||||
summarybuilder.AppendLine();
|
|
||||||
|
|
||||||
bodyBuilder.Append("- ");
|
|
||||||
bodyBuilder.Append(summary);
|
|
||||||
bodyBuilder.AppendLine();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_cancelToken.IsCancellationRequested)
|
if (_cancelToken.IsCancellationRequested)
|
||||||
return "";
|
return "";
|
||||||
|
|
||||||
_onProgress?.Invoke($"Generating commit message...");
|
_onProgress?.Invoke($"Generating commit message...");
|
||||||
|
var builder = new StringBuilder();
|
||||||
|
builder.Append(GenerateSubject(string.Join("", summaries)));
|
||||||
|
builder.Append("\n");
|
||||||
|
foreach (var summary in summaries)
|
||||||
|
{
|
||||||
|
builder.Append("\n- ");
|
||||||
|
builder.Append(summary.Trim());
|
||||||
|
}
|
||||||
|
|
||||||
var body = bodyBuilder.ToString();
|
return builder.ToString();
|
||||||
var subject = GenerateSubject(summarybuilder.ToString());
|
|
||||||
return string.Format("{0}\n\n{1}", subject, body);
|
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
|
@ -75,7 +70,15 @@ namespace SourceGit.Commands
|
||||||
var rs = new GetDiffContent(_repo, new Models.DiffOption(change, false)).ReadToEnd();
|
var rs = new GetDiffContent(_repo, new Models.DiffOption(change, false)).ReadToEnd();
|
||||||
var diff = rs.IsSuccess ? rs.StdOut : "unknown change";
|
var diff = rs.IsSuccess ? rs.StdOut : "unknown change";
|
||||||
|
|
||||||
var rsp = Models.OpenAI.Chat(Models.OpenAI.AnalyzeDiffPrompt, $"Here is the `git diff` output: {diff}", _cancelToken);
|
var prompt = new StringBuilder();
|
||||||
|
prompt.AppendLine("You are an expert developer specialist in creating commits.");
|
||||||
|
prompt.AppendLine("Provide a super concise one sentence overall changes summary of the user `git diff` output following strictly the next rules:");
|
||||||
|
prompt.AppendLine("- Do not use any code snippets, imports, file routes or bullets points.");
|
||||||
|
prompt.AppendLine("- Do not mention the route of file that has been change.");
|
||||||
|
prompt.AppendLine("- Simply describe the MAIN GOAL of the changes.");
|
||||||
|
prompt.AppendLine("- Output directly the summary in plain text.`");
|
||||||
|
|
||||||
|
var rsp = Models.OpenAI.Chat(prompt.ToString(), $"Here is the `git diff` output: {diff}", _cancelToken);
|
||||||
if (rsp != null && rsp.Choices.Count > 0)
|
if (rsp != null && rsp.Choices.Count > 0)
|
||||||
return rsp.Choices[0].Message.Content;
|
return rsp.Choices[0].Message.Content;
|
||||||
|
|
||||||
|
@ -84,7 +87,24 @@ namespace SourceGit.Commands
|
||||||
|
|
||||||
private string GenerateSubject(string summary)
|
private string GenerateSubject(string summary)
|
||||||
{
|
{
|
||||||
var rsp = Models.OpenAI.Chat(Models.OpenAI.GenerateSubjectPrompt, $"Here are the summaries changes:\n{summary}", _cancelToken);
|
var prompt = new StringBuilder();
|
||||||
|
prompt.AppendLine("You are an expert developer specialist in creating commits messages.");
|
||||||
|
prompt.AppendLine("Your only goal is to retrieve a single commit message.");
|
||||||
|
prompt.AppendLine("Based on the provided user changes, combine them in ONE SINGLE commit message retrieving the global idea, following strictly the next rules:");
|
||||||
|
prompt.AppendLine("- Assign the commit {type} according to the next conditions:");
|
||||||
|
prompt.AppendLine(" feat: Only when adding a new feature.");
|
||||||
|
prompt.AppendLine(" fix: When fixing a bug.");
|
||||||
|
prompt.AppendLine(" docs: When updating documentation.");
|
||||||
|
prompt.AppendLine(" style: When changing elements styles or design and/or making changes to the code style (formatting, missing semicolons, etc.) without changing the code logic.");
|
||||||
|
prompt.AppendLine(" test: When adding or updating tests. ");
|
||||||
|
prompt.AppendLine(" chore: When making changes to the build process or auxiliary tools and libraries. ");
|
||||||
|
prompt.AppendLine(" revert: When undoing a previous commit.");
|
||||||
|
prompt.AppendLine(" refactor: When restructuring code without changing its external behavior, or is any of the other refactor types.");
|
||||||
|
prompt.AppendLine("- Do not add any issues numeration, explain your output nor introduce your answer.");
|
||||||
|
prompt.AppendLine("- Output directly only one commit message in plain text with the next format: {type}: {commit_message}.");
|
||||||
|
prompt.AppendLine("- Be as concise as possible, keep the message under 50 characters.");
|
||||||
|
|
||||||
|
var rsp = Models.OpenAI.Chat(prompt.ToString(), $"Here are the summaries changes: {summary}", _cancelToken);
|
||||||
if (rsp != null && rsp.Choices.Count > 0)
|
if (rsp != null && rsp.Choices.Count > 0)
|
||||||
return rsp.Choices[0].Message.Content;
|
return rsp.Choices[0].Message.Content;
|
||||||
|
|
||||||
|
|
|
@ -1,70 +0,0 @@
|
||||||
namespace SourceGit.Models
|
|
||||||
{
|
|
||||||
public static class NumericSort
|
|
||||||
{
|
|
||||||
public static int Compare(string s1, string s2)
|
|
||||||
{
|
|
||||||
int len1 = s1.Length;
|
|
||||||
int len2 = s2.Length;
|
|
||||||
|
|
||||||
int marker1 = 0;
|
|
||||||
int marker2 = 0;
|
|
||||||
|
|
||||||
char[] tmp1 = new char[len1];
|
|
||||||
char[] tmp2 = new char[len2];
|
|
||||||
|
|
||||||
while (marker1 < len1 && marker2 < len2)
|
|
||||||
{
|
|
||||||
char c1 = s1[marker1];
|
|
||||||
char c2 = s2[marker2];
|
|
||||||
int loc1 = 0;
|
|
||||||
int loc2 = 0;
|
|
||||||
|
|
||||||
bool isDigit1 = char.IsDigit(c1);
|
|
||||||
do
|
|
||||||
{
|
|
||||||
tmp1[loc1] = c1;
|
|
||||||
loc1++;
|
|
||||||
marker1++;
|
|
||||||
|
|
||||||
if (marker1 < len1)
|
|
||||||
c1 = s1[marker1];
|
|
||||||
else
|
|
||||||
break;
|
|
||||||
} while (char.IsDigit(c1) == isDigit1);
|
|
||||||
|
|
||||||
bool isDigit2 = char.IsDigit(c2);
|
|
||||||
do
|
|
||||||
{
|
|
||||||
tmp2[loc2] = c2;
|
|
||||||
loc2++;
|
|
||||||
marker2++;
|
|
||||||
|
|
||||||
if (marker2 < len2)
|
|
||||||
c2 = s2[marker2];
|
|
||||||
else
|
|
||||||
break;
|
|
||||||
} while (char.IsDigit(c2) == isDigit2);
|
|
||||||
|
|
||||||
string sub1 = new string(tmp1, 0, loc1);
|
|
||||||
string sub2 = new string(tmp2, 0, loc2);
|
|
||||||
int result;
|
|
||||||
if (isDigit1 && isDigit2)
|
|
||||||
{
|
|
||||||
int num1 = int.Parse(sub1);
|
|
||||||
int num2 = int.Parse(sub2);
|
|
||||||
result = num1 - num2;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
result = string.Compare(sub1, sub2, System.StringComparison.Ordinal);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result != 0)
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
return len1 - len2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -94,18 +94,6 @@ namespace SourceGit.Models
|
||||||
set;
|
set;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static string AnalyzeDiffPrompt
|
|
||||||
{
|
|
||||||
get;
|
|
||||||
set;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static string GenerateSubjectPrompt
|
|
||||||
{
|
|
||||||
get;
|
|
||||||
set;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static bool IsValid
|
public static bool IsValid
|
||||||
{
|
{
|
||||||
get => !string.IsNullOrEmpty(Server) && !string.IsNullOrEmpty(Model);
|
get => !string.IsNullOrEmpty(Server) && !string.IsNullOrEmpty(Model);
|
||||||
|
@ -125,14 +113,14 @@ namespace SourceGit.Models
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var task = client.PostAsync(Server, req, cancellation);
|
var task = client.PostAsync(Server, req, cancellation);
|
||||||
task.Wait(cancellation);
|
task.Wait();
|
||||||
|
|
||||||
var rsp = task.Result;
|
var rsp = task.Result;
|
||||||
if (!rsp.IsSuccessStatusCode)
|
if (!rsp.IsSuccessStatusCode)
|
||||||
throw new Exception($"AI service returns error code {rsp.StatusCode}");
|
throw new Exception($"AI service returns error code {rsp.StatusCode}");
|
||||||
|
|
||||||
var reader = rsp.Content.ReadAsStringAsync(cancellation);
|
var reader = rsp.Content.ReadAsStringAsync(cancellation);
|
||||||
reader.Wait(cancellation);
|
reader.Wait();
|
||||||
|
|
||||||
return JsonSerializer.Deserialize(reader.Result, JsonCodeGen.Default.OpenAIChatResponse);
|
return JsonSerializer.Deserialize(reader.Result, JsonCodeGen.Default.OpenAIChatResponse);
|
||||||
}
|
}
|
||||||
|
|
|
@ -169,32 +169,6 @@ namespace SourceGit.Models
|
||||||
return rule;
|
return rule;
|
||||||
}
|
}
|
||||||
|
|
||||||
public IssueTrackerRule AddGitLabIssueTracker(string repoURL)
|
|
||||||
{
|
|
||||||
var rule = new IssueTrackerRule()
|
|
||||||
{
|
|
||||||
Name = "GitLab ISSUE",
|
|
||||||
RegexString = "#(\\d+)",
|
|
||||||
URLTemplate = string.IsNullOrEmpty(repoURL) ? "https://gitlab.com/username/repository/-/issues/$1" : $"{repoURL}/-/issues/$1",
|
|
||||||
};
|
|
||||||
|
|
||||||
IssueTrackerRules.Add(rule);
|
|
||||||
return rule;
|
|
||||||
}
|
|
||||||
|
|
||||||
public IssueTrackerRule AddGitLabMergeRequestTracker(string repoURL)
|
|
||||||
{
|
|
||||||
var rule = new IssueTrackerRule()
|
|
||||||
{
|
|
||||||
Name = "GitLab MR",
|
|
||||||
RegexString = "!(\\d+)",
|
|
||||||
URLTemplate = string.IsNullOrEmpty(repoURL) ? "https://gitlab.com/username/repository/-/merge_requests/$1" : $"{repoURL}/-/merge_requests/$1",
|
|
||||||
};
|
|
||||||
|
|
||||||
IssueTrackerRules.Add(rule);
|
|
||||||
return rule;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void RemoveIssueTracker(IssueTrackerRule rule)
|
public void RemoveIssueTracker(IssueTrackerRule rule)
|
||||||
{
|
{
|
||||||
if (rule != null)
|
if (rule != null)
|
||||||
|
|
|
@ -147,8 +147,6 @@
|
||||||
<x:String x:Key="Text.Configure.IssueTracker" xml:space="preserve">ISSUE TRACKER</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker" xml:space="preserve">ISSUE TRACKER</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGithub" xml:space="preserve">Add Sample Github Rule</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGithub" xml:space="preserve">Add Sample Github Rule</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleJira" xml:space="preserve">Add Sample Jira Rule</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.AddSampleJira" xml:space="preserve">Add Sample Jira Rule</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGitLabIssue" xml:space="preserve">Add Sample GitLab Issue Rule</x:String>
|
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGitLabMergeRequest" xml:space="preserve">Add Sample GitLab Merge Request Rule</x:String>
|
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.NewRule" xml:space="preserve">New Rule</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.NewRule" xml:space="preserve">New Rule</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.Regex" xml:space="preserve">Issue Regex Expression:</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.Regex" xml:space="preserve">Issue Regex Expression:</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.RuleName" xml:space="preserve">Rule Name:</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.RuleName" xml:space="preserve">Rule Name:</x:String>
|
||||||
|
@ -402,11 +400,9 @@
|
||||||
<x:String x:Key="Text.Period.YearsAgo" xml:space="preserve">{0} years ago</x:String>
|
<x:String x:Key="Text.Period.YearsAgo" xml:space="preserve">{0} years ago</x:String>
|
||||||
<x:String x:Key="Text.Preference" xml:space="preserve">Preference</x:String>
|
<x:String x:Key="Text.Preference" xml:space="preserve">Preference</x:String>
|
||||||
<x:String x:Key="Text.Preference.AI" xml:space="preserve">OPEN AI</x:String>
|
<x:String x:Key="Text.Preference.AI" xml:space="preserve">OPEN AI</x:String>
|
||||||
<x:String x:Key="Text.Preference.AI.AnalyzeDiffPrompt" xml:space="preserve">Analyze Diff Prompt</x:String>
|
|
||||||
<x:String x:Key="Text.Preference.AI.ApiKey" xml:space="preserve">API Key</x:String>
|
|
||||||
<x:String x:Key="Text.Preference.AI.GenerateSubjectPrompt" xml:space="preserve">Generate Subject Prompt</x:String>
|
|
||||||
<x:String x:Key="Text.Preference.AI.Model" xml:space="preserve">Model</x:String>
|
|
||||||
<x:String x:Key="Text.Preference.AI.Server" xml:space="preserve">Server</x:String>
|
<x:String x:Key="Text.Preference.AI.Server" xml:space="preserve">Server</x:String>
|
||||||
|
<x:String x:Key="Text.Preference.AI.ApiKey" xml:space="preserve">API Key</x:String>
|
||||||
|
<x:String x:Key="Text.Preference.AI.Model" xml:space="preserve">Model</x:String>
|
||||||
<x:String x:Key="Text.Preference.Appearance" xml:space="preserve">APPEARANCE</x:String>
|
<x:String x:Key="Text.Preference.Appearance" xml:space="preserve">APPEARANCE</x:String>
|
||||||
<x:String x:Key="Text.Preference.Appearance.DefaultFont" xml:space="preserve">Default Font</x:String>
|
<x:String x:Key="Text.Preference.Appearance.DefaultFont" xml:space="preserve">Default Font</x:String>
|
||||||
<x:String x:Key="Text.Preference.Appearance.DefaultFontSize" xml:space="preserve">Default Font Size</x:String>
|
<x:String x:Key="Text.Preference.Appearance.DefaultFontSize" xml:space="preserve">Default Font Size</x:String>
|
||||||
|
|
|
@ -150,8 +150,6 @@
|
||||||
<x:String x:Key="Text.Configure.IssueTracker" xml:space="preserve">ISSUE追踪</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker" xml:space="preserve">ISSUE追踪</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGithub" xml:space="preserve">新增匹配Github Issue规则</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGithub" xml:space="preserve">新增匹配Github Issue规则</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleJira" xml:space="preserve">新增匹配Jira规则</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.AddSampleJira" xml:space="preserve">新增匹配Jira规则</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGitLabIssue" xml:space="preserve">新增匹配GitLab议题规则</x:String>
|
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGitLabMergeRequest" xml:space="preserve">新增匹配GitLab合并请求规则</x:String>
|
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.NewRule" xml:space="preserve">新增自定义规则</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.NewRule" xml:space="preserve">新增自定义规则</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.Regex" xml:space="preserve">匹配ISSUE的正则表达式 :</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.Regex" xml:space="preserve">匹配ISSUE的正则表达式 :</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.RuleName" xml:space="preserve">规则名 :</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.RuleName" xml:space="preserve">规则名 :</x:String>
|
||||||
|
|
|
@ -150,8 +150,6 @@
|
||||||
<x:String x:Key="Text.Configure.IssueTracker" xml:space="preserve">Issue 追蹤</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker" xml:space="preserve">Issue 追蹤</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGithub" xml:space="preserve">新增符合 GitHub Issue 規則</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGithub" xml:space="preserve">新增符合 GitHub Issue 規則</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleJira" xml:space="preserve">新增符合 Jira 規則</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.AddSampleJira" xml:space="preserve">新增符合 Jira 規則</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGitLabIssue" xml:space="preserve">新增符合 GitLab 議題規則</x:String>
|
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.AddSampleGitLabMergeRequest" xml:space="preserve">新增符合 GitLab 合併請求規則</x:String>
|
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.NewRule" xml:space="preserve">新增自訂規則</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.NewRule" xml:space="preserve">新增自訂規則</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.Regex" xml:space="preserve">符合 Issue 的正則表達式:</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.Regex" xml:space="preserve">符合 Issue 的正則表達式:</x:String>
|
||||||
<x:String x:Key="Text.Configure.IssueTracker.RuleName" xml:space="preserve">規則名稱:</x:String>
|
<x:String x:Key="Text.Configure.IssueTracker.RuleName" xml:space="preserve">規則名稱:</x:String>
|
||||||
|
|
|
@ -194,9 +194,9 @@ namespace SourceGit.ViewModels
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
if (l.Backend is Models.Branch)
|
if (l.Backend is Models.Branch)
|
||||||
return r.Backend is Models.Branch ? Models.NumericSort.Compare(l.Name, r.Name) : 1;
|
return r.Backend is Models.Branch ? string.Compare(l.Name, r.Name, StringComparison.Ordinal) : 1;
|
||||||
|
|
||||||
return r.Backend is Models.Branch ? -1 : Models.NumericSort.Compare(l.Name, r.Name);
|
return r.Backend is Models.Branch ? -1 : string.Compare(l.Name, r.Name, StringComparison.Ordinal);
|
||||||
});
|
});
|
||||||
|
|
||||||
foreach (var node in nodes)
|
foreach (var node in nodes)
|
||||||
|
|
|
@ -114,9 +114,9 @@ namespace SourceGit.ViewModels
|
||||||
|
|
||||||
nodes.Sort((l, r) =>
|
nodes.Sort((l, r) =>
|
||||||
{
|
{
|
||||||
if (l.IsFolder == r.IsFolder)
|
if (l.IsFolder)
|
||||||
return Models.NumericSort.Compare(l.FullPath, r.FullPath);
|
return r.IsFolder ? string.Compare(l.FullPath, r.FullPath, StringComparison.Ordinal) : -1;
|
||||||
return l.IsFolder ? -1 : 1;
|
return r.IsFolder ? 1 : string.Compare(l.FullPath, r.FullPath, StringComparison.Ordinal);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -25,7 +25,6 @@ namespace SourceGit.ViewModels
|
||||||
_instance.PrepareGit();
|
_instance.PrepareGit();
|
||||||
_instance.PrepareShellOrTerminal();
|
_instance.PrepareShellOrTerminal();
|
||||||
_instance.PrepareWorkspaces();
|
_instance.PrepareWorkspaces();
|
||||||
_instance.PrepareOpenAIPrompt();
|
|
||||||
|
|
||||||
return _instance;
|
return _instance;
|
||||||
}
|
}
|
||||||
|
@ -316,32 +315,6 @@ namespace SourceGit.ViewModels
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public string OpenAIAnalyzeDiffPrompt
|
|
||||||
{
|
|
||||||
get => Models.OpenAI.AnalyzeDiffPrompt;
|
|
||||||
set
|
|
||||||
{
|
|
||||||
if (value != Models.OpenAI.AnalyzeDiffPrompt)
|
|
||||||
{
|
|
||||||
Models.OpenAI.AnalyzeDiffPrompt = value;
|
|
||||||
OnPropertyChanged();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public string OpenAIGenerateSubjectPrompt
|
|
||||||
{
|
|
||||||
get => Models.OpenAI.GenerateSubjectPrompt;
|
|
||||||
set
|
|
||||||
{
|
|
||||||
if (value != Models.OpenAI.GenerateSubjectPrompt)
|
|
||||||
{
|
|
||||||
Models.OpenAI.GenerateSubjectPrompt = value;
|
|
||||||
OnPropertyChanged();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public uint StatisticsSampleColor
|
public uint StatisticsSampleColor
|
||||||
{
|
{
|
||||||
get => _statisticsSampleColor;
|
get => _statisticsSampleColor;
|
||||||
|
@ -554,45 +527,6 @@ namespace SourceGit.ViewModels
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void PrepareOpenAIPrompt()
|
|
||||||
{
|
|
||||||
if (string.IsNullOrEmpty(Models.OpenAI.AnalyzeDiffPrompt))
|
|
||||||
{
|
|
||||||
Models.OpenAI.AnalyzeDiffPrompt = """
|
|
||||||
You are an expert developer specialist in creating commits.
|
|
||||||
Provide a super concise one sentence overall changes summary of the user `git diff` output following strictly the next rules:
|
|
||||||
- Do not use any code snippets, imports, file routes or bullets points.
|
|
||||||
- Do not mention the route of file that has been change.
|
|
||||||
- Write clear, concise, and descriptive messages that explain the MAIN GOAL made of the changes.
|
|
||||||
- Use the present tense and active voice in the message, for example, "Fix bug" instead of "Fixed bug.".
|
|
||||||
- Use the imperative mood, which gives the message a sense of command, e.g. "Add feature" instead of "Added feature".
|
|
||||||
- Avoid using general terms like "update" or "change", be specific about what was updated or changed.
|
|
||||||
- Avoid using terms like "The main goal of", just output directly the summary in plain text
|
|
||||||
""";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(Models.OpenAI.GenerateSubjectPrompt))
|
|
||||||
{
|
|
||||||
Models.OpenAI.GenerateSubjectPrompt = """
|
|
||||||
You are an expert developer specialist in creating commits messages.
|
|
||||||
Your only goal is to retrieve a single commit message.
|
|
||||||
Based on the provided user changes, combine them in ONE SINGLE commit message retrieving the global idea, following strictly the next rules:
|
|
||||||
- Assign the commit {type} according to the next conditions:
|
|
||||||
feat: Only when adding a new feature.
|
|
||||||
fix: When fixing a bug.
|
|
||||||
docs: When updating documentation.
|
|
||||||
style: When changing elements styles or design and/or making changes to the code style (formatting, missing semicolons, etc.) without changing the code logic.
|
|
||||||
test: When adding or updating tests.
|
|
||||||
chore: When making changes to the build process or auxiliary tools and libraries.
|
|
||||||
revert: When undoing a previous commit.
|
|
||||||
refactor: When restructuring code without changing its external behavior, or is any of the other refactor types.
|
|
||||||
- Do not add any issues numeration, explain your output nor introduce your answer.
|
|
||||||
- Output directly only one commit message in plain text with the next format: {type}: {commit_message}.
|
|
||||||
- Be as concise as possible, keep the message under 50 characters.
|
|
||||||
""";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private RepositoryNode FindNodeRecursive(string id, List<RepositoryNode> collection)
|
private RepositoryNode FindNodeRecursive(string id, List<RepositoryNode> collection)
|
||||||
{
|
{
|
||||||
foreach (var node in collection)
|
foreach (var node in collection)
|
||||||
|
|
|
@ -172,34 +172,6 @@ namespace SourceGit.ViewModels
|
||||||
SelectedIssueTrackerRule = _repo.Settings.AddJiraIssueTracker();
|
SelectedIssueTrackerRule = _repo.Settings.AddJiraIssueTracker();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void AddSampleGitLabIssueTracker()
|
|
||||||
{
|
|
||||||
foreach (var remote in _repo.Remotes)
|
|
||||||
{
|
|
||||||
if (remote.TryGetVisitURL(out string url))
|
|
||||||
{
|
|
||||||
SelectedIssueTrackerRule = _repo.Settings.AddGitLabIssueTracker(url);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
SelectedIssueTrackerRule = _repo.Settings.AddGitLabIssueTracker(null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void AddSampleGitLabMergeRequestTracker()
|
|
||||||
{
|
|
||||||
foreach (var remote in _repo.Remotes)
|
|
||||||
{
|
|
||||||
if (remote.TryGetVisitURL(out string url))
|
|
||||||
{
|
|
||||||
SelectedIssueTrackerRule = _repo.Settings.AddGitLabMergeRequestTracker(url);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
SelectedIssueTrackerRule = _repo.Settings.AddGitLabMergeRequestTracker(null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void NewIssueTracker()
|
public void NewIssueTracker()
|
||||||
{
|
{
|
||||||
SelectedIssueTrackerRule = _repo.Settings.AddNewIssueTracker();
|
SelectedIssueTrackerRule = _repo.Settings.AddNewIssueTracker();
|
||||||
|
|
|
@ -465,7 +465,7 @@
|
||||||
<TextBlock Classes="bold" Margin="4,0,0,0" Text="{DynamicResource Text.Preference.AI}"/>
|
<TextBlock Classes="bold" Margin="4,0,0,0" Text="{DynamicResource Text.Preference.AI}"/>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
<Rectangle Margin="0,8" Fill="{DynamicResource Brush.Border2}" Height=".6" HorizontalAlignment="Stretch"/>
|
<Rectangle Margin="0,8" Fill="{DynamicResource Brush.Border2}" Height=".6" HorizontalAlignment="Stretch"/>
|
||||||
<Grid Margin="8,0,0,0" RowDefinitions="32,32,32,128,128">
|
<Grid Margin="8,0,0,0" RowDefinitions="32,32,32">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="Auto" SharedSizeGroup="IntegrationLabel"/>
|
<ColumnDefinition Width="Auto" SharedSizeGroup="IntegrationLabel"/>
|
||||||
<ColumnDefinition Width="*"/>
|
<ColumnDefinition Width="*"/>
|
||||||
|
@ -497,30 +497,6 @@
|
||||||
Height="28"
|
Height="28"
|
||||||
CornerRadius="3"
|
CornerRadius="3"
|
||||||
Text="{Binding OpenAIApiKey, Mode=TwoWay}"/>
|
Text="{Binding OpenAIApiKey, Mode=TwoWay}"/>
|
||||||
|
|
||||||
<TextBlock Grid.Row="3" Grid.Column="0"
|
|
||||||
Text="{DynamicResource Text.Preference.AI.AnalyzeDiffPrompt}"
|
|
||||||
HorizontalAlignment="Right"
|
|
||||||
Margin="0,0,16,0"/>
|
|
||||||
<TextBox Grid.Row="3" Grid.Column="1"
|
|
||||||
Height="120"
|
|
||||||
CornerRadius="3"
|
|
||||||
VerticalContentAlignment="Top"
|
|
||||||
Text="{Binding OpenAIAnalyzeDiffPrompt, Mode=TwoWay}"
|
|
||||||
AcceptsReturn="true"
|
|
||||||
TextWrapping="Wrap"/>
|
|
||||||
|
|
||||||
<TextBlock Grid.Row="4" Grid.Column="0"
|
|
||||||
Text="{DynamicResource Text.Preference.AI.GenerateSubjectPrompt}"
|
|
||||||
HorizontalAlignment="Right"
|
|
||||||
Margin="0,0,16,0"/>
|
|
||||||
<TextBox Grid.Row="4" Grid.Column="1"
|
|
||||||
Height="120"
|
|
||||||
CornerRadius="3"
|
|
||||||
VerticalContentAlignment="Top"
|
|
||||||
Text="{Binding OpenAIGenerateSubjectPrompt, Mode=TwoWay}"
|
|
||||||
AcceptsReturn="true"
|
|
||||||
TextWrapping="Wrap"/>
|
|
||||||
</Grid>
|
</Grid>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</TabItem>
|
</TabItem>
|
||||||
|
|
|
@ -287,8 +287,6 @@
|
||||||
<MenuItem Header="-"/>
|
<MenuItem Header="-"/>
|
||||||
<MenuItem Header="{DynamicResource Text.Configure.IssueTracker.AddSampleGithub}" Command="{Binding AddSampleGithubIssueTracker}"/>
|
<MenuItem Header="{DynamicResource Text.Configure.IssueTracker.AddSampleGithub}" Command="{Binding AddSampleGithubIssueTracker}"/>
|
||||||
<MenuItem Header="{DynamicResource Text.Configure.IssueTracker.AddSampleJira}" Command="{Binding AddSampleJiraIssueTracker}"/>
|
<MenuItem Header="{DynamicResource Text.Configure.IssueTracker.AddSampleJira}" Command="{Binding AddSampleJiraIssueTracker}"/>
|
||||||
<MenuItem Header="{DynamicResource Text.Configure.IssueTracker.AddSampleGitLabIssue}" Command="{Binding AddSampleGitLabIssueTracker}"/>
|
|
||||||
<MenuItem Header="{DynamicResource Text.Configure.IssueTracker.AddSampleGitLabMergeRequest}" Command="{Binding AddSampleGitLabMergeRequestTracker}"/>
|
|
||||||
</MenuFlyout>
|
</MenuFlyout>
|
||||||
</Button.Flyout>
|
</Button.Flyout>
|
||||||
<Path Width="14" Height="14" Data="{StaticResource Icons.Plus}"/>
|
<Path Width="14" Height="14" Data="{StaticResource Icons.Plus}"/>
|
||||||
|
|
|
@ -284,7 +284,7 @@ namespace SourceGit.Views
|
||||||
node.Children.Sort((l, r) =>
|
node.Children.Sort((l, r) =>
|
||||||
{
|
{
|
||||||
if (l.IsFolder == r.IsFolder)
|
if (l.IsFolder == r.IsFolder)
|
||||||
return Models.NumericSort.Compare(l.Name, r.Name);
|
return string.Compare(l.Name, r.Name, StringComparison.Ordinal);
|
||||||
return l.IsFolder ? -1 : 1;
|
return l.IsFolder ? -1 : 1;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue