-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcreate_meeting.go
126 lines (106 loc) · 3.5 KB
/
create_meeting.go
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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
package whereby
import (
"bytes"
"context"
"encoding/json"
"errors"
"fmt"
"net/http"
"strings"
"time"
)
type createMeetingPayload struct {
IsLocked bool `json:"isLocked"`
RoomNamePrefix string `json:"roomNamePrefix,omitempty"`
RoomNamePattern string `json:"roomNamePattern,omitempty"`
RoomMode string `json:"roomMode,omitempty"`
Start string `json:"startDate,omitempty"`
End string `json:"endDate"`
Fields []string `json:"fields,omitempty"`
}
type CreateMeetingOutput = GetMeetingOutput
// CreateMeeting creates a meeting as specified. It will also create a transient
// room that is guaranteed to be available for specified start and end time.
// Some time after the meeting has ended, the transient room will be
// automatically deleted. The URL to this room is present in the response.
//
// See https://whereby.dev/http-api/#/paths/~1meetings/post for more details.
func (c *Client) CreateMeeting(ctx context.Context, input CreateMeetingInput) (CreateMeetingOutput, error) {
var out CreateMeetingOutput
if err := validateCreateMeetingInput(input); err != nil {
return out, err
}
payload, err := json.Marshal(c.getCreateMeetingInput(input))
if err != nil {
return out, fmt.Errorf("failed to encode request body: %w", err)
}
req, err := http.NewRequestWithContext(ctx, http.MethodPost, createMeetingEndpoint, bytes.NewBuffer(payload))
if err != nil {
return out, fmt.Errorf("failed create request: %w", err)
}
req.Header.Set("content-type", "application/json")
res, err := c.do(req)
if err != nil {
return out, fmt.Errorf("failed to make request to the Whereby API: %w", err)
}
if res.StatusCode < 200 || res.StatusCode > 299 {
return out, handleBadStatus(res)
}
var innerRes meeting
if err := json.NewDecoder(res.Body).Decode(&innerRes); err != nil {
return out, fmt.Errorf("failed to decode payload from Whereby: %w", err)
}
if err := createGetMeetingOutput(&out, innerRes); err != nil {
return out, err
}
return out, nil
}
// createGetMeetingOutput creates the user-friendly output object from the
// internal JSON representation.
func createGetMeetingOutput(dst *GetMeetingOutput, src meeting) error {
dst.MeetingID = src.MeetingId
dst.URL = src.RoomURL
if hu := src.HostRoomURL; hu != nil {
dst.HostURL = *hu
}
if sd := src.StartDate; sd != nil && *sd != "" {
start, err := time.Parse(time.RFC3339, *sd)
if err != nil {
return err
}
dst.Start = start
}
end, err := time.Parse(time.RFC3339, src.EndDate)
if err != nil {
return err
}
dst.End = end
return nil
}
// getCreateMeetingInput converts the CreateMeetingInput object into the inner
// representation for JSON marshalling.
func (c *Client) getCreateMeetingInput(in CreateMeetingInput) createMeetingPayload {
var out createMeetingPayload
out.IsLocked = in.IsLocked
out.RoomNamePrefix = in.RoomNamePrefix
out.RoomNamePattern = string(in.RoomNamePattern)
out.RoomMode = string(in.RoomMode)
out.Start = in.Start.Format(time.RFC3339)
out.End = in.End.Format(time.RFC3339)
if in.WithHostURL {
out.Fields = append(out.Fields, "hostRoomUrl")
}
return out
}
// validateCreateMeetingInput validates the provided CreateMeetingInput.
func validateCreateMeetingInput(input CreateMeetingInput) error {
if input.RoomNamePrefix != "" {
if strings.ToLower(input.RoomNamePrefix) != input.RoomNamePrefix {
return errors.New("room name should be lowercase")
}
}
if input.End.IsZero() {
return errors.New("meeting end time must be specified")
}
return nil
}