1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
|
package api
import (
"context"
"fmt"
"log"
"sort"
"strings"
"time"
"task-dashboard/internal/models"
"google.golang.org/api/calendar/v3"
"google.golang.org/api/option"
)
type GoogleCalendarClient struct {
srv *calendar.Service
calendarIDs []string
}
// NewGoogleCalendarClient creates a client that fetches from multiple calendars.
// calendarIDs can be comma-separated (e.g., "cal1@group.calendar.google.com,cal2@group.calendar.google.com")
func NewGoogleCalendarClient(ctx context.Context, credentialsFile, calendarIDs string) (*GoogleCalendarClient, error) {
srv, err := calendar.NewService(ctx, option.WithCredentialsFile(credentialsFile))
if err != nil {
return nil, fmt.Errorf("unable to retrieve Calendar client: %v", err)
}
// Parse comma-separated calendar IDs
ids := strings.Split(calendarIDs, ",")
var trimmedIDs []string
for _, id := range ids {
if trimmed := strings.TrimSpace(id); trimmed != "" {
trimmedIDs = append(trimmedIDs, trimmed)
}
}
return &GoogleCalendarClient{
srv: srv,
calendarIDs: trimmedIDs,
}, nil
}
func (c *GoogleCalendarClient) GetUpcomingEvents(ctx context.Context, maxResults int) ([]models.CalendarEvent, error) {
t := time.Now().Format(time.RFC3339)
var allEvents []models.CalendarEvent
for _, calendarID := range c.calendarIDs {
events, err := c.srv.Events.List(calendarID).ShowDeleted(false).
SingleEvents(true).TimeMin(t).MaxResults(int64(maxResults)).OrderBy("startTime").Do()
if err != nil {
log.Printf("Warning: failed to fetch events from calendar %s: %v", calendarID, err)
continue // Don't fail entirely, just skip this calendar
}
for _, item := range events.Items {
var start, end time.Time
if item.Start.DateTime == "" {
// All-day event
start, _ = time.Parse("2006-01-02", item.Start.Date)
end, _ = time.Parse("2006-01-02", item.End.Date)
} else {
start, _ = time.Parse(time.RFC3339, item.Start.DateTime)
end, _ = time.Parse(time.RFC3339, item.End.DateTime)
}
allEvents = append(allEvents, models.CalendarEvent{
ID: item.Id,
Summary: item.Summary,
Description: item.Description,
Start: start,
End: end,
HTMLLink: item.HtmlLink,
})
}
}
// Sort all events by start time
sort.Slice(allEvents, func(i, j int) bool {
return allEvents[i].Start.Before(allEvents[j].Start)
})
// Limit to maxResults
if len(allEvents) > maxResults {
allEvents = allEvents[:maxResults]
}
return allEvents, nil
}
|