From d60522abc201f351989a7a6d425712cd2840651d Mon Sep 17 00:00:00 2001 From: tradulescu Date: Sat, 23 Sep 2023 23:56:25 +0200 Subject: [PATCH 1/7] calendar parsing url support --- calendar.go | 18 ++++++++++++++++++ calendar_test.go | 15 ++++++++++++++- 2 files changed, 32 insertions(+), 1 deletion(-) diff --git a/calendar.go b/calendar.go index 5b3d7c4..43aaa42 100644 --- a/calendar.go +++ b/calendar.go @@ -6,6 +6,7 @@ import ( "errors" "fmt" "io" + "net/http" "time" ) @@ -436,6 +437,23 @@ func (calendar *Calendar) Events() (r []*VEvent) { return } +func ParseCalendarFromUrl(url string) (*Calendar, error) { + http_client := &http.Client{} + + req, err := http.NewRequest("GET", url, nil) + if err != nil { + return nil, err + } + + resp, err := http_client.Do(req) + if err != nil { + return nil, err + } + defer resp.Body.Close() + + return ParseCalendar(resp.Body) +} + func ParseCalendar(r io.Reader) (*Calendar, error) { state := "begin" c := &Calendar{} diff --git a/calendar_test.go b/calendar_test.go index abbd39b..f008406 100644 --- a/calendar_test.go +++ b/calendar_test.go @@ -1,7 +1,6 @@ package ics import ( - "github.com/stretchr/testify/assert" "io" "io/ioutil" "os" @@ -11,6 +10,8 @@ import ( "testing" "time" "unicode/utf8" + + "github.com/stretchr/testify/assert" ) func TestTimeParsing(t *testing.T) { @@ -360,3 +361,15 @@ func TestIssue52(t *testing.T) { t.Fatalf("cannot read test directory: %v", err) } } + +func TestIssue77(t *testing.T) { + url := "https://proseconsult.umontpellier.fr/jsp/custom/modules/plannings/direct_cal.jsp?data=58c99062bab31d256bee14356aca3f2423c0f022cb9660eba051b2653be722c4c7f281e4e3ad06b85d3374100ac416a4dc5c094f7d1a811b903031bde802c7f50e0bd1077f9461bed8f9a32b516a3c63525f110c026ed6da86f487dd451ca812c1c60bb40b1502b6511435cf9908feb2166c54e36382c1aa3eb0ff5cb8980cdb,1" + + cal, err := ParseCalendarFromUrl(url) + + if err != nil { + t.Fatalf("Error reading file: %s", err) + } + + t.Log(cal.CalendarProperties) +} From dadd2901aeea6c3befd881444752ca04cd329847 Mon Sep 17 00:00:00 2001 From: tradulescu Date: Sun, 24 Sep 2023 00:03:17 +0200 Subject: [PATCH 2/7] usage example, README --- README.md | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 5c4122d..9798ed0 100644 --- a/README.md +++ b/README.md @@ -6,13 +6,18 @@ A ICS / ICal parser and serialiser for Golang. Because the other libraries didn't quite do what I needed. Usage, parsing: -``` +```golang cal, err := ParseCalendar(strings.NewReader(input)) ``` -Creating: +Usage, parsing from a URL : +```golang + cal, err := ParseCalendar("an-ics-url") ``` + +Creating: +```golang cal := ics.NewCalendar() cal.SetMethod(ics.MethodRequest) event := cal.AddEvent(fmt.Sprintf("id@domain", p.SessionKey.IntID())) From 6d9f3441c2b26f801b51359eeedcb4dcc6717951 Mon Sep 17 00:00:00 2001 From: Courtcircuits Date: Tue, 24 Oct 2023 15:16:23 +0200 Subject: [PATCH 3/7] added functionnal option pattern for url parsing --- calendar.go | 41 ++++++++++++++++++++++++++++++++++++----- 1 file changed, 36 insertions(+), 5 deletions(-) diff --git a/calendar.go b/calendar.go index 43aaa42..0d99af3 100644 --- a/calendar.go +++ b/calendar.go @@ -437,15 +437,46 @@ func (calendar *Calendar) Events() (r []*VEvent) { return } -func ParseCalendarFromUrl(url string) (*Calendar, error) { - http_client := &http.Client{} +type OnlineParsingOpts struct { + client *http.Client + req *http.Request +} + +type OnlineParsingOptsFunc func(*OnlineParsingOpts) +func defaultOnlineParsingOpts(url string) *OnlineParsingOpts { req, err := http.NewRequest("GET", url, nil) - if err != nil { - return nil, err + if err != nil { //shouldn't return an error because NewRequest throws an error when context equals nil or http method doesn't exist. GET exists and the context is Background + panic(err) // so it should never panic + } + return &OnlineParsingOpts{ + client: &http.Client{}, + req: req, + } +} + +func withCustomClient(client *http.Client) OnlineParsingOptsFunc { + return func(opo *OnlineParsingOpts) { + opo.client = client } +} + +func withCustomRequest(request *http.Request) OnlineParsingOptsFunc { + return func(opo *OnlineParsingOpts) { + opo.req = request + } +} + +func ParseCalendarFromUrl(url string, opts ...OnlineParsingOptsFunc) (*Calendar, error) { + calendar_opts := defaultOnlineParsingOpts(url) + for _, fn := range opts { + fn(calendar_opts) + } + return parseCalendarFromUrl(calendar_opts.client, calendar_opts.req) +} - resp, err := http_client.Do(req) +func parseCalendarFromUrl(client *http.Client, request *http.Request) (*Calendar, error) { + resp, err := client.Do(request) if err != nil { return nil, err } From 59af956462b8ce050dfd85e2e34154d91a76e817 Mon Sep 17 00:00:00 2001 From: Arran Ubels Date: Sat, 28 Sep 2024 18:34:28 +1000 Subject: [PATCH 4/7] Resynced and moved some functions around https://github.com/arran4/golang-ical/pull/78 --- calendar.go | 16 ++++++++++++++++ components.go | 44 -------------------------------------------- 2 files changed, 16 insertions(+), 44 deletions(-) diff --git a/calendar.go b/calendar.go index 49f16dc..f273931 100644 --- a/calendar.go +++ b/calendar.go @@ -445,6 +445,22 @@ func (calendar *Calendar) Events() (r []*VEvent) { return } +func (calendar *Calendar) RemoveEvent(id string) { + for i := range calendar.Components { + switch event := calendar.Components[i].(type) { + case *VEvent: + if event.Id() == id { + if len(calendar.Components) > i+1 { + calendar.Components = append(calendar.Components[:i], calendar.Components[i+1:]...) + } else { + calendar.Components = calendar.Components[:i] + } + return + } + } + } +} + type OnlineParsingOpts struct { client *http.Client req *http.Request diff --git a/components.go b/components.go index 29e3916..12b6cc7 100644 --- a/components.go +++ b/components.go @@ -420,50 +420,6 @@ func (event *VEvent) Serialize() string { return b.String() } -func NewEvent(uniqueId string) *VEvent { - e := &VEvent{ - NewComponent(uniqueId), - } - return e -} - -func (calendar *Calendar) AddEvent(id string) *VEvent { - e := NewEvent(id) - calendar.Components = append(calendar.Components, e) - return e -} - -func (calendar *Calendar) AddVEvent(e *VEvent) { - calendar.Components = append(calendar.Components, e) -} - -func (calendar *Calendar) RemoveEvent(id string) { - for i := range calendar.Components { - switch event := calendar.Components[i].(type) { - case *VEvent: - if event.Id() == id { - if len(calendar.Components) > i+1 { - calendar.Components = append(calendar.Components[:i], calendar.Components[i+1:]...) - } else { - calendar.Components = calendar.Components[:i] - } - return - } - } - } -} - -func (calendar *Calendar) Events() (r []*VEvent) { - r = []*VEvent{} - for i := range calendar.Components { - switch event := calendar.Components[i].(type) { - case *VEvent: - r = append(r, event) - } - } - return -} - func (event *VEvent) SetEndAt(t time.Time, props ...PropertyParameter) { event.SetProperty(ComponentPropertyDtEnd, t.UTC().Format(icalTimestampFormatUtc), props...) } From 3719f659aaf12f2a83ab35415880ac8df2334439 Mon Sep 17 00:00:00 2001 From: Arran Ubels Date: Sat, 28 Sep 2024 23:16:45 +1000 Subject: [PATCH 5/7] Test fixed. --- calendar.go | 91 +++++++++++++++++++++++++++++------------------- calendar_test.go | 27 ++++++++++++-- 2 files changed, 79 insertions(+), 39 deletions(-) diff --git a/calendar.go b/calendar.go index f273931..be67ed6 100644 --- a/calendar.go +++ b/calendar.go @@ -3,10 +3,12 @@ package ics import ( "bufio" "bytes" + "context" "errors" "fmt" "io" "net/http" + "reflect" "time" ) @@ -461,52 +463,69 @@ func (calendar *Calendar) RemoveEvent(id string) { } } -type OnlineParsingOpts struct { - client *http.Client - req *http.Request -} - -type OnlineParsingOptsFunc func(*OnlineParsingOpts) - -func defaultOnlineParsingOpts(url string) *OnlineParsingOpts { - req, err := http.NewRequest("GET", url, nil) - if err != nil { //shouldn't return an error because NewRequest throws an error when context equals nil or http method doesn't exist. GET exists and the context is Background - panic(err) // so it should never panic - } - return &OnlineParsingOpts{ - client: &http.Client{}, - req: req, +func WithCustomClient(client *http.Client) *http.Client { + return client +} + +func WithCustomRequest(request *http.Request) *http.Request { + return request +} + +func ParseCalendarFromUrl(url string, opts ...any) (*Calendar, error) { + var ctx context.Context + var req *http.Request + var client HttpClientLike = http.DefaultClient + for opti, opt := range opts { + switch opt := opt.(type) { + case *http.Client: + client = opt + case HttpClientLike: + client = opt + case func() *http.Client: + client = opt() + case *http.Request: + req = opt + case func() *http.Request: + req = opt() + case context.Context: + ctx = opt + case func() context.Context: + ctx = opt() + default: + return nil, fmt.Errorf("unknown optional argument %d on ParseCalendarFromUrl: %s", opti, reflect.TypeOf(opt)) + } } -} - -func withCustomClient(client *http.Client) OnlineParsingOptsFunc { - return func(opo *OnlineParsingOpts) { - opo.client = client + if ctx == nil { + ctx = context.Background() } -} - -func withCustomRequest(request *http.Request) OnlineParsingOptsFunc { - return func(opo *OnlineParsingOpts) { - opo.req = request + if req == nil { + var err error + req, err = http.NewRequestWithContext(ctx, "GET", url, nil) + if err != nil { + return nil, fmt.Errorf("creating http request: %w", err) + } } + return parseCalendarFromHttpRequest(client, req) } -func ParseCalendarFromUrl(url string, opts ...OnlineParsingOptsFunc) (*Calendar, error) { - calendar_opts := defaultOnlineParsingOpts(url) - for _, fn := range opts { - fn(calendar_opts) - } - return parseCalendarFromUrl(calendar_opts.client, calendar_opts.req) +type HttpClientLike interface { + Do(req *http.Request) (*http.Response, error) } -func parseCalendarFromUrl(client *http.Client, request *http.Request) (*Calendar, error) { +func parseCalendarFromHttpRequest(client HttpClientLike, request *http.Request) (*Calendar, error) { resp, err := client.Do(request) if err != nil { - return nil, err + return nil, fmt.Errorf("http request: %w", err) } - defer resp.Body.Close() - - return ParseCalendar(resp.Body) + defer func(closer io.ReadCloser) { + if derr := closer.Close(); derr != nil && err == nil { + err = fmt.Errorf("http request close: %w", derr) + } + }(resp.Body) + var cal *Calendar + cal, err = ParseCalendar(resp.Body) + // This allows the defer func to change the error + return cal, err } func ParseCalendar(r io.Reader) (*Calendar, error) { diff --git a/calendar_test.go b/calendar_test.go index c61e961..8477bf5 100644 --- a/calendar_test.go +++ b/calendar_test.go @@ -1,7 +1,10 @@ package ics import ( + "bytes" + _ "embed" "io" + "net/http" "os" "path/filepath" "regexp" @@ -411,14 +414,32 @@ func TestIssue52(t *testing.T) { } } +type MockHttpClient struct { + Response *http.Response + Error error +} + +func (m *MockHttpClient) Do(req *http.Request) (*http.Response, error) { + return m.Response, m.Error +} + +var ( + _ HttpClientLike = &MockHttpClient{} + //go:embed "testdata/rfc5545sec4/input1.ics" + input1TestData []byte +) + func TestIssue77(t *testing.T) { url := "https://proseconsult.umontpellier.fr/jsp/custom/modules/plannings/direct_cal.jsp?data=58c99062bab31d256bee14356aca3f2423c0f022cb9660eba051b2653be722c4c7f281e4e3ad06b85d3374100ac416a4dc5c094f7d1a811b903031bde802c7f50e0bd1077f9461bed8f9a32b516a3c63525f110c026ed6da86f487dd451ca812c1c60bb40b1502b6511435cf9908feb2166c54e36382c1aa3eb0ff5cb8980cdb,1" - cal, err := ParseCalendarFromUrl(url) + _, err := ParseCalendarFromUrl(url, &MockHttpClient{ + Response: &http.Response{ + StatusCode: 200, + Body: io.NopCloser(bytes.NewReader(input1TestData)), + }, + }) if err != nil { t.Fatalf("Error reading file: %s", err) } - - t.Log(cal.CalendarProperties) } From bdf47c9f12ffa600e67b257c9f1f115b63a1f206 Mon Sep 17 00:00:00 2001 From: Arran Ubels Date: Tue, 15 Oct 2024 18:10:58 +1100 Subject: [PATCH 6/7] Some multiple-ness. --- calendar.go | 103 ++++++++++++++++++++++++++++++++++++++++++++++++++ components.go | 72 +++++++++++++++++++++++++++++++++-- 2 files changed, 172 insertions(+), 3 deletions(-) diff --git a/calendar.go b/calendar.go index cf2c3c3..612b777 100644 --- a/calendar.go +++ b/calendar.go @@ -61,8 +61,111 @@ const ( ComponentPropertyTzid = ComponentProperty(PropertyTzid) ComponentPropertyComment = ComponentProperty(PropertyComment) ComponentPropertyRelatedTo = ComponentProperty(PropertyRelatedTo) + ComponentPropertyMethod = ComponentProperty(PropertyMethod) + ComponentPropertyRecurrenceId = ComponentProperty(PropertyRecurrenceId) + ComponentPropertyDuration = ComponentProperty(PropertyDuration) + ComponentPropertyContact = ComponentProperty(PropertyContact) + ComponentPropertyRequestStatus = ComponentProperty(PropertyRequestStatus) + ComponentPropertyRDate = ComponentProperty(PropertyRdate) ) +// Required returns the rules from the RFC as to if they are required or not for any particular component type +// If unspecified or incomplete, it returns false. -- This list is incomplete verify source. Happy to take PRs with reference +// iana-prop and x-props are not covered as it would always be true and require an exhaustive list. +func (cp ComponentProperty) Required(c Component) bool { + // https://www.rfc-editor.org/rfc/rfc5545#section-3.6.1 + switch cp { + case ComponentPropertyDtstamp, ComponentPropertyUniqueId: + switch c.(type) { + case *VEvent: + return true + } + case ComponentPropertyDtStart: + switch c := c.(type) { + case *VEvent: + return !c.HasProperty(ComponentPropertyMethod) + } + } + return false +} + +// Exclusive returns the ComponentProperty's using the rules from the RFC as to if one or more existing properties are prohibiting this one +// If unspecified or incomplete, it returns false. -- This list is incomplete verify source. Happy to take PRs with reference +// iana-prop and x-props are not covered as it would always be true and require an exhaustive list. +func (cp ComponentProperty) Exclusive(c Component) []ComponentProperty { + // https://www.rfc-editor.org/rfc/rfc5545#section-3.6.1 + switch cp { + case ComponentPropertyDtEnd: + switch c := c.(type) { + case *VEvent: + if c.HasProperty(ComponentPropertyDuration) { + return []ComponentProperty{ComponentPropertyDuration} + } + } + case ComponentPropertyDuration: + switch c := c.(type) { + case *VEvent: + if c.HasProperty(ComponentPropertyDtEnd) { + return []ComponentProperty{ComponentPropertyDtEnd} + } + } + } + return nil +} + +// Singular returns the rules from the RFC as to if the spec states that if "Must not occur more than once" +// iana-prop and x-props are not covered as it would always be true and require an exhaustive list. +func (cp ComponentProperty) Singular(c Component) bool { + // https://www.rfc-editor.org/rfc/rfc5545#section-3.6.1 + switch cp { + case ComponentPropertyClass, ComponentPropertyCreated, ComponentPropertyDescription, ComponentPropertyGeo, + ComponentPropertyLastModified, ComponentPropertyLocation, ComponentPropertyOrganizer, ComponentPropertyPriority, + ComponentPropertySequence, ComponentPropertyStatus, ComponentPropertySummary, ComponentPropertyTransp, + ComponentPropertyUrl, ComponentPropertyRecurrenceId: + switch c.(type) { + case *VEvent: + return true + } + } + return false +} + +// Optional returns the rules from the RFC as to if the spec states that if these are optional +// iana-prop and x-props are not covered as it would always be true and require an exhaustive list. +func (cp ComponentProperty) Optional(c Component) bool { + // https://www.rfc-editor.org/rfc/rfc5545#section-3.6.1 + switch cp { + case ComponentPropertyClass, ComponentPropertyCreated, ComponentPropertyDescription, ComponentPropertyGeo, + ComponentPropertyLastModified, ComponentPropertyLocation, ComponentPropertyOrganizer, ComponentPropertyPriority, + ComponentPropertySequence, ComponentPropertyStatus, ComponentPropertySummary, ComponentPropertyTransp, + ComponentPropertyUrl, ComponentPropertyRecurrenceId, ComponentPropertyRrule, ComponentPropertyAttach, + ComponentPropertyAttendee, ComponentPropertyCategories, ComponentPropertyComment, + ComponentPropertyContact, ComponentPropertyExdate, ComponentPropertyRequestStatus, ComponentPropertyRelatedTo, + ComponentPropertyResources, ComponentPropertyRDate: + switch c.(type) { + case *VEvent: + return true + } + } + return false +} + +// Multiple returns the rules from the RFC as to if the spec states explicitly if multiple are allowed +// iana-prop and x-props are not covered as it would always be true and require an exhaustive list. +func (cp ComponentProperty) Multiple(c Component) bool { + // https://www.rfc-editor.org/rfc/rfc5545#section-3.6.1 + switch cp { + case ComponentPropertyAttach, ComponentPropertyAttendee, ComponentPropertyCategories, ComponentPropertyComment, + ComponentPropertyContact, ComponentPropertyExdate, ComponentPropertyRequestStatus, ComponentPropertyRelatedTo, + ComponentPropertyResources, ComponentPropertyRDate: + switch c.(type) { + case *VEvent: + return true + } + } + return false +} + type Property string const ( diff --git a/components.go b/components.go index 3f6f639..dd61f6b 100644 --- a/components.go +++ b/components.go @@ -62,6 +62,8 @@ func NewComponent(uniqueId string) ComponentBase { } } +// GetProperty returns the first match for the particular property you're after. Please consider using: +// ComponentProperty.Required to determine if GetProperty or GetProperties is more appropriate. func (cb *ComponentBase) GetProperty(componentProperty ComponentProperty) *IANAProperty { for i := range cb.Properties { if cb.Properties[i].IANAToken == string(componentProperty) { @@ -71,6 +73,31 @@ func (cb *ComponentBase) GetProperty(componentProperty ComponentProperty) *IANAP return nil } +// GetProperties returns all matches for the particular property you're after. Please consider using: +// ComponentProperty.Singular/ComponentProperty.Multiple to determine if GetProperty or GetProperties is more appropriate. +func (cb *ComponentBase) GetProperties(componentProperty ComponentProperty) []*IANAProperty { + var result []*IANAProperty + for i := range cb.Properties { + if cb.Properties[i].IANAToken == string(componentProperty) { + result = append(result, &cb.Properties[i]) + } + } + return result +} + +// HasProperty returns true if a component property is in the component. +func (cb *ComponentBase) HasProperty(componentProperty ComponentProperty) bool { + for i := range cb.Properties { + if cb.Properties[i].IANAToken == string(componentProperty) { + return true + } + } + return false +} + +// SetProperty replaces the first match for the particular property you're setting, otherwise adds it. Please consider using: +// ComponentProperty.Singular/ComponentProperty.Multiple to determine if AddProperty, SetProperty or ReplaceProperty is +// more appropriate. func (cb *ComponentBase) SetProperty(property ComponentProperty, value string, params ...PropertyParameter) { for i := range cb.Properties { if cb.Properties[i].IANAToken == string(property) { @@ -86,6 +113,17 @@ func (cb *ComponentBase) SetProperty(property ComponentProperty, value string, p cb.AddProperty(property, value, params...) } +// ReplaceProperty replaces all matches of the particular property you're setting, otherwise adds it. Returns a slice +// of removed properties. Please consider using: +// ComponentProperty.Singular/ComponentProperty.Multiple to determine if AddProperty, SetProperty or ReplaceProperty is +// more appropriate. +func (cb *ComponentBase) ReplaceProperty(property ComponentProperty, value string, params ...PropertyParameter) []IANAProperty { + removed := cb.RemoveProperty(property) + cb.AddProperty(property, value, params...) + return removed +} + +// AddProperty appends a property func (cb *ComponentBase) AddProperty(property ComponentProperty, value string, params ...PropertyParameter) { r := IANAProperty{ BaseProperty{ @@ -101,16 +139,44 @@ func (cb *ComponentBase) AddProperty(property ComponentProperty, value string, p cb.Properties = append(cb.Properties, r) } -// RemoveProperty removes from the component all properties that has -// the name passed in removeProp. -func (cb *ComponentBase) RemoveProperty(removeProp ComponentProperty) { +// RemoveProperty removes from the component all properties that is of a particular property type, returning an slice of +// removed entities +func (cb *ComponentBase) RemoveProperty(removeProp ComponentProperty) []IANAProperty { var keptProperties []IANAProperty + var removedProperties []IANAProperty for i := range cb.Properties { if cb.Properties[i].IANAToken != string(removeProp) { keptProperties = append(keptProperties, cb.Properties[i]) + } else { + removedProperties = append(removedProperties, cb.Properties[i]) + } + } + cb.Properties = keptProperties + return removedProperties +} + +// RemovePropertyByValue removes from the component all properties that has a particular property type and value, +// return a count of removed properties +func (cb *ComponentBase) RemovePropertyByValue(removeProp ComponentProperty, value string) []IANAProperty { + return cb.RemovePropertyByFunc(removeProp, func(p IANAProperty) bool { + return p.Value == value + }) +} + +// RemovePropertyByFunc removes from the component all properties that has a particular property type and the function +// remove returns true for +func (cb *ComponentBase) RemovePropertyByFunc(removeProp ComponentProperty, remove func(p IANAProperty) bool) []IANAProperty { + var keptProperties []IANAProperty + var removedProperties []IANAProperty + for i := range cb.Properties { + if cb.Properties[i].IANAToken != string(removeProp) && remove(cb.Properties[i]) { + keptProperties = append(keptProperties, cb.Properties[i]) + } else { + removedProperties = append(removedProperties, cb.Properties[i]) } } cb.Properties = keptProperties + return removedProperties } const ( From 7bc0b76f842ca210b8e3d4a0997e5858e445b34d Mon Sep 17 00:00:00 2001 From: Arran Ubels Date: Tue, 15 Oct 2024 18:19:08 +1100 Subject: [PATCH 7/7] Duplication issue fixed. --- calendar.go | 11 -------- components.go | 77 +++++++++++++-------------------------------------- 2 files changed, 20 insertions(+), 68 deletions(-) diff --git a/calendar.go b/calendar.go index a180958..cab2c48 100644 --- a/calendar.go +++ b/calendar.go @@ -415,17 +415,6 @@ func (cal *Calendar) setProperty(property Property, value string, params ...Prop cal.CalendarProperties = append(cal.CalendarProperties, r) } -func NewEvent(uniqueId string) *VEvent { - e := &VEvent{ - ComponentBase{ - Properties: []IANAProperty{ - {BaseProperty{IANAToken: ToText(string(ComponentPropertyUniqueId)), Value: uniqueId}}, - }, - }, - } - return e -} - func (calendar *Calendar) AddEvent(id string) *VEvent { e := NewEvent(id) calendar.Components = append(calendar.Components, e) diff --git a/components.go b/components.go index f8668e6..dead939 100644 --- a/components.go +++ b/components.go @@ -458,13 +458,13 @@ type VEvent struct { ComponentBase } -func (c *VEvent) SerializeTo(w io.Writer) { - c.ComponentBase.serializeThis(w, "VEVENT") +func (event *VEvent) SerializeTo(w io.Writer) { + event.ComponentBase.serializeThis(w, "VEVENT") } -func (c *VEvent) Serialize() string { +func (event *VEvent) Serialize() string { b := &bytes.Buffer{} - c.ComponentBase.serializeThis(b, "VEVENT") + event.ComponentBase.serializeThis(b, "VEVENT") return b.String() } @@ -475,43 +475,6 @@ func NewEvent(uniqueId string) *VEvent { return e } -func (cal *Calendar) AddEvent(id string) *VEvent { - e := NewEvent(id) - cal.Components = append(cal.Components, e) - return e -} - -func (cal *Calendar) AddVEvent(e *VEvent) { - cal.Components = append(cal.Components, e) -} - -func (cal *Calendar) RemoveEvent(id string) { - for i := range cal.Components { - switch event := cal.Components[i].(type) { - case *VEvent: - if event.Id() == id { - if len(cal.Components) > i+1 { - cal.Components = append(cal.Components[:i], cal.Components[i+1:]...) - } else { - cal.Components = cal.Components[:i] - } - return - } - } - } -} - -func (cal *Calendar) Events() []*VEvent { - var r []*VEvent - for i := range cal.Components { - switch event := cal.Components[i].(type) { - case *VEvent: - r = append(r, event) - } - } - return r -} - func (event *VEvent) SetEndAt(t time.Time, props ...PropertyParameter) { event.SetProperty(ComponentPropertyDtEnd, t.UTC().Format(icalTimestampFormatUtc), props...) } @@ -520,32 +483,32 @@ func (event *VEvent) SetLastModifiedAt(t time.Time, props ...PropertyParameter) event.SetProperty(ComponentPropertyLastModified, t.UTC().Format(icalTimestampFormatUtc), props...) } -func (c *VEvent) SetGeo(lat interface{}, lng interface{}, params ...PropertyParameter) { - c.setGeo(lat, lng, params...) +func (event *VEvent) SetGeo(lat interface{}, lng interface{}, params ...PropertyParameter) { + event.setGeo(lat, lng, params...) } -func (c *VEvent) SetPriority(p int, params ...PropertyParameter) { - c.setPriority(p, params...) +func (event *VEvent) SetPriority(p int, params ...PropertyParameter) { + event.setPriority(p, params...) } -func (c *VEvent) SetResources(r string, params ...PropertyParameter) { - c.setResources(r, params...) +func (event *VEvent) SetResources(r string, params ...PropertyParameter) { + event.setResources(r, params...) } -func (c *VEvent) AddAlarm() *VAlarm { - return c.addAlarm() +func (event *VEvent) AddAlarm() *VAlarm { + return event.addAlarm() } -func (c *VEvent) AddVAlarm(a *VAlarm) { - c.addVAlarm(a) +func (event *VEvent) AddVAlarm(a *VAlarm) { + event.addVAlarm(a) } -func (c *VEvent) Alarms() []*VAlarm { - return c.alarms() +func (event *VEvent) Alarms() []*VAlarm { + return event.alarms() } -func (c *VEvent) GetAllDayEndAt() (time.Time, error) { - return c.getTimeProp(ComponentPropertyDtEnd, true) +func (event *VEvent) GetAllDayEndAt() (time.Time, error) { + return event.getTimeProp(ComponentPropertyDtEnd, true) } type TimeTransparency string @@ -555,8 +518,8 @@ const ( TransparencyTransparent TimeTransparency = "TRANSPARENT" ) -func (c *VEvent) SetTimeTransparency(v TimeTransparency, params ...PropertyParameter) { - c.SetProperty(ComponentPropertyTransp, string(v), params...) +func (event *VEvent) SetTimeTransparency(v TimeTransparency, params ...PropertyParameter) { + event.SetProperty(ComponentPropertyTransp, string(v), params...) } type VTodo struct {