Skip to content

Commit

Permalink
Address flvdnd and spolu comments.
Browse files Browse the repository at this point in the history
  • Loading branch information
lasryaric committed Apr 5, 2024
1 parent 70ab647 commit e7d9cb6
Show file tree
Hide file tree
Showing 5 changed files with 151 additions and 74 deletions.
36 changes: 0 additions & 36 deletions connectors/src/connectors/slack/bot.ts
Original file line number Diff line number Diff line change
Expand Up @@ -626,42 +626,6 @@ export async function getBotEnabled(
return new Ok(slackConfig.botEnabled);
}

export async function toggleSlackbot(
connectorId: ModelId,
botEnabled: boolean
): Promise<Result<void, Error>> {
const slackConfig = await SlackConfigurationResource.fetchByConnectorId(
connectorId
);

if (!slackConfig) {
return new Err(
new Error(
`Failed to find a Slack configuration for connector ${connectorId}`
)
);
}

if (botEnabled) {
const otherSlackConfigWithBotEnabled =
await SlackConfigurationResource.fetchByActiveBot(
slackConfig.slackTeamId
);

if (otherSlackConfigWithBotEnabled) {
return new Err(
new Error(
"Another Dust workspace has already enabled the slack bot for your Slack workspace."
)
);
}
}

await slackConfig.update({ botEnabled: botEnabled });

return new Ok(void 0);
}

