mirror of
https://github.com/YspCoder/clawgo.git
synced 2026-04-13 13:17:28 +08:00
Enhance spec-driven coding workflow
This commit is contained in:
540
pkg/agent/spec_coding.go
Normal file
540
pkg/agent/spec_coding.go
Normal file
@@ -0,0 +1,540 @@
|
||||
package agent
|
||||
|
||||
import (
|
||||
"crypto/sha1"
|
||||
"encoding/hex"
|
||||
"fmt"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"regexp"
|
||||
"strings"
|
||||
"time"
|
||||
)
|
||||
|
||||
var specCodingDocNames = []string{"spec.md", "tasks.md", "checklist.md"}
|
||||
var reWhitespace = regexp.MustCompile(`\s+`)
|
||||
var reSpecTaskMeta = regexp.MustCompile(`\s*<!--\s*spec-task-id:([a-f0-9]+)\s*-->`)
|
||||
var reChecklistItem = regexp.MustCompile(`(?m)^- \[( |x)\] (.+)$`)
|
||||
|
||||
type specCodingTaskRef struct {
|
||||
ID string
|
||||
Summary string
|
||||
}
|
||||
|
||||
func (al *AgentLoop) maybeEnsureSpecCodingDocs(currentMessage string) error {
|
||||
if al == nil || al.contextBuilder == nil {
|
||||
return nil
|
||||
}
|
||||
if !al.contextBuilder.shouldUseSpecCoding(currentMessage) {
|
||||
return nil
|
||||
}
|
||||
projectRoot := al.contextBuilder.projectRootPath()
|
||||
if strings.TrimSpace(projectRoot) == "" {
|
||||
return nil
|
||||
}
|
||||
_, err := ensureSpecCodingDocs(al.workspace, projectRoot)
|
||||
return err
|
||||
}
|
||||
|
||||
func (al *AgentLoop) maybeStartSpecCodingTask(currentMessage string) (specCodingTaskRef, error) {
|
||||
if al == nil || al.contextBuilder == nil || !al.contextBuilder.shouldUseSpecCoding(currentMessage) {
|
||||
return specCodingTaskRef{}, nil
|
||||
}
|
||||
projectRoot := al.contextBuilder.projectRootPath()
|
||||
if strings.TrimSpace(projectRoot) == "" {
|
||||
return specCodingTaskRef{}, nil
|
||||
}
|
||||
return upsertSpecCodingTask(projectRoot, currentMessage)
|
||||
}
|
||||
|
||||
func (al *AgentLoop) maybeCompleteSpecCodingTask(taskRef specCodingTaskRef, response string) error {
|
||||
if al == nil || al.contextBuilder == nil {
|
||||
return nil
|
||||
}
|
||||
taskRef = normalizeSpecCodingTaskRef(taskRef)
|
||||
if taskRef.Summary == "" {
|
||||
return nil
|
||||
}
|
||||
projectRoot := al.contextBuilder.projectRootPath()
|
||||
if strings.TrimSpace(projectRoot) == "" {
|
||||
return nil
|
||||
}
|
||||
if err := completeSpecCodingTask(projectRoot, taskRef, response); err != nil {
|
||||
return err
|
||||
}
|
||||
return updateSpecCodingChecklist(projectRoot, taskRef, response)
|
||||
}
|
||||
|
||||
func (al *AgentLoop) maybeReopenSpecCodingTask(taskRef specCodingTaskRef, currentMessage, reason string) error {
|
||||
if al == nil || al.contextBuilder == nil {
|
||||
return nil
|
||||
}
|
||||
taskRef = normalizeSpecCodingTaskRef(taskRef)
|
||||
if taskRef.Summary == "" {
|
||||
return nil
|
||||
}
|
||||
if strings.TrimSpace(reason) == "" && !shouldReopenSpecCodingTask(currentMessage) {
|
||||
return nil
|
||||
}
|
||||
projectRoot := al.contextBuilder.projectRootPath()
|
||||
if strings.TrimSpace(projectRoot) == "" {
|
||||
return nil
|
||||
}
|
||||
note := strings.TrimSpace(reason)
|
||||
if note == "" {
|
||||
note = strings.TrimSpace(currentMessage)
|
||||
}
|
||||
if err := reopenSpecCodingTask(projectRoot, taskRef, note); err != nil {
|
||||
return err
|
||||
}
|
||||
return resetSpecCodingChecklist(projectRoot, taskRef, note)
|
||||
}
|
||||
|
||||
func ensureSpecCodingDocs(workspace, projectRoot string) ([]string, error) {
|
||||
workspace = strings.TrimSpace(workspace)
|
||||
projectRoot = strings.TrimSpace(projectRoot)
|
||||
if workspace == "" || projectRoot == "" {
|
||||
return nil, nil
|
||||
}
|
||||
projectRoot = filepath.Clean(projectRoot)
|
||||
if err := os.MkdirAll(projectRoot, 0755); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
templatesDir := filepath.Join(workspace, "skills", "spec-coding", "templates")
|
||||
created := make([]string, 0, len(specCodingDocNames))
|
||||
for _, name := range specCodingDocNames {
|
||||
targetPath := filepath.Join(projectRoot, name)
|
||||
if _, err := os.Stat(targetPath); err == nil {
|
||||
continue
|
||||
} else if !os.IsNotExist(err) {
|
||||
return created, err
|
||||
}
|
||||
templatePath := filepath.Join(templatesDir, name)
|
||||
data, err := os.ReadFile(templatePath)
|
||||
if err != nil {
|
||||
return created, fmt.Errorf("read spec-coding template %s failed: %w", templatePath, err)
|
||||
}
|
||||
if err := os.WriteFile(targetPath, data, 0644); err != nil {
|
||||
return created, err
|
||||
}
|
||||
created = append(created, targetPath)
|
||||
}
|
||||
return created, nil
|
||||
}
|
||||
|
||||
func upsertSpecCodingTask(projectRoot, currentMessage string) (specCodingTaskRef, error) {
|
||||
projectRoot = strings.TrimSpace(projectRoot)
|
||||
taskSummary := summarizeSpecCodingTask(currentMessage)
|
||||
if projectRoot == "" || taskSummary == "" {
|
||||
return specCodingTaskRef{}, nil
|
||||
}
|
||||
taskID := stableSpecCodingTaskID(taskSummary)
|
||||
taskRef := specCodingTaskRef{ID: taskID, Summary: taskSummary}
|
||||
tasksPath := filepath.Join(projectRoot, "tasks.md")
|
||||
data, err := os.ReadFile(tasksPath)
|
||||
if err != nil {
|
||||
return specCodingTaskRef{}, err
|
||||
}
|
||||
text := string(data)
|
||||
if line, done, ok := findSpecCodingTaskLine(text, taskRef); ok {
|
||||
if done {
|
||||
if shouldReopenSpecCodingTask(currentMessage) {
|
||||
if err := reopenSpecCodingTask(projectRoot, taskRef, currentMessage); err != nil {
|
||||
return specCodingTaskRef{}, err
|
||||
}
|
||||
}
|
||||
}
|
||||
if strings.Contains(line, "<!--") {
|
||||
return specTaskRefFromLine(line), nil
|
||||
}
|
||||
return taskRef, nil
|
||||
}
|
||||
doneLine := renderSpecCodingTaskLine(true, taskRef)
|
||||
openLine := renderSpecCodingTaskLine(false, taskRef)
|
||||
if strings.Contains(text, "- [x] "+taskSummary) || strings.Contains(text, doneLine) {
|
||||
if shouldReopenSpecCodingTask(currentMessage) {
|
||||
if err := reopenSpecCodingTask(projectRoot, taskRef, currentMessage); err != nil {
|
||||
return specCodingTaskRef{}, err
|
||||
}
|
||||
}
|
||||
return taskRef, nil
|
||||
}
|
||||
if strings.Contains(text, "- [ ] "+taskSummary) || strings.Contains(text, openLine) {
|
||||
return taskRef, nil
|
||||
}
|
||||
if shouldReopenSpecCodingTask(currentMessage) {
|
||||
if reopened, err := maybeReopenCompletedTaskFromMessage(projectRoot, currentMessage, text); err != nil {
|
||||
return specCodingTaskRef{}, err
|
||||
} else if reopened.Summary != "" {
|
||||
return reopened, nil
|
||||
}
|
||||
}
|
||||
section := "## Current Coding Tasks\n"
|
||||
entry := openLine + "\n"
|
||||
if strings.Contains(text, section) {
|
||||
text = strings.Replace(text, section, section+entry, 1)
|
||||
} else if idx := strings.Index(text, "## Progress Notes"); idx >= 0 {
|
||||
text = text[:idx] + section + "\n" + entry + "\n" + text[idx:]
|
||||
} else {
|
||||
if !strings.HasSuffix(text, "\n") {
|
||||
text += "\n"
|
||||
}
|
||||
text += "\n" + section + "\n" + entry
|
||||
}
|
||||
if err := os.WriteFile(tasksPath, []byte(text), 0644); err != nil {
|
||||
return specCodingTaskRef{}, err
|
||||
}
|
||||
return taskRef, nil
|
||||
}
|
||||
|
||||
func completeSpecCodingTask(projectRoot string, taskRef specCodingTaskRef, response string) error {
|
||||
projectRoot = strings.TrimSpace(projectRoot)
|
||||
taskRef = normalizeSpecCodingTaskRef(taskRef)
|
||||
if projectRoot == "" || taskRef.Summary == "" {
|
||||
return nil
|
||||
}
|
||||
tasksPath := filepath.Join(projectRoot, "tasks.md")
|
||||
data, err := os.ReadFile(tasksPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
text := string(data)
|
||||
if line, done, ok := findSpecCodingTaskLine(text, taskRef); ok {
|
||||
if !done {
|
||||
text = strings.Replace(text, line, renderSpecCodingTaskLine(true, specTaskRefFromLine(line)), 1)
|
||||
}
|
||||
} else if strings.Contains(text, "- [ ] "+taskRef.Summary) {
|
||||
text = strings.Replace(text, "- [ ] "+taskRef.Summary, renderSpecCodingTaskLine(true, taskRef), 1)
|
||||
}
|
||||
note := fmt.Sprintf("- %s %s -> %s\n", time.Now().Format("2006-01-02 15:04"), taskRef.Summary, summarizeSpecCodingTask(response))
|
||||
if strings.Contains(text, note) {
|
||||
return nil
|
||||
}
|
||||
if strings.Contains(text, "## Progress Notes\n") {
|
||||
text = strings.Replace(text, "## Progress Notes\n", "## Progress Notes\n"+note, 1)
|
||||
} else {
|
||||
if !strings.HasSuffix(text, "\n") {
|
||||
text += "\n"
|
||||
}
|
||||
text += "\n## Progress Notes\n" + note
|
||||
}
|
||||
return os.WriteFile(tasksPath, []byte(text), 0644)
|
||||
}
|
||||
|
||||
func maybeReopenCompletedTaskFromMessage(projectRoot, currentMessage, tasksText string) (specCodingTaskRef, error) {
|
||||
completed := extractSpecCodingTasks(tasksText, "- [x] ")
|
||||
if len(completed) == 0 {
|
||||
return specCodingTaskRef{}, nil
|
||||
}
|
||||
if len(completed) == 1 {
|
||||
return completed[0], reopenSpecCodingTask(projectRoot, completed[0], currentMessage)
|
||||
}
|
||||
best := specCodingTaskRef{}
|
||||
bestScore := 0
|
||||
for _, item := range completed {
|
||||
score := scoreSpecCodingTaskMatch(item.Summary, currentMessage)
|
||||
if score > bestScore {
|
||||
best = item
|
||||
bestScore = score
|
||||
}
|
||||
}
|
||||
if best.Summary == "" || bestScore == 0 {
|
||||
return specCodingTaskRef{}, nil
|
||||
}
|
||||
return best, reopenSpecCodingTask(projectRoot, best, currentMessage)
|
||||
}
|
||||
|
||||
func reopenSpecCodingTask(projectRoot string, taskRef specCodingTaskRef, reason string) error {
|
||||
projectRoot = strings.TrimSpace(projectRoot)
|
||||
taskRef = normalizeSpecCodingTaskRef(taskRef)
|
||||
if projectRoot == "" || taskRef.Summary == "" {
|
||||
return nil
|
||||
}
|
||||
tasksPath := filepath.Join(projectRoot, "tasks.md")
|
||||
data, err := os.ReadFile(tasksPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
text := string(data)
|
||||
if line, done, ok := findSpecCodingTaskLine(text, taskRef); ok {
|
||||
if done {
|
||||
text = strings.Replace(text, line, renderSpecCodingTaskLine(false, specTaskRefFromLine(line)), 1)
|
||||
}
|
||||
} else if strings.Contains(text, "- [x] "+taskRef.Summary) {
|
||||
text = strings.Replace(text, "- [x] "+taskRef.Summary, renderSpecCodingTaskLine(false, taskRef), 1)
|
||||
}
|
||||
note := fmt.Sprintf("- %s reopened %s -> %s\n", time.Now().Format("2006-01-02 15:04"), taskRef.Summary, summarizeSpecCodingTask(reason))
|
||||
if strings.Contains(text, note) {
|
||||
return nil
|
||||
}
|
||||
if strings.Contains(text, "## Progress Notes\n") {
|
||||
text = strings.Replace(text, "## Progress Notes\n", "## Progress Notes\n"+note, 1)
|
||||
} else {
|
||||
if !strings.HasSuffix(text, "\n") {
|
||||
text += "\n"
|
||||
}
|
||||
text += "\n## Progress Notes\n" + note
|
||||
}
|
||||
return os.WriteFile(tasksPath, []byte(text), 0644)
|
||||
}
|
||||
|
||||
func summarizeSpecCodingTask(text string) string {
|
||||
text = strings.TrimSpace(text)
|
||||
if text == "" {
|
||||
return ""
|
||||
}
|
||||
text = strings.ReplaceAll(text, "\r\n", "\n")
|
||||
text = strings.ReplaceAll(text, "\n", " ")
|
||||
text = reWhitespace.ReplaceAllString(text, " ")
|
||||
text = strings.TrimSpace(text)
|
||||
if len(text) > 120 {
|
||||
text = strings.TrimSpace(text[:117]) + "..."
|
||||
}
|
||||
return text
|
||||
}
|
||||
|
||||
func shouldReopenSpecCodingTask(text string) bool {
|
||||
text = strings.ToLower(strings.TrimSpace(text))
|
||||
if text == "" {
|
||||
return false
|
||||
}
|
||||
hints := []string{
|
||||
"bug", "issue", "problem", "broken", "fix again", "regression", "failing", "failure", "debug", "rework",
|
||||
"有问题", "有 bug", "修一下", "重新修复", "回归失败", "排查", "重做", "返工", "异常", "坏了",
|
||||
}
|
||||
for _, hint := range hints {
|
||||
if strings.Contains(text, hint) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func updateSpecCodingChecklist(projectRoot string, taskRef specCodingTaskRef, response string) error {
|
||||
projectRoot = strings.TrimSpace(projectRoot)
|
||||
taskRef = normalizeSpecCodingTaskRef(taskRef)
|
||||
if projectRoot == "" || taskRef.Summary == "" {
|
||||
return nil
|
||||
}
|
||||
checklistPath := filepath.Join(projectRoot, "checklist.md")
|
||||
data, err := os.ReadFile(checklistPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
text := string(data)
|
||||
checked := make(map[string]bool)
|
||||
checked["Scope implemented"] = true
|
||||
|
||||
evidence := strings.ToLower(taskRef.Summary + "\n" + response)
|
||||
if containsAny(evidence, "test", "tests", "go test", "验证", "校验", "回归", "通过", "passed", "validated") {
|
||||
checked["Tests added or updated where needed"] = true
|
||||
checked["Validation run"] = true
|
||||
}
|
||||
if containsAny(evidence, "edge", "corner", "boundary", "边界", "异常场景", "边缘", "case reviewed") {
|
||||
checked["Edge cases reviewed"] = true
|
||||
}
|
||||
if containsAny(evidence, "doc", "docs", "readme", "prompt", "config", "文档", "配置", "说明") {
|
||||
checked["Docs / prompts / config updated if required"] = true
|
||||
}
|
||||
if !containsAny(evidence, "todo", "follow-up", "remaining", "pending", "未完成", "后续", "待处理", "blocker", "blocked") {
|
||||
checked["No known missing follow-up inside current scope"] = true
|
||||
}
|
||||
|
||||
text = reChecklistItem.ReplaceAllStringFunc(text, func(line string) string {
|
||||
matches := reChecklistItem.FindStringSubmatch(line)
|
||||
if len(matches) != 3 {
|
||||
return line
|
||||
}
|
||||
label := strings.TrimSpace(matches[2])
|
||||
if checked[label] {
|
||||
return "- [x] " + label
|
||||
}
|
||||
return line
|
||||
})
|
||||
|
||||
note := fmt.Sprintf("- %s verified %s -> %s\n", time.Now().Format("2006-01-02 15:04"), taskRef.Summary, summarizeChecklistEvidence(checked))
|
||||
text = upsertChecklistNotes(text, note)
|
||||
return os.WriteFile(checklistPath, []byte(text), 0644)
|
||||
}
|
||||
|
||||
func resetSpecCodingChecklist(projectRoot string, taskRef specCodingTaskRef, reason string) error {
|
||||
projectRoot = strings.TrimSpace(projectRoot)
|
||||
taskRef = normalizeSpecCodingTaskRef(taskRef)
|
||||
if projectRoot == "" || taskRef.Summary == "" {
|
||||
return nil
|
||||
}
|
||||
checklistPath := filepath.Join(projectRoot, "checklist.md")
|
||||
data, err := os.ReadFile(checklistPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
text := reChecklistItem.ReplaceAllString(string(data), "- [ ] $2")
|
||||
note := fmt.Sprintf("- %s reopened %s -> %s\n", time.Now().Format("2006-01-02 15:04"), taskRef.Summary, summarizeSpecCodingTask(reason))
|
||||
text = upsertChecklistNotes(text, note)
|
||||
return os.WriteFile(checklistPath, []byte(text), 0644)
|
||||
}
|
||||
|
||||
func upsertChecklistNotes(text, note string) string {
|
||||
if strings.Contains(text, note) {
|
||||
return text
|
||||
}
|
||||
section := "## Verification Notes\n"
|
||||
if strings.Contains(text, section) {
|
||||
return strings.Replace(text, section, section+note, 1)
|
||||
}
|
||||
if !strings.HasSuffix(text, "\n") {
|
||||
text += "\n"
|
||||
}
|
||||
return text + "\n" + section + note
|
||||
}
|
||||
|
||||
func summarizeChecklistEvidence(checked map[string]bool) string {
|
||||
if len(checked) == 0 {
|
||||
return "no checklist items matched"
|
||||
}
|
||||
order := []string{
|
||||
"Scope implemented",
|
||||
"Edge cases reviewed",
|
||||
"Tests added or updated where needed",
|
||||
"Validation run",
|
||||
"Docs / prompts / config updated if required",
|
||||
"No known missing follow-up inside current scope",
|
||||
}
|
||||
items := make([]string, 0, len(order))
|
||||
for _, label := range order {
|
||||
if checked[label] {
|
||||
items = append(items, label)
|
||||
}
|
||||
}
|
||||
if len(items) == 0 {
|
||||
return "no checklist items matched"
|
||||
}
|
||||
return strings.Join(items, "; ")
|
||||
}
|
||||
|
||||
func containsAny(text string, parts ...string) bool {
|
||||
for _, part := range parts {
|
||||
if strings.Contains(text, strings.ToLower(strings.TrimSpace(part))) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func extractSpecCodingTasks(tasksText, prefix string) []specCodingTaskRef {
|
||||
lines := strings.Split(tasksText, "\n")
|
||||
out := make([]specCodingTaskRef, 0, 4)
|
||||
for _, line := range lines {
|
||||
t := strings.TrimSpace(line)
|
||||
if !strings.HasPrefix(t, prefix) {
|
||||
continue
|
||||
}
|
||||
ref := specTaskRefFromLine(t)
|
||||
if ref.Summary == "" {
|
||||
continue
|
||||
}
|
||||
out = append(out, ref)
|
||||
}
|
||||
return out
|
||||
}
|
||||
|
||||
func stableSpecCodingTaskID(summary string) string {
|
||||
summary = summarizeSpecCodingTask(summary)
|
||||
if summary == "" {
|
||||
return ""
|
||||
}
|
||||
sum := sha1.Sum([]byte(strings.ToLower(summary)))
|
||||
return hex.EncodeToString(sum[:])[:12]
|
||||
}
|
||||
|
||||
func normalizeSpecCodingTaskRef(taskRef specCodingTaskRef) specCodingTaskRef {
|
||||
taskRef.Summary = summarizeSpecCodingTask(taskRef.Summary)
|
||||
taskRef.ID = strings.ToLower(strings.TrimSpace(taskRef.ID))
|
||||
if taskRef.Summary == "" {
|
||||
return specCodingTaskRef{}
|
||||
}
|
||||
if taskRef.ID == "" {
|
||||
taskRef.ID = stableSpecCodingTaskID(taskRef.Summary)
|
||||
}
|
||||
return taskRef
|
||||
}
|
||||
|
||||
func renderSpecCodingTaskLine(done bool, taskRef specCodingTaskRef) string {
|
||||
taskRef = normalizeSpecCodingTaskRef(taskRef)
|
||||
if taskRef.Summary == "" {
|
||||
return ""
|
||||
}
|
||||
state := " "
|
||||
if done {
|
||||
state = "x"
|
||||
}
|
||||
return fmt.Sprintf("- [%s] %s <!-- spec-task-id:%s -->", state, taskRef.Summary, taskRef.ID)
|
||||
}
|
||||
|
||||
func specTaskRefFromLine(line string) specCodingTaskRef {
|
||||
line = strings.TrimSpace(line)
|
||||
line = strings.TrimPrefix(line, "- [ ] ")
|
||||
line = strings.TrimPrefix(line, "- [x] ")
|
||||
ref := specCodingTaskRef{}
|
||||
if matches := reSpecTaskMeta.FindStringSubmatch(line); len(matches) == 2 {
|
||||
ref.ID = strings.ToLower(strings.TrimSpace(matches[1]))
|
||||
line = strings.TrimSpace(reSpecTaskMeta.ReplaceAllString(line, ""))
|
||||
}
|
||||
ref.Summary = summarizeSpecCodingTask(line)
|
||||
return normalizeSpecCodingTaskRef(ref)
|
||||
}
|
||||
|
||||
func findSpecCodingTaskLine(tasksText string, taskRef specCodingTaskRef) (string, bool, bool) {
|
||||
taskRef = normalizeSpecCodingTaskRef(taskRef)
|
||||
if taskRef.Summary == "" {
|
||||
return "", false, false
|
||||
}
|
||||
for _, raw := range strings.Split(tasksText, "\n") {
|
||||
line := strings.TrimSpace(raw)
|
||||
if !strings.HasPrefix(line, "- [ ] ") && !strings.HasPrefix(line, "- [x] ") {
|
||||
continue
|
||||
}
|
||||
ref := specTaskRefFromLine(line)
|
||||
if ref.Summary == "" {
|
||||
continue
|
||||
}
|
||||
if ref.ID != "" && ref.ID == taskRef.ID {
|
||||
return line, strings.HasPrefix(line, "- [x] "), true
|
||||
}
|
||||
if ref.Summary == taskRef.Summary {
|
||||
return line, strings.HasPrefix(line, "- [x] "), true
|
||||
}
|
||||
}
|
||||
return "", false, false
|
||||
}
|
||||
|
||||
func scoreSpecCodingTaskMatch(taskSummary, currentMessage string) int {
|
||||
taskSummary = summarizeSpecCodingTask(taskSummary)
|
||||
currentMessage = summarizeSpecCodingTask(currentMessage)
|
||||
if taskSummary == "" || currentMessage == "" {
|
||||
return 0
|
||||
}
|
||||
if strings.Contains(currentMessage, taskSummary) || strings.Contains(taskSummary, currentMessage) {
|
||||
return len(taskSummary)
|
||||
}
|
||||
shorter := taskSummary
|
||||
longer := currentMessage
|
||||
if len([]rune(shorter)) > len([]rune(longer)) {
|
||||
shorter, longer = longer, shorter
|
||||
}
|
||||
runes := []rune(shorter)
|
||||
best := 0
|
||||
for i := 0; i < len(runes); i++ {
|
||||
for j := i + 2; j <= len(runes); j++ {
|
||||
frag := string(runes[i:j])
|
||||
if strings.TrimSpace(frag) == "" {
|
||||
continue
|
||||
}
|
||||
if strings.Contains(longer, frag) && len([]rune(frag)) > best {
|
||||
best = len([]rune(frag))
|
||||
}
|
||||
}
|
||||
}
|
||||
return best
|
||||
}
|
||||
Reference in New Issue
Block a user