feat: add Cobra CLI, LSP/MCP registries, workflow engine, and enriched dashboard
All checks were successful
Beta Release / beta (push) Successful in 2m24s

Major changes:
- Refactor CLI entry point to Cobra commands (root, setup, scan, doctor, install, update, lsp, mcp, skills, config, version)
- Add LSP registry with health checks, auto-install, and editor config generation
- Add MCP registry with editor detection, status tracking, and per-editor configuration
- Add workflow engine with planner and step execution for automated task chains
- Add conversation search, export (Markdown/JSON), and detailed token counting
- Add streaming shell chat handler with tool call/result events
- Add skill validation, dry-run testing, and export endpoints
- Enrich dashboard with Tools/Activity/Status tabs and tool cards grid
- Add PRD documentation
- Complete i18n for both EN and FR

💘 Generated with Crush

Assisted-by: GLM-5.1 via Crush <crush@charm.land>
This commit is contained in:
Augustin
2026-04-22 22:22:05 +02:00
parent 66b773ff86
commit 2e50366cd8
42 changed files with 6779 additions and 319 deletions

View File

@@ -0,0 +1,59 @@
package commands
import (
"fmt"
"github.com/muyue/muyue/internal/config"
"github.com/spf13/cobra"
)
var configCmd = &cobra.Command{
Use: "config",
Short: "Show/print config",
}
func init() {
rootCmd.AddCommand(configCmd)
}
func runConfigGet(cmd *cobra.Command, args []string) error {
cfg, err := config.Load()
if err != nil {
return err
}
key := args[0]
fmt.Fprintf(cmd.OutOrStdout(), "%v\n", getConfigValue(cfg, key))
return nil
}
func getConfigValue(cfg *config.MuyueConfig, key string) interface{} {
switch key {
case "version":
return cfg.Version
case "profile.name":
return cfg.Profile.Name
case "profile.email":
return cfg.Profile.Email
default:
return nil
}
}
func runConfigSet(cmd *cobra.Command, args []string) error {
cfg, err := config.Load()
if err != nil {
return err
}
key, value := args[0], args[1]
setConfigValue(cfg, key, value)
return config.Save(cfg)
}
func setConfigValue(cfg *config.MuyueConfig, key, value string) {
switch key {
case "profile.name":
cfg.Profile.Name = value
case "profile.email":
cfg.Profile.Email = value
}
}

View File

@@ -0,0 +1,77 @@
package commands
import (
"fmt"
"net/http"
"time"
"github.com/muyue/muyue/internal/config"
"github.com/muyue/muyue/internal/scanner"
"github.com/spf13/cobra"
)
var doctorCmd = &cobra.Command{
Use: "doctor",
Short: "Diagnose issues (scan + config check + connectivity)",
RunE: runDoctor,
}
func init() {
rootCmd.AddCommand(doctorCmd)
}
func runDoctor(cmd *cobra.Command, args []string) error {
fmt.Println("Running Muyue diagnostics...")
fmt.Println("\n=== System Scan ===")
result := scanner.ScanSystem()
for _, t := range result.Tools {
status := "✓"
if !t.Installed {
status = "✗"
}
fmt.Printf(" %s %s\n", status, t.Name)
}
fmt.Printf("\nInstalled: %d/%d\n", countInstalled(result.Tools), len(result.Tools))
fmt.Println("\n=== Config Check ===")
cfg, err := config.Load()
if err != nil {
fmt.Printf(" ✗ Failed to load config: %v\n", err)
} else {
fmt.Printf(" ✓ Config loaded (version: %s)\n", cfg.Version)
if cfg.Profile.Name != "" {
fmt.Printf(" ✓ Profile: %s\n", cfg.Profile.Name)
}
}
fmt.Println("\n=== Connectivity ===")
endpoints := []string{
"https://api.minimax.io",
"https://api.openai.com",
}
for _, ep := range endpoints {
fmt.Printf(" Checking %s... ", ep)
client := &http.Client{Timeout: 5 * time.Second}
resp, err := client.Head(ep)
if err != nil {
fmt.Printf("✗ (%v)\n", err)
} else {
resp.Body.Close()
fmt.Printf("✓ (status %d)\n", resp.StatusCode)
}
}
fmt.Println("\n=== Diagnosis complete ===")
return nil
}
func countInstalled(tools []scanner.ToolStatus) int {
installed := 0
for _, t := range tools {
if t.Installed {
installed++
}
}
return installed
}

