Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adding support for transcripts, recording, AI summarization and meeting subscription to channels #377

Open
wants to merge 42 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
db76929
Adding support for transcripts, recording and AI summarization
jespino May 9, 2024
a305b43
WIP
jespino May 10, 2024
dd47bbf
Adding chat support
jespino May 10, 2024
12815af
WIP
jespino May 10, 2024
161b660
Adding support for subscription of meetings to channels
jespino May 15, 2024
467399e
Removing debug log messages
jespino May 15, 2024
e448589
Fixing tests
jespino May 15, 2024
df60c28
fixing linter errors
jespino May 15, 2024
bc696fd
Fixing linter errors
jespino May 15, 2024
513006a
Addressing PR review comments
jespino May 16, 2024
8b30a5d
Updating webhooks subscriptions needed in the documentation
jespino May 16, 2024
a810064
Addressing PR review comments
jespino May 16, 2024
9ae260c
Addressing PR review comments
jespino May 16, 2024
42bfc3a
Avoid subscriptions to personal meetings
jespino May 16, 2024
4548448
Merge remote-tracking branch 'origin/master' into transcripts-recordi…
jespino May 16, 2024
4f6233f
Migrating to typescript
jespino May 16, 2024
0cc74d9
Migrating to typescript
jespino May 16, 2024
bf0599d
Fixing linter
jespino May 17, 2024
7777670
Adding tests for transcript and chat handlers
jespino May 17, 2024
b67ed82
Fixing some linter errors
jespino May 17, 2024
3b4d73d
Addressing some PR review comments
jespino May 23, 2024
5efacfd
Fixing types
jespino May 24, 2024
18f9706
Merge remote-tracking branch 'origin/master' into transcripts-recordi…
jespino Jul 2, 2024
0702a63
Merge remote-tracking branch 'origin/master' into transcripts-recordi…
jespino Jul 12, 2024
dd108cf
Adding the meeting UUID
jespino Jul 12, 2024
e6b0a50
Merging master and fixed problems related to the merge
jespino Jul 12, 2024
d75b8a5
Fixing a crash on subscription
jespino Jul 12, 2024
5c9a13a
Fixing the alteration on the length after receiving transcriptions/re…
jespino Jul 12, 2024
30d8b15
Merge remote-tracking branch 'origin/master' into transcripts-recordi…
jespino Sep 26, 2024
0280fc4
Merge remote-tracking branch 'origin/master' into transcripts-recordi…
jespino Jan 11, 2025
d3067af
Fixing linter error
jespino Jan 11, 2025
fc9d542
fix: Add missing mock expectation for KVGet in test case
jespino Jan 11, 2025
191ef21
fix: Add missing mock expectation for GetUser in webhook test
jespino Jan 11, 2025
0af0204
fix: Add missing mock expectation for bot user's Zoom token
jespino Jan 11, 2025
d3f70c4
fix: Add missing KVSetWithExpiry mock for meeting post ID test
jespino Jan 11, 2025
70f1af5
fix: Add missing mock expectation for KVSetWithExpiry in test
jespino Jan 11, 2025
5fe64f0
test: Add mock expectation for PublishWebSocketEvent in test case
jespino Jan 11, 2025
9dc2a3c
fix: Initialize plugin client in webhook test to resolve nil pointer …
jespino Jan 11, 2025
c90d235
fix: Initialize plugin client in TestWebhookHandleRecordingCompleted …
jespino Jan 11, 2025
fb6ae2f
Fixing tests
jespino Jan 11, 2025
e9ea52a
Fixing linter error
jespino Jan 11, 2025
a35591e
test: Add tests for `handleMeetingStarted` webhook functionality
jespino Jan 11, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
60 changes: 49 additions & 11 deletions server/command.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ package main

