app.go

 1package app
 2
 3import (
 4	"context"
 5	"database/sql"
 6	"maps"
 7	"sync"
 8	"time"
 9
10	"github.com/kujtimiihoxha/termai/internal/db"
11	"github.com/kujtimiihoxha/termai/internal/history"
12	"github.com/kujtimiihoxha/termai/internal/llm/agent"
13	"github.com/kujtimiihoxha/termai/internal/logging"
14	"github.com/kujtimiihoxha/termai/internal/lsp"
15	"github.com/kujtimiihoxha/termai/internal/message"
16	"github.com/kujtimiihoxha/termai/internal/permission"
17	"github.com/kujtimiihoxha/termai/internal/session"
18)
19
20type App struct {
21	Sessions    session.Service
22	Messages    message.Service
23	Files       history.Service
24	Permissions permission.Service
25
26	CoderAgent agent.Service
27
28	LSPClients map[string]*lsp.Client
29
30	clientsMutex sync.RWMutex
31
32	watcherCancelFuncs []context.CancelFunc
33	cancelFuncsMutex   sync.Mutex
34	watcherWG          sync.WaitGroup
35}
36
37func New(ctx context.Context, conn *sql.DB) (*App, error) {
38	q := db.New(conn)
39	sessions := session.NewService(q)
40	messages := message.NewService(q)
41	files := history.NewService(q)
42
43	app := &App{
44		Sessions:    sessions,
45		Messages:    messages,
46		Files:       files,
47		Permissions: permission.NewPermissionService(),
48		LSPClients:  make(map[string]*lsp.Client),
49	}
50
51	app.initLSPClients(ctx)
52
53	var err error
54	app.CoderAgent, err = agent.NewCoderAgent(
55		app.Permissions,
56		app.Sessions,
57		app.Messages,
58		app.LSPClients,
59	)
60	if err != nil {
61		logging.Error("Failed to create coder agent", err)
62		return nil, err
63	}
64
65	return app, nil
66}
67
68// Shutdown performs a clean shutdown of the application
69func (app *App) Shutdown() {
70	// Cancel all watcher goroutines
71	app.cancelFuncsMutex.Lock()
72	for _, cancel := range app.watcherCancelFuncs {
73		cancel()
74	}
75	app.cancelFuncsMutex.Unlock()
76	app.watcherWG.Wait()
77
78	// Perform additional cleanup for LSP clients
79	app.clientsMutex.RLock()
80	clients := make(map[string]*lsp.Client, len(app.LSPClients))
81	maps.Copy(clients, app.LSPClients)
82	app.clientsMutex.RUnlock()
83
84	for name, client := range clients {
85		shutdownCtx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
86		if err := client.Shutdown(shutdownCtx); err != nil {
87			logging.Error("Failed to shutdown LSP client", "name", name, "error", err)
88		}
89		cancel()
90	}
91}