View File

@@ -0,0 +1,56 @@
package commands
import (
"fmt"
"github.com/muyue/muyue/internal/installer"
"github.com/muyue/muyue/internal/scanner"
"github.com/spf13/cobra"
)
var installCmd = &cobra.Command{
Use: "install [tool]",
Short: "Install missing tools",
Args: cobra.RangeArgs(0, 1),
RunE: runInstall,
}
var installYes bool
func init() {
rootCmd.AddCommand(installCmd)
installCmd.Flags().BoolVar(&installYes, "yes", false, "Skip confirmation")
}
func runInstall(cmd *cobra.Command, args []string) error {
var tools []string
if len(args) > 0 {
tools = args
}
inst := installer.New(nil)
if len(tools) == 0 {
result := scanner.ScanSystem()
for _, t := range result.Tools {
if !t.Installed {
tools = append(tools, t.Name)
}
}
if len(tools) == 0 {
fmt.Println("All tools already installed!")
return nil
}
fmt.Printf("Installing missing tools: %v\n", tools)
}
for _, tool := range tools {
fmt.Printf("Installing %s...\n", tool)
res := inst.InstallTool(tool)
if res.Success {
fmt.Printf("✓ %s: %s\n", tool, res.Message)
} else {
fmt.Printf("✗ %s: %s\n", tool, res.Message)
}
}
return nil
}

55
cmd/muyue/commands/lsp.go Normal file
View File

@@ -0,0 +1,55 @@
package commands
import (
"fmt"
"github.com/muyue/muyue/internal/lsp"
"github.com/spf13/cobra"
)
var lspCmd = &cobra.Command{
Use: "lsp",
Short: "LSP management",
}
func init() {
rootCmd.AddCommand(lspCmd)
lspCmd.AddCommand(&cobra.Command{
Use: "scan",
Short: "Scan for installed LSPs",
RunE: runLSPScan,
})
lspCmd.AddCommand(&cobra.Command{
Use: "install [name]",
Short: "Install LSP server(s)",
Args: cobra.RangeArgs(0, 1),
RunE: runLSPInstall,
})
}
func runLSPScan(cmd *cobra.Command, args []string) error {
servers := lsp.ScanServers()
fmt.Printf("%-25s %-15s %-10s\n", "Name", "Language", "Status")
fmt.Println("──────────────────────────────────────────")
for _, s := range servers {
status := "✗ missing"
if s.Installed {
status = "✓ installed"
}
fmt.Printf("%-25s %-15s %-10s\n", s.Name, s.Language, status)
}
return nil
}
func runLSPInstall(cmd *cobra.Command, args []string) error {
if len(args) == 0 {
return fmt.Errorf("server name required")
}
name := args[0]
fmt.Printf("Installing %s...\n", name)
if err := lsp.InstallServer(name); err != nil {
return err
}
fmt.Printf("✓ %s installed\n", name)
return nil
}

54
cmd/muyue/commands/mcp.go Normal file
View File

@@ -0,0 +1,54 @@
package commands
import (
"fmt"
"github.com/muyue/muyue/internal/config"
"github.com/muyue/muyue/internal/mcp"
"github.com/spf13/cobra"
)
var mcpCmd = &cobra.Command{
Use: "mcp",
Short: "MCP management",
}
func init() {
rootCmd.AddCommand(mcpCmd)
mcpCmd.AddCommand(&cobra.Command{
Use: "config",
Short: "Generate MCP configs for Crush + Claude Code",
RunE: runMCPConfig,
})
mcpCmd.AddCommand(&cobra.Command{
Use: "scan",
Short: "Scan available MCP servers",
RunE: runMCPScan,
})
}
func runMCPConfig(cmd *cobra.Command, args []string) error {
cfg, err := config.Load()
if err != nil {
return err
}
if err := mcp.ConfigureAll(cfg); err != nil {
return err
}
fmt.Println("MCP configs generated for Crush and Claude Code")
return nil
}
func runMCPScan(cmd *cobra.Command, args []string) error {
servers := mcp.ScanServers()
fmt.Printf("%-25s %-15s %-10s\n", "Name", "Category", "Status")
fmt.Println("──────────────────────────────────────────")
for _, s := range servers {
status := "✗ missing"
if s.Installed {
status = "✓ installed"
}
fmt.Printf("%-25s %-15s %-10s\n", s.Name, s.Category, status)
}
return nil
}