import (
"fmt"
"strconv"
"strings"

"github.com/mattermost/mattermost-plugin-zoom/server/zoom"
Expand All @@ -25,11 +26,13 @@ const (
)

const (
actionConnect = "connect"
actionStart = "start"
actionDisconnect = "disconnect"
actionHelp = "help"
settings = "settings"
actionConnect = "connect"
actionSubscribe = "subscribe"
actionUnsubscribe = "unsubscribe"
actionStart = "start"
actionDisconnect = "disconnect"
actionHelp = "help"
settings = "settings"
)

func (p *Plugin) getCommand() (*model.Command, error) {
Expand Down Expand Up @@ -64,7 +67,7 @@ func (p *Plugin) postCommandResponse(args *model.CommandArgs, text string) {
_ = p.API.SendEphemeralPost(args.UserId, post)
}

func (p *Plugin) parseCommand(rawCommand string) (cmd, action, topic string) {
func (p *Plugin) parseCommand(rawCommand string) (cmd, action, topic string, meetingID int) {
split := strings.Fields(rawCommand)
cmd = split[0]
if len(split) > 1 {
Expand All @@ -73,11 +76,14 @@ func (p *Plugin) parseCommand(rawCommand string) (cmd, action, topic string) {
if action == actionStart {
topic = strings.Join(split[2:], " ")
}
return cmd, action, topic
if len(split) > 2 && (action == actionSubscribe || action == actionUnsubscribe) {
meetingID, _ = strconv.Atoi(split[2])
}
return cmd, action, topic, meetingID
}

func (p *Plugin) executeCommand(c *plugin.Context, args *model.CommandArgs) (string, error) {
command, action, topic := p.parseCommand(args.Command)
command, action, topic, meetingID := p.parseCommand(args.Command)

if command != "/zoom" {
return fmt.Sprintf("Command '%s' is not /zoom. Please try again.", command), nil
Expand All @@ -96,6 +102,10 @@ func (p *Plugin) executeCommand(c *plugin.Context, args *model.CommandArgs) (str
switch action {
case actionConnect:
return p.runConnectCommand(user, args)
case actionSubscribe:
return p.runSubscribeCommand(user, args, meetingID)
jespino marked this conversation as resolved.
Show resolved Hide resolved
case actionUnsubscribe:
return p.runUnsubscribeCommand(user, args, meetingID)
case actionStart:
return p.runStartCommand(args, user, topic)
case actionDisconnect:
Expand Down Expand Up @@ -180,7 +190,12 @@ func (p *Plugin) runStartCommand(args *model.CommandArgs, user *model.User, topi
}
}

if postMeetingErr := p.postMeeting(user, meetingID, args.ChannelId, args.RootId, topic); postMeetingErr != nil {
meeting, err := p.getMeeting(user, meetingID)
if err != nil {
return "", errors.Wrap(err, "failed to get the meeting")
}

if postMeetingErr := p.postMeeting(user, meetingID, meeting.UUID, args.ChannelId, args.RootId, topic); postMeetingErr != nil {
return "", postMeetingErr
}

Expand Down Expand Up @@ -225,6 +240,23 @@ func (p *Plugin) runConnectCommand(user *model.User, extra *model.CommandArgs) (
return oauthMsg, nil
}

func (p *Plugin) runSubscribeCommand(user *model.User, extra *model.CommandArgs, meetingID int) (string, error) {
if appErr := p.storeChannelForMeeting(meetingID, extra.ChannelId); appErr != nil {
return "", errors.Wrap(appErr, "cannot subscribing to meeting")
jespino marked this conversation as resolved.
Show resolved Hide resolved
}
return "Channel subscribed to meeting", nil
}

func (p *Plugin) runUnsubscribeCommand(user *model.User, extra *model.CommandArgs, meetingID int) (string, error) {
if channelID, appErr := p.fetchChannelForMeeting(meetingID); appErr != nil || channelID == "" {
return "Can not unsubscribe from meeting: meeting not found", errors.New("meeting not found")
}
if appErr := p.deleteChannelForMeeting(meetingID); appErr != nil {
return "Can not unsubscribe from meeting: unable to delete the meeting subscription", errors.Wrap(appErr, "cannot unsubscribing from meeting")
jespino marked this conversation as resolved.
Show resolved Hide resolved
}
return "Channel unsubscribed from meeting", nil
}

// runDisconnectCommand runs command to disconnect from Zoom. Will fail if user cannot connect.
func (p *Plugin) runDisconnectCommand(user *model.User) (string, error) {
if !p.canConnect(user) {
Expand Down Expand Up @@ -285,9 +317,9 @@ func (p *Plugin) updateUserPersonalSettings(usePMIValue, userID string) *model.A
func (p *Plugin) getAutocompleteData() *model.AutocompleteData {
canConnect := !p.configuration.AccountLevelApp

available := "start, help, settings"
available := "start, help, settings, subscribe, unsubscribe"
if canConnect {
available = "start, connect, disconnect, help, settings"
available = "start, connect, disconnect, help, settings, subscribe, unsubscribe"
}

zoom := model.NewAutocompleteData("zoom", "[command]", fmt.Sprintf("Available commands: %s", available))
Expand All @@ -306,6 +338,12 @@ func (p *Plugin) getAutocompleteData() *model.AutocompleteData {
setting := model.NewAutocompleteData("settings", "", "Update your meeting ID preferences")
zoom.AddCommand(setting)

subscribe := model.NewAutocompleteData("subscribe", "[meeting id]", "Subscribe this channel to a Zoom meeting")
zoom.AddCommand(subscribe)

unsubscribe := model.NewAutocompleteData("unsubscribe", "[meeting id]", "Unsubscribe this channel from a Zoom meeting")
zoom.AddCommand(unsubscribe)

help := model.NewAutocompleteData("help", "", "Display usage")
zoom.AddCommand(help)

Expand Down
42 changes: 36 additions & 6 deletions server/http.go
Original file line number Diff line number Diff line change
Expand Up @@ -180,7 +180,13 @@ func (p *Plugin) startMeeting(action, userID, channelID, rootID string) {
}
}

if postMeetingErr := p.postMeeting(user, meetingID, channelID, rootID, defaultMeetingTopic); postMeetingErr != nil {
meeting, err := p.getMeeting(user, meetingID)
if err != nil {
p.API.LogWarn("failed to get the meeting", "Error", err.Error())
return
}

if postMeetingErr := p.postMeeting(user, meetingID, meeting.UUID, channelID, rootID, defaultMeetingTopic); postMeetingErr != nil {
p.API.LogWarn("failed to post the meeting", "Error", postMeetingErr.Error())
return
}
Expand Down Expand Up @@ -365,7 +371,8 @@ func (p *Plugin) completeUserOAuthToZoom(w http.ResponseWriter, r *http.Request)
}

meetingID := meeting.ID
if err = p.postMeeting(user, meetingID, channelID, "", ""); err != nil {
meetingUUID := meeting.UUID
if err = p.postMeeting(user, meetingID, meetingUUID, channelID, "", ""); err != nil {
p.API.LogWarn("Failed to post the meeting", "error", err.Error())
http.Error(w, err.Error(), http.StatusInternalServerError)
return
Expand All @@ -392,14 +399,14 @@ func (p *Plugin) completeUserOAuthToZoom(w http.ResponseWriter, r *http.Request)
}
}

func (p *Plugin) postMeeting(creator *model.User, meetingID int, channelID string, rootID string, topic string) error {
func (p *Plugin) postMeeting(creator *model.User, meetingID int, meetingUUID string, channelID string, rootID string, topic string) error {
meetingURL := p.getMeetingURL(creator, meetingID)

if topic == "" {
topic = defaultMeetingTopic
}

if !p.API.HasPermissionToChannel(creator.Id, channelID, model.PermissionCreatePost) {
if p.botUserID != creator.Id && !p.API.HasPermissionToChannel(creator.Id, channelID, model.PermissionCreatePost) {
return errors.New("this channel is not accessible, you might not have permissions to write in this channel. Contact the administrator of this channel to find out if you have access permissions")
}

Expand All @@ -418,6 +425,7 @@ func (p *Plugin) postMeeting(creator *model.User, meetingID int, channelID strin
Props: map[string]interface{}{
"attachments": []*model.SlackAttachment{&slackAttachment},
"meeting_id": meetingID,
"meeting_uuid": meetingUUID,
"meeting_link": meetingURL,
"meeting_status": zoom.WebhookStatusStarted,
"meeting_personal": false,
Expand All @@ -432,7 +440,7 @@ func (p *Plugin) postMeeting(creator *model.User, meetingID int, channelID strin
return appErr
}

if appErr = p.storeMeetingPostID(meetingID, createdPost.Id); appErr != nil {
if appErr = p.storeMeetingPostID(meetingUUID, createdPost.Id); appErr != nil {
p.API.LogDebug("failed to store post id", "error", appErr)
}

Expand Down Expand Up @@ -621,7 +629,14 @@ func (p *Plugin) handleStartMeeting(w http.ResponseWriter, r *http.Request) {
}
}

if err = p.postMeeting(user, meetingID, req.ChannelID, req.RootID, topic); err != nil {
meeting, err := p.getMeeting(user, meetingID)
if err != nil {
p.API.LogWarn("failed to get the meeting", "Error", err.Error())
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}

if err = p.postMeeting(user, meetingID, meeting.UUID, req.ChannelID, req.RootID, topic); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
Expand Down Expand Up @@ -655,6 +670,21 @@ func (p *Plugin) createMeetingWithoutPMI(user *model.User, zoomUser *zoom.User,
return meeting.ID, nil
}

func (p *Plugin) getMeeting(user *model.User, meetingID int) (*zoom.Meeting, error) {
client, _, err := p.getActiveClient(user)
if err != nil {
p.API.LogWarn("could not get the active zoom client", "error", err.Error())
return nil, err
}

meeting, err := client.GetMeeting(meetingID)
if err != nil {
p.API.LogDebug("failed to get meeting")
return nil, err
}
return meeting, nil
}

func (p *Plugin) getMeetingURL(user *model.User, meetingID int) string {
defaultURL := fmt.Sprintf("%s/j/%v", p.getZoomURL(), meetingID)
client, _, err := p.getActiveClient(user)
Expand Down
5 changes: 3 additions & 2 deletions server/plugin_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -61,10 +61,10 @@ func TestPlugin(t *testing.T) {
meetingRequest := httptest.NewRequest("POST", "/api/v1/meetings", strings.NewReader("{\"channel_id\": \"thechannelid\"}"))
meetingRequest.Header.Add("Mattermost-User-Id", "theuserid")

endedPayload := `{"event": "meeting.ended", "payload": {"object": {"id": "234"}}}`
endedPayload := `{"event": "meeting.ended", "payload": {"object": {"id": "234", "uuid": "234"}}}`
validStoppedWebhookRequest := httptest.NewRequest("POST", "/webhook?secret=thewebhooksecret", strings.NewReader(endedPayload))

validStartedWebhookRequest := httptest.NewRequest("POST", "/webhook?secret=thewebhooksecret", strings.NewReader(`{"event": "meeting.started"}`))
validStartedWebhookRequest := httptest.NewRequest("POST", "/webhook?secret=thewebhooksecret", strings.NewReader(`{"event": "meeting.started", "payload": {"object": {"id": "234"}}}`))

noSecretWebhookRequest := httptest.NewRequest("POST", "/webhook", strings.NewReader(endedPayload))

Expand Down Expand Up @@ -153,6 +153,7 @@ func TestPlugin(t *testing.T) {

api.On("KVGet", fmt.Sprintf("%v%v", postMeetingKey, 234)).Return([]byte("thepostid"), nil)
api.On("KVGet", fmt.Sprintf("%v%v", postMeetingKey, 123)).Return([]byte("thepostid"), nil)
api.On("KVGet", fmt.Sprintf("%v%v", meetingChannelKey, 234)).Return([]byte(""), nil)

api.On("KVDelete", fmt.Sprintf("%v%v", postMeetingKey, 234)).Return(nil)

Expand Down
35 changes: 29 additions & 6 deletions server/store.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import (

const (
postMeetingKey = "post_meeting_"
meetingChannelKey = "meeting_channel_"
zoomStateKeyPrefix = "zoomuserstate"
zoomUserByMMID = "zoomtoken_"
zoomUserByZoomID = "zoomtokenbyzoomid_"
Expand Down Expand Up @@ -134,14 +135,14 @@ func (p *Plugin) deleteUserState(userID string) *model.AppError {
return p.API.KVDelete(key)
}

func (p *Plugin) storeMeetingPostID(meetingID int, postID string) *model.AppError {
key := fmt.Sprintf("%v%v", postMeetingKey, meetingID)
func (p *Plugin) storeMeetingPostID(meetingUUID string, postID string) *model.AppError {
key := fmt.Sprintf("%v%v", postMeetingKey, meetingUUID)
bytes := []byte(postID)
return p.API.KVSetWithExpiry(key, bytes, meetingPostIDTTL)
}

func (p *Plugin) fetchMeetingPostID(meetingID string) (string, *model.AppError) {
key := fmt.Sprintf("%v%v", postMeetingKey, meetingID)
func (p *Plugin) fetchMeetingPostID(meetingUUID string) (string, *model.AppError) {
key := fmt.Sprintf("%v%v", postMeetingKey, meetingUUID)
postID, appErr := p.API.KVGet(key)
if appErr != nil {
p.API.LogDebug("Could not get meeting post from KVStore", "error", appErr.Error())
Expand All @@ -156,8 +157,30 @@ func (p *Plugin) fetchMeetingPostID(meetingID string) (string, *model.AppError)
return string(postID), nil
}

func (p *Plugin) deleteMeetingPostID(postID string) *model.AppError {
key := fmt.Sprintf("%v%v", postMeetingKey, postID)
func (p *Plugin) storeChannelForMeeting(meetingID int, channelID string) *model.AppError {
key := fmt.Sprintf("%v%v", meetingChannelKey, meetingID)
bytes := []byte(channelID)
return p.API.KVSet(key, bytes)
}

func (p *Plugin) fetchChannelForMeeting(meetingID int) (string, *model.AppError) {
key := fmt.Sprintf("%v%v", meetingChannelKey, meetingID)
channelID, appErr := p.API.KVGet(key)
if appErr != nil {
p.API.LogDebug("Could not get channel meeting from KVStore", "error", appErr.Error())
return "", appErr
}

if channelID == nil {
p.API.LogWarn("Stored channel meeting not found")
return "", appErr
}

return string(channelID), nil
}

func (p *Plugin) deleteChannelForMeeting(meetingID int) *model.AppError {
key := fmt.Sprintf("%v%v", meetingChannelKey, meetingID)
return p.API.KVDelete(key)
}

Expand Down
Loading
Loading