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

Make msteams use common recipient with filled data field #48908

Open
wants to merge 4 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
9 changes: 0 additions & 9 deletions integrations/access/common/recipient.go
Original file line number Diff line number Diff line change
Expand Up @@ -150,15 +150,6 @@ func (s *RecipientSet) ToSlice() []Recipient {
return recipientSlice
}

// GetNames returns a slice of the recipient names in the set.
func (s *RecipientSet) GetNames() []string {
names := make([]string, 0, len(s.recipients))
for _, recipient := range s.recipients {
names = append(names, recipient.Name)
}
return names
}

// ForEach applies run the given func with each recipient in the set as the argument.
func (s *RecipientSet) ForEach(f func(r Recipient)) {
for _, v := range s.recipients {
Expand Down
8 changes: 6 additions & 2 deletions integrations/access/msteams/app.go
Original file line number Diff line number Diff line change
Expand Up @@ -185,15 +185,19 @@ func (a *App) initBot(ctx context.Context) error {
a.log.InfoContext(ctx, "Preloading recipient data...")

for _, recipient := range a.conf.Recipients.GetAllRawRecipients() {
recipientData, err := a.bot.FetchRecipient(ctx, recipient)
fullRecipient, err := a.bot.FetchRecipient(ctx, recipient)
if err != nil {
return trace.Wrap(err)
}
recipientData, err := getRecipientData(fullRecipient)
if err != nil {
return trace.Wrap(err)
}
a.log.InfoContext(ctx, "Recipient and chat found",
slog.Group("recipient",
"raw", recipient,
"recipient_chat_id", recipientData.Chat.ID,
"recipient_kind", recipientData.Kind,
"recipient_kind", fullRecipient.Kind,
),
)
}
Expand Down
81 changes: 52 additions & 29 deletions integrations/access/msteams/bot.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,16 +41,12 @@ const (

type RecipientKind string

// RecipientData represents cached data for a recipient (user or channel)
type RecipientData struct {
// ID identifies the recipient, for users it is the UserID, for channels it is "tenant/group/channelName"
ID string
// recipientData represents cached data for a recipient (user or channel)
type recipientData struct {
// App installation for the recipient
App msapi.InstalledApp
// Chat for the recipient
Chat msapi.Chat
// Kind of the recipient (user or channel)
Kind RecipientKind
}

// Channel represents a MSTeams channel parsed from its web URL
Expand All @@ -74,8 +70,8 @@ type Bot struct {
botClient *msapi.BotFrameworkClient
// mu recipients access mutex
mu *sync.RWMutex
// recipients represents the cache of potential message recipients
recipients map[string]RecipientData
// fullRecipients represents the cache of potential message fullRecipients
fullRecipients map[string]common.Recipient
// webProxyURL represents Web UI address, if enabled
webProxyURL *url.URL
// clusterName cluster name
Expand Down Expand Up @@ -105,7 +101,7 @@ func NewBot(c *Config, clusterName, webProxyAddr string, log *slog.Logger) (*Bot
Config: c.MSAPI,
graphClient: msapi.NewGraphClient(c.MSAPI),
botClient: msapi.NewBotFrameworkClient(c.MSAPI),
recipients: make(map[string]RecipientData),
fullRecipients: make(map[string]common.Recipient),
webProxyURL: webProxyURL,
clusterName: clusterName,
mu: &sync.RWMutex{},
Expand Down Expand Up @@ -181,7 +177,7 @@ func (b *Bot) UninstallAppForUser(ctx context.Context, userIDOrEmail string) err
// FetchRecipient checks if recipient is a user or a channel, installs app for a user if missing, fetches chat id
// and saves everything to cache. This method is used for priming the cache. Returns trace.NotFound if a
// user was not found.
func (b *Bot) FetchRecipient(ctx context.Context, recipient string) (*RecipientData, error) {
func (b *Bot) FetchRecipient(ctx context.Context, recipient string) (*common.Recipient, error) {
if b.teamsApp == nil {
return nil, trace.Errorf("Bot is not configured, run GetTeamsApp first")
}
Expand All @@ -190,16 +186,20 @@ func (b *Bot) FetchRecipient(ctx context.Context, recipient string) (*RecipientD
log.DebugContext(ctx, "Fetching recipient")

b.mu.RLock()
d, ok := b.recipients[recipient]
d, ok := b.fullRecipients[recipient]
b.mu.RUnlock()
rd, err := getRecipientData(&d)
if err != nil {
return nil, trace.Wrap(err)
}
if ok {
log.DebugContext(ctx, "Found recipient in cache",
slog.Group("recipient",
"id", d.ID,
"installed_app_id", d.App.ID,
"chat_id", d.Chat.ID,
"chat_url", d.Chat.WebURL,
"chat_tenant_id", d.Chat.TenantID,
"installed_app_id", rd.App.ID,
"chat_id", rd.Chat.ID,
"chat_url", rd.Chat.WebURL,
"chat_tenant_id", rd.Chat.TenantID,
"kind", d.Kind,
),
)
Expand All @@ -220,24 +220,27 @@ func (b *Bot) FetchRecipient(ctx context.Context, recipient string) (*RecipientD
),
)
// A team and a group are different but in MsTeams the team is associated to a group and will have the same id.

log = log.With("teams_app_id", b.teamsApp.ID, "channel_group", channel.Group)
log.DebugContext(ctx, "Retrieving the app installation ID for the team")
installedApp, err := b.graphClient.GetAppForTeam(ctx, b.teamsApp, channel.Group)
if err != nil {
log.ErrorContext(ctx, "Failed to retrieve the app installation ID for the team", "error", err)
return nil, trace.Wrap(err)
}
d = RecipientData{
ID: fmt.Sprintf("%s/%s/%s", channel.Tenant, channel.Group, channel.Name),
App: *installedApp,
Chat: msapi.Chat{
ID: channel.ChatID,
TenantID: channel.Tenant,
WebURL: channel.URL.String(),
d = common.Recipient{
Name: recipient,
ID: fmt.Sprintf("%s/%s/%s", channel.Tenant, channel.Group, channel.Name),
Kind: string(RecipientKindChannel),
Data: recipientData{
App: *installedApp,
Chat: msapi.Chat{
ID: channel.ChatID,
TenantID: channel.Tenant,
WebURL: channel.URL.String(),
},
},
Kind: RecipientKindChannel,
}

log.DebugContext(ctx, "Retrieved the app installation ID for the team", "recipient_installed_app_id", installedApp.ID)

// If the recipient is not a channel, it means it is a user (either email or userID)
Expand All @@ -246,7 +249,7 @@ func (b *Bot) FetchRecipient(ctx context.Context, recipient string) (*RecipientD
userID, err := b.getUserID(ctx, recipient)
if err != nil {
log.ErrorContext(ctx, "Failed to resolve recipient", "error", err)
return &RecipientData{}, trace.Wrap(err)
return &common.Recipient{}, trace.Wrap(err)
}
log = log.With("user_id", userID)
log.DebugContext(ctx, "Successfully resolve user recipient")
Expand Down Expand Up @@ -285,11 +288,19 @@ func (b *Bot) FetchRecipient(ctx context.Context, recipient string) (*RecipientD
}
log.DebugContext(ctx, "Found the chat ID for the user", "chat_id", chat.ID)

d = RecipientData{userID, *installedApp, chat, RecipientKindUser}
d = common.Recipient{
Name: recipient,
ID: userID,
Kind: string(RecipientKindUser),
Data: recipientData{
App: *installedApp,
Chat: chat,
},
}
}

b.mu.Lock()
b.recipients[recipient] = d
b.fullRecipients[recipient] = d
b.mu.Unlock()

return &d, nil
Expand All @@ -316,8 +327,12 @@ func (b *Bot) getUserID(ctx context.Context, userIDOrEmail string) (string, erro
}

// PostAdaptiveCardActivity sends the AdaptiveCard to a user
func (b *Bot) PostAdaptiveCardActivity(ctx context.Context, recipient, cardBody, updateID string) (string, error) {
recipientData, err := b.FetchRecipient(ctx, recipient)
func (b *Bot) PostAdaptiveCardActivity(ctx context.Context, recipient string, cardBody, updateID string) (string, error) {
fullRecipient, err := b.FetchRecipient(ctx, recipient)
if err != nil {
return "", trace.Wrap(err)
}
recipientData, err := getRecipientData(fullRecipient)
if err != nil {
return "", trace.Wrap(err)
}
Expand Down Expand Up @@ -474,3 +489,11 @@ func (b *Bot) CheckHealth(ctx context.Context) error {
}
return trace.Wrap(err)
}

func getRecipientData(recipient *common.Recipient) (recipientData, error) {
rd, ok := recipient.Data.(recipientData)
if !ok {
return recipientData{}, trace.BadParameter("unexpected recipient data type")
}
return rd, nil
}
6 changes: 5 additions & 1 deletion integrations/access/msteams/validate.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,11 @@ func Validate(configPath, recipient string) error {
recipient = userID
}

recipientData, err := b.FetchRecipient(context.Background(), recipient)
fullRecipient, err := b.FetchRecipient(ctx, recipient)
if err != nil {
return trace.Wrap(err)
}
recipientData, err := getRecipientData(fullRecipient)
if err != nil {
return trace.Wrap(err)
}
Expand Down
Loading