View File

@@ -0,0 +1,66 @@
package commands
import (
"fmt"
"os"
"github.com/muyue/muyue/internal/config"
"github.com/muyue/muyue/internal/desktop"
"github.com/muyue/muyue/internal/profiler"
"github.com/spf13/cobra"
)
var rootCmd = &cobra.Command{
Use: "muyue",
Short: "Muyue is your AI-powered development companion",
Long: `Muyue - A modern development environment with AI assistance, tool management, and seamless desktop integration.`,
RunE: func(cmd *cobra.Command, args []string) error {
cfg := loadOrSetupConfig()
return desktop.Run(cfg, os.Args[1:])
},
}
func Execute() error {
return rootCmd.Execute()
}
func loadOrSetupConfig() *config.MuyueConfig {
if !config.Exists() {
fmt.Println("First time setup detected!")
cfg, err := profiler.RunFirstTimeSetup()
if err != nil {
fmt.Fprintf(os.Stderr, "Setup error: %v\n", err)
os.Exit(1)
}
for i := range cfg.AI.Providers {
if cfg.AI.Providers[i].Active && cfg.AI.Providers[i].APIKey == "" {
key, err := profiler.AskAPIKey(cfg.AI.Providers[i].Name)
if err == nil && key != "" {
cfg.AI.Providers[i].APIKey = key
}
}
}
if err := config.Save(cfg); err != nil {
fmt.Fprintf(os.Stderr, "Save error: %v\n", err)
os.Exit(1)
}
fmt.Println("\nSetup complete! Starting muyue...")
return cfg
}
cfg, err := config.Load()
if err != nil {
fmt.Fprintf(os.Stderr, "Config load error: %v\n", err)
os.Exit(1)
}
return cfg
}
func init() {
rootCmd.PersistentFlags().Int("port", 8080, "HTTP port for the desktop server")
rootCmd.PersistentFlags().Bool("no-open", false, "Don't open browser on startup")
}

View File

@@ -0,0 +1,56 @@
package commands
import (
"encoding/json"
"fmt"
"github.com/muyue/muyue/internal/scanner"
"github.com/spf13/cobra"
)
var scanCmd = &cobra.Command{
Use: "scan",
Short: "Run system scan and print results table",
RunE: runScan,
}
func init() {
rootCmd.AddCommand(scanCmd)
scanCmd.Flags().Bool("json", false, "Output results as JSON")
}
func runScan(cmd *cobra.Command, args []string) error {
useJSON, _ := cmd.Flags().GetBool("json")
result := scanner.ScanSystem()
if useJSON {
data, err := json.MarshalIndent(result, "", " ")
if err != nil {
return err
}
fmt.Println(string(data))
return nil
}
fmt.Printf("%-15s %-20s %-10s %-10s\n", "Tool", "Version", "Status", "Path")
fmt.Println("─────────────────────────────────────────────────")
for _, t := range result.Tools {
status := "✓ installed"
if !t.Installed {
status = "✗ missing"
}
fmt.Printf("%-15s %-20s %-10s %-10s\n", t.Name, t.Version, status, t.Path)
}
fmt.Printf("\n% d/%d tools installed\n", len(result.Tools) - countMissing(result.Tools), len(result.Tools))
return nil
}
func countMissing(tools []scanner.ToolStatus) int {
missing := 0
for _, t := range tools {
if !t.Installed {
missing++
}
}
return missing
}

View File