async function makeContentFragment(
slackClient: WebClient,
channelId: string,
Expand Down
22 changes: 16 additions & 6 deletions connectors/src/connectors/slack/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,10 +14,7 @@ import type {
ConnectorPermissionRetriever,
} from "@connectors/connectors/interface";
import { getChannels } from "@connectors/connectors/slack//temporal/activities";
import {
getBotEnabled,
toggleSlackbot,
} from "@connectors/connectors/slack/bot";
import { getBotEnabled } from "@connectors/connectors/slack/bot";
import { joinChannel } from "@connectors/connectors/slack/lib/channels";
import {
getSlackAccessToken,
Expand Down Expand Up @@ -620,8 +617,21 @@ export async function setSlackConfig(

switch (configKey) {
case "botEnabled": {
const res = await toggleSlackbot(connectorId, configValue === "true");
return res;
const slackConfig = await SlackConfigurationResource.fetchByConnectorId(
connectorId
);
if (!slackConfig) {
return new Err(
new Error(
`Slack configuration not found for connector ${connectorId}`
)
);
}
if (configValue === "true") {
return slackConfig.enableBot();
} else {
return slackConfig.disableBot();
}
}

default: {
Expand Down
14 changes: 6 additions & 8 deletions connectors/src/connectors/slack/lib/workspace_limits.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,9 @@ import type {} from "@slack/web-api/dist/response/UsersInfoResponse";
import type { SlackUserInfo } from "@connectors/connectors/slack/lib/slack_client";
import { getSlackConversationInfo } from "@connectors/connectors/slack/lib/slack_client";
import { dataSourceConfigFromConnector } from "@connectors/lib/api/data_source_config";
import { SlackBotWhitelistModel } from "@connectors/lib/models/slack";
import logger from "@connectors/logger/logger";
import type { ConnectorResource } from "@connectors/resources/connector_resource";
import { SlackConfigurationResource } from "@connectors/resources/slack_configuration_resource";

async function getActiveMemberEmails(
connector: ConnectorResource
Expand Down Expand Up @@ -213,13 +213,11 @@ async function isBotAllowed(
// This means that even a non verified user of a given Slack workspace who can trigger a bot
// that talks to our bot (@dust) will be able to use the Dust bot.
// Make sure to be explicit about this with users as you whitelist a new bot.
// Example: non-verfied-user -> @AnyWhitelistedBot -> @dust -> Dust answers with potentially private information.
const whitelist = await SlackBotWhitelistModel.findOne({
where: {
connectorId: connector.id,
botName: names,
},
});
// Example: non-verified-user -> @AnyWhitelistedBot -> @dust -> Dust answers with potentially private information.
const slackConfig = await SlackConfigurationResource.fetchByConnectorId(
connector.id
);
const whitelist = await slackConfig?.isBotWhitelisted(names);
if (!whitelist) {
logger.info(
{ user: slackUserInfo, connectorId: connector.id },
Expand Down
19 changes: 14 additions & 5 deletions connectors/src/lib/cli.ts
Original file line number Diff line number Diff line change
Expand Up @@ -70,7 +70,6 @@ import {
upsertPageWorkflow,
} from "@connectors/connectors/notion/temporal/workflows";
import { uninstallSlack } from "@connectors/connectors/slack";
import { toggleSlackbot } from "@connectors/connectors/slack/bot";
import { maybeLaunchSlackSyncWorkflowForChannelId } from "@connectors/connectors/slack/lib/cli";
import { launchSlackSyncOneThreadWorkflow } from "@connectors/connectors/slack/temporal/client";
import { launchCrawlWebsiteSchedulerWorkflow } from "@connectors/connectors/webcrawler/temporal/client";
Expand Down Expand Up @@ -917,7 +916,19 @@ export const slack = async ({
if (!connector) {
throw new Error(`Could not find connector for workspace ${args.wId}`);
}
await throwOnError(toggleSlackbot(connector.id, true));
const slackConfig = await SlackConfigurationResource.fetchByConnectorId(
connector.id
);
if (!slackConfig) {
throw new Error(
`Could not find slack configuration for connector ${connector.id}`
);
}

const res = await slackConfig.enableBot();
if (res.isErr()) {
throw res.error;
}
return { success: true };
}

Expand Down Expand Up @@ -1075,9 +1086,7 @@ export const slack = async ({
connector.id
);
if (slackConfig) {
await slackConfig.update({
whitelistedDomains: whitelistedDomainsArray,
});
await slackConfig.setWhitelistedDomains(whitelistedDomainsArray);
}

return { success: true };
Expand Down
134 changes: 115 additions & 19 deletions connectors/src/resources/slack_configuration_resource.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,14 @@ import type { ModelId, Result } from "@dust-tt/types";
import { Err, Ok } from "@dust-tt/types";
import type { Attributes, ModelStatic, Transaction } from "sequelize";

import { SlackConfigurationModel } from "@connectors/lib/models/slack";
import {
SlackBotWhitelistModel,
SlackChannel,
SlackConfigurationModel,
SlackMessages,
} from "@connectors/lib/models/slack";
import logger from "@connectors/logger/logger";
import { BaseResource } from "@connectors/resources/base_resource";
import { sequelizeConnection } from "@connectors/resources/storage";
import type { ReadonlyAttributesType } from "@connectors/resources/storage/types";

// Attributes are marked as read-only to reflect the stateless nature of our Resource.
Expand Down Expand Up @@ -50,10 +55,10 @@ export class SlackConfigurationResource extends BaseResource<SlackConfigurationM
);
}

static async fetchByConnectorId(id: ModelId) {
static async fetchByConnectorId(connectorId: ModelId) {
const blob = await this.model.findOne({
where: {
connectorId: id,
connectorId: connectorId,
},
});
if (!blob) {
Expand All @@ -79,6 +84,15 @@ export class SlackConfigurationResource extends BaseResource<SlackConfigurationM
return new this(this.model, blob.get());
}

async isBotWhitelisted(botName: string | string[]): Promise<boolean> {
return !!(await SlackBotWhitelistModel.findOne({
where: {
connectorId: this.id,
botName: botName,
},
}));
}

static async listAll() {
const blobs = await SlackConfigurationResource.model.findAll({});

Expand All @@ -88,7 +102,9 @@ export class SlackConfigurationResource extends BaseResource<SlackConfigurationM
);
}

static async listForTeamId(slackTeamId: string) {
static async listForTeamId(
slackTeamId: string
): Promise<SlackConfigurationResource[]> {
const blobs = await this.model.findAll({
where: {
slackTeamId,
Expand All @@ -101,20 +117,100 @@ export class SlackConfigurationResource extends BaseResource<SlackConfigurationM
);
}

async delete(): Promise<Result<undefined, Error>> {
return sequelizeConnection.transaction(async (transaction) => {
try {
await this.model.destroy({
where: {
id: this.id,
},
transaction,
});

return new Ok(undefined);
} catch (err) {
return new Err(err as Error);
async enableBot(): Promise<Result<undefined, Error>> {
const otherSlackConfigurationWithBotEnabled =
await SlackConfigurationModel.findOne({
where: {
slackTeamId: this.slackTeamId,
botEnabled: true,
},
});
if (
otherSlackConfigurationWithBotEnabled &&
otherSlackConfigurationWithBotEnabled.id !== this.id
) {
logger.error(
{
slackTeamId: this.slackTeamId,
},
"Another Dust workspace has already enabled the slack bot for your Slack workspace."
);
return new Err(
new Error(
"Another Dust workspace has already enabled the slack bot for your Slack workspace."
)
);
}
await this.model.update(
{ botEnabled: true },
{
where: {
id: this.id,
},
}
});
);

return new Ok(undefined);
}

async disableBot(): Promise<Result<undefined, Error>> {
await this.model.update(
{ botEnabled: false },
{
where: {
id: this.id,
},
}
);

return new Ok(undefined);
}

async setWhitelistedDomains(domain: string[]) {
await this.model.update(
{ whitelistedDomains: domain },
{
where: {
id: this.id,
},
}
);

return new Ok(undefined);
}
async delete(transaction: Transaction): Promise<Result<undefined, Error>> {
try {
await SlackChannel.destroy({
where: {
connectorId: this.id,
},
transaction,
});

await SlackMessages.destroy({
where: {
connectorId: this.id,
},
transaction,
});

await SlackBotWhitelistModel.destroy({
where: {
connectorId: this.id,
},
transaction,
});

await this.model.destroy({
where: {
id: this.id,
},
transaction,
});

return new Ok(undefined);
} catch (err) {
return new Err(err as Error);
}
}
}

0 comments on commit e7d9cb6

Please sign in to comment.