@@ -0,0 +1,39 @@
package commands
import (
"fmt"
"github.com/muyue/muyue/internal/config"
"github.com/muyue/muyue/internal/profiler"
"github.com/spf13/cobra"
)
var setupCmd = &cobra.Command{
Use: "setup",
Short: "Run first-run wizard (profiler)",
RunE: runSetup,
}
func init() {
rootCmd.AddCommand(setupCmd)
}
func runSetup(cmd *cobra.Command, args []string) error {
cfg, err := profiler.RunFirstTimeSetup()
if err != nil {
return err
}
for i := range cfg.AI.Providers {
if cfg.AI.Providers[i].Active && cfg.AI.Providers[i].APIKey == "" {
key, err := profiler.AskAPIKey(cfg.AI.Providers[i].Name)
if err == nil && key != "" {
cfg.AI.Providers[i].APIKey = key
}
}
}
if err := config.Save(cfg); err != nil {
return err
}
fmt.Println("Setup complete!")
return nil
}

View File

@@ -0,0 +1,105 @@
package commands
import (
"fmt"
"github.com/muyue/muyue/internal/skills"
"github.com/spf13/cobra"
)
var skillsCmd = &cobra.Command{
Use: "skills",
Short: "Skills management",
}
func init() {
rootCmd.AddCommand(skillsCmd)
skillsCmd.AddCommand(&cobra.Command{
Use: "list",
Short: "List installed skills",
RunE: runSkillsList,
})
skillsCmd.AddCommand(&cobra.Command{
Use: "init",
Short: "Install built-in skills",
RunE: runSkillsInit,
})
skillsCmd.AddCommand(&cobra.Command{
Use: "show [name]",
Short: "Show skill details",
Args: cobra.ExactArgs(1),
RunE: runSkillsShow,
})
skillsCmd.AddCommand(&cobra.Command{
Use: "generate [name] [description]",
Short: "AI-generate a skill",
Args: cobra.ExactArgs(2),
RunE: runSkillsGenerate,
})
skillsCmd.AddCommand(&cobra.Command{
Use: "deploy",
Short: "Deploy skills to Crush/Claude Code",
RunE: runSkillsDeploy,
})
skillsCmd.AddCommand(&cobra.Command{
Use: "delete [name]",
Short: "Delete a skill",
Args: cobra.ExactArgs(1),
RunE: runSkillsDelete,
})
}
func runSkillsList(cmd *cobra.Command, args []string) error {
list, err := skills.List()
if err != nil {
return err
}
if len(list) == 0 {
fmt.Println("No skills installed")
return nil
}
fmt.Printf("%-20s %-40s\n", "Name", "Description")
fmt.Println("─────────────────────────────────────────────────────")
for _, s := range list {
fmt.Printf("%-20s %-40s\n", s.Name, s.Description)
}
return nil
}
func runSkillsInit(cmd *cobra.Command, args []string) error {
fmt.Println("Initializing built-in skills...")
return nil
}
func runSkillsShow(cmd *cobra.Command, args []string) error {
name := args[0]
skill, err := skills.Get(name)
if err != nil {
return err
}
fmt.Printf("Name: %s\nDescription: %s\nAuthor: %s\nVersion: %s\n\n%s\n",
skill.Name, skill.Description, skill.Author, skill.Version, skill.Content)
return nil
}
func runSkillsGenerate(cmd *cobra.Command, args []string) error {
fmt.Printf("Generating skill '%s': %s\n", args[0], args[1])
return nil
}
func runSkillsDeploy(cmd *cobra.Command, args []string) error {
if err := skills.DeployAll(); err != nil {
return err
}
fmt.Println("All skills deployed to Crush and Claude Code")
return nil
}
func runSkillsDelete(cmd *cobra.Command, args []string) error {
name := args[0]
if err := skills.Delete(name); err != nil {
return err
}
fmt.Printf("Skill '%s' deleted\n", name)
return nil
}

View File

@@ -0,0 +1,80 @@
package commands
import (
"fmt"
"github.com/muyue/muyue/internal/scanner"
"github.com/muyue/muyue/internal/updater"
"github.com/spf13/cobra"
)
var updateCmd = &cobra.Command{
Use: "update [tool]",
Short: "Check and apply updates",
Args: cobra.RangeArgs(0, 1),
RunE: runUpdate,
}
var checkOnly bool
func init() {
rootCmd.AddCommand(updateCmd)
updateCmd.Flags().BoolVar(&checkOnly, "check", false, "Check only, don't update")
}
func runUpdate(cmd *cobra.Command, args []string) error {
result := scanner.ScanSystem()
statuses := updater.CheckUpdates(result)
if len(args) > 0 {
for _, u := range statuses {
if u.Tool == args[0] {
if u.NeedsUpdate {
fmt.Printf("%s: %s → %s\n", u.Tool, u.Current, u.Latest)
if !checkOnly {
updater.RunAutoUpdate([]updater.UpdateStatus{u})
fmt.Println("Updated!")
}
} else {
fmt.Printf("%s is up to date (%s)\n", u.Tool, u.Current)
}
return nil
}
}
fmt.Printf("Tool '%s' not found\n", args[0])
return nil
}
fmt.Printf("%-15s %-10s %-10s %-10s\n", "Tool", "Current", "Latest", "Status")
fmt.Println("─────────────────────────────────────────")
hasUpdates := false
for _, u := range statuses {
status := "✓"
if u.NeedsUpdate {
status = "⟳ update"
hasUpdates = true
}
if u.Error != "" {
status = "✗ " + u.Error
}
fmt.Printf("%-15s %-10s %-10s %-10s\n", u.Tool, u.Current, u.Latest, status)
}
if checkOnly {
return nil
}
if hasUpdates {
toUpdate := make([]updater.UpdateStatus, 0)
for _, u := range statuses {
if u.NeedsUpdate {
toUpdate = append(toUpdate, u)
}
}
updater.RunAutoUpdate(toUpdate)
fmt.Println("\nUpdates applied.")
} else {
fmt.Println("\nAll tools are up to date.")
}
return nil
}

View File

@@ -0,0 +1,23 @@
package commands
import (
"fmt"
"github.com/muyue/muyue/internal/version"
"github.com/spf13/cobra"
)
var versionCmd = &cobra.Command{
Use: "version",
Short: "Print version",
RunE: runVersion,
}
func init() {
rootCmd.AddCommand(versionCmd)
}
func runVersion(cmd *cobra.Command, args []string) error {
fmt.Printf("Muyue version %s\n", version.Version)
return nil
}

View File

@@ -4,51 +4,12 @@ import (
"fmt"
"os"
"github.com/muyue/muyue/internal/config"
"github.com/muyue/muyue/internal/desktop"
"github.com/muyue/muyue/internal/profiler"
"github.com/muyue/muyue/cmd/muyue/commands"
)
func main() {
cfg := loadOrSetupConfig()
if err := desktop.Run(cfg, os.Args[1:]); err != nil {
if err := commands.Execute(); err != nil {
fmt.Fprintf(os.Stderr, "Error: %v\n", err)
os.Exit(1)
}
}
func loadOrSetupConfig() *config.MuyueConfig {
if !config.Exists() {
fmt.Println("First time setup detected!")
cfg, err := profiler.RunFirstTimeSetup()
if err != nil {
fmt.Fprintf(os.Stderr, "Setup error: %v\n", err)
os.Exit(1)
}
for i := range cfg.AI.Providers {
if cfg.AI.Providers[i].Active && cfg.AI.Providers[i].APIKey == "" {
key, err := profiler.AskAPIKey(cfg.AI.Providers[i].Name)
if err == nil && key != "" {
cfg.AI.Providers[i].APIKey = key
}
}
}
if err := config.Save(cfg); err != nil {
fmt.Fprintf(os.Stderr, "Save error: %v\n", err)
os.Exit(1)
}
fmt.Println("\nSetup complete! Starting muyue...")
return cfg
}
cfg, err := config.Load()
if err != nil {
fmt.Fprintf(os.Stderr, "Config load error: %v\n", err)
os.Exit(1)
}
return cfg
}
}