Add support for Challenge Hound (#924)
* Add challenge hound connection type * Add config * Add bridge bindings * Add reader implementation. * Obvious renames. * bit more tidying * refactor * fix imports * fix import * Start feed reader and recognise service. * Move to using IDs rather than URLs for better security. * lint * Validate that challenge exists. * Drive-by refactors. * Add add/remove commands for challenge hound. * Add challenge hound docs. * Refactor icons * add some more activity definitions * changelog * cleanup feed work
1
changelog.d/924.feature
Normal file
@ -0,0 +1 @@
|
||||
Add support for Challenge Hound.
|
@ -9,6 +9,7 @@
|
||||
- [GitLab](./setup/gitlab.md)
|
||||
- [JIRA](./setup/jira.md)
|
||||
- [Webhooks](./setup/webhooks.md)
|
||||
- [ChallengeHound](./setup/challengehound.md)
|
||||
- [👤 Usage](./usage.md)
|
||||
- [Dynamic Rooms](./usage/dynamic_rooms.md)
|
||||
- [Authenticating](./usage/auth.md)
|
||||
|
Before Width: | Height: | Size: 760 B After Width: | Height: | Size: 760 B |
Before Width: | Height: | Size: 716 B After Width: | Height: | Size: 716 B |
Before Width: | Height: | Size: 866 B After Width: | Height: | Size: 866 B |
Before Width: | Height: | Size: 887 B After Width: | Height: | Size: 887 B |
BIN
docs/_site/icons/hound.png
Normal file
After Width: | Height: | Size: 706 B |
Before Width: | Height: | Size: 990 B After Width: | Height: | Size: 990 B |
Before Width: | Height: | Size: 543 B After Width: | Height: | Size: 543 B |
Before Width: | Height: | Size: 1.2 KiB After Width: | Height: | Size: 1.2 KiB |
@ -26,33 +26,39 @@
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
/* icons for headers */
|
||||
|
||||
/* icons for headers */
|
||||
/* We use base64 to avoid having to deal with pathing issues. */
|
||||
.chapter > li:nth-child(3) > ol:nth-child(1) > li:nth-child(2) strong:after {
|
||||
content: ' ' url('/matrix-hookshot/latest/icons/feeds.png');
|
||||
content: ' ' url('');
|
||||
}
|
||||
|
||||
.chapter > li:nth-child(3) > ol:nth-child(1) > li:nth-child(3) strong:after {
|
||||
content: ' ' url('/matrix-hookshot/latest/icons/figma.png');
|
||||
content: ' ' url('');
|
||||
}
|
||||
|
||||
.chapter > li:nth-child(3) > ol:nth-child(1) > li:nth-child(4) strong:after {
|
||||
content: ' ' url('/matrix-hookshot/latest/icons/github.png');
|
||||
content: ' ' url('');
|
||||
}
|
||||
|
||||
.chapter > li:nth-child(3) > ol:nth-child(1) > li:nth-child(5) strong:after {
|
||||
content: ' ' url('/matrix-hookshot/latest/icons/gitlab.png');
|
||||
content: ' ' url('');
|
||||
}
|
||||
|
||||
.chapter > li:nth-child(3) > ol:nth-child(1) > li:nth-child(6) strong:after {
|
||||
content: ' ' url('/matrix-hookshot/latest/icons/jira.png');
|
||||
content: ' ' url('');
|
||||
}
|
||||
|
||||
.chapter > li:nth-child(3) > ol:nth-child(1) > li:nth-child(7) strong:after {
|
||||
content: ' ' url('/matrix-hookshot/latest/icons/webhooks.png');
|
||||
content: ' ' url('');
|
||||
}
|
||||
|
||||
.chapter > li:nth-child(3) > ol:nth-child(1) > li:nth-child(8) strong:after {
|
||||
content: ' ' url('');
|
||||
}
|
||||
|
||||
|
||||
.chapter li:nth-child(7) > a:nth-child(1) > strong:after {
|
||||
content: ' ' url('/matrix-hookshot/latest/icons/sentry.png');
|
||||
content: ' ' url('');
|
||||
}
|
||||
|
||||
|
@ -120,6 +120,7 @@ Each permission set can have a service. The `service` field can be:
|
||||
- `feed`
|
||||
- `figma`
|
||||
- `webhooks`
|
||||
- `challengehound`
|
||||
- `*`, for any service.
|
||||
|
||||
The `level` can be:
|
||||
|
42
docs/setup/challengehound.md
Normal file
@ -0,0 +1,42 @@
|
||||
# ChallengeHound
|
||||
|
||||
You can configure Hookshot to bridge [ChallengeHound](https://www.challengehound.com/) activites
|
||||
into Matrix.
|
||||
|
||||
### Getting the API secret.
|
||||
|
||||
Unfortunately, there is no way to directly request a persistent Challenge Hound API token. The
|
||||
only way to authenticate with the service at present is to login with an email address and receive
|
||||
a magic token in an email. This is not something Hookshot has the capability to do on it's own.
|
||||
|
||||
In order to extract the token for use with the bridge, login to Challenge Hound. Once logged in,
|
||||
please locate the local storage via the devtools of your browser. Inside you will find a `ch:user`
|
||||
entry with a `token` value. That value should be used as the secret for your Hookshot config.
|
||||
|
||||
```yaml
|
||||
challengeHound:
|
||||
token: <the token>
|
||||
```
|
||||
|
||||
This token tends to expire roughly once a month, and for the moment you'll need to manually
|
||||
replace it. You can also ask Challenge Hound's support for an API key, although this has not
|
||||
been tested.
|
||||
|
||||
## Usage
|
||||
|
||||
You can add a new challenge hound challenge by command:
|
||||
|
||||
```
|
||||
challengehound add https://www.challengehound.com/challenge/abc-def
|
||||
```
|
||||
|
||||
and remove it with the same command
|
||||
|
||||
```
|
||||
challengehound remove https://www.challengehound.com/challenge/abc-def
|
||||
```.
|
||||
|
||||
Hookshot will periodically refetch activities from the challenge and send a notice when a new
|
||||
one is completed. Note that Hookshot uses your configured cache to store seen activities. If
|
||||
you have not configured Redis caching, it will default to in-memory storage which means activites
|
||||
**will** repeat on restart.
|
@ -41,6 +41,8 @@ import { SetupWidget } from "./Widgets/SetupWidget";
|
||||
import { FeedEntry, FeedError, FeedReader, FeedSuccess } from "./feeds/FeedReader";
|
||||
import PQueue from "p-queue";
|
||||
import * as Sentry from '@sentry/node';
|
||||
import { HoundConnection, HoundPayload } from "./Connections/HoundConnection";
|
||||
import { HoundReader } from "./hound/reader";
|
||||
|
||||
const log = new Logger("Bridge");
|
||||
|
||||
@ -53,6 +55,7 @@ export class Bridge {
|
||||
private github?: GithubInstance;
|
||||
private adminRooms: Map<string, AdminRoom> = new Map();
|
||||
private feedReader?: FeedReader;
|
||||
private houndReader?: HoundReader;
|
||||
private provisioningApi?: Provisioner;
|
||||
private replyProcessor = new RichRepliesPreprocessor(true);
|
||||
|
||||
@ -78,6 +81,7 @@ export class Bridge {
|
||||
|
||||
public stop() {
|
||||
this.feedReader?.stop();
|
||||
this.houndReader?.stop();
|
||||
this.tokenStore.stop();
|
||||
this.as.stop();
|
||||
if (this.queue.stop) this.queue.stop();
|
||||
@ -678,6 +682,12 @@ export class Bridge {
|
||||
(c, data) => c.handleFeedError(data),
|
||||
);
|
||||
|
||||
this.bindHandlerToQueue<HoundPayload, HoundConnection>(
|
||||
"hound.activity",
|
||||
(data) => connManager.getConnectionsForHoundChallengeId(data.challengeId),
|
||||
(c, data) => c.handleNewActivity(data.activity)
|
||||
);
|
||||
|
||||
const queue = new PQueue({
|
||||
concurrency: 2,
|
||||
});
|
||||
@ -785,6 +795,15 @@ export class Bridge {
|
||||
);
|
||||
}
|
||||
|
||||
if (this.config.challengeHound?.token) {
|
||||
this.houndReader = new HoundReader(
|
||||
this.config.challengeHound,
|
||||
this.connectionManager,
|
||||
this.queue,
|
||||
this.storage,
|
||||
);
|
||||
}
|
||||
|
||||
const webhookHandler = new Webhooks(this.config);
|
||||
this.listener.bindResource('webhooks', webhookHandler.expressRouter);
|
||||
|
||||
|
@ -8,7 +8,8 @@ import { Appservice, Intent, StateEvent } from "matrix-bot-sdk";
|
||||
import { ApiError, ErrCode } from "./api";
|
||||
import { BridgeConfig, BridgePermissionLevel, GitLabInstance } from "./config/Config";
|
||||
import { CommentProcessor } from "./CommentProcessor";
|
||||
import { ConnectionDeclaration, ConnectionDeclarations, GenericHookConnection, GitHubDiscussionConnection, GitHubDiscussionSpace, GitHubIssueConnection, GitHubProjectConnection, GitHubRepoConnection, GitHubUserSpace, GitLabIssueConnection, GitLabRepoConnection, IConnection, IConnectionState, JiraProjectConnection } from "./Connections";
|
||||
import { ConnectionDeclaration, ConnectionDeclarations, GenericHookConnection, GitHubDiscussionConnection, GitHubDiscussionSpace, GitHubIssueConnection,
|
||||
GitHubProjectConnection, GitHubRepoConnection, GitHubUserSpace, GitLabIssueConnection, GitLabRepoConnection, IConnection, IConnectionState, JiraProjectConnection } from "./Connections";
|
||||
import { FigmaFileConnection, FeedConnection } from "./Connections";
|
||||
import { GetConnectionTypeResponseItem } from "./provisioning/api";
|
||||
import { GitLabClient } from "./Gitlab/Client";
|
||||
@ -22,6 +23,7 @@ import BotUsersManager from "./Managers/BotUsersManager";
|
||||
import { retry, retryMatrixErrorFilter } from "./PromiseUtil";
|
||||
import Metrics from "./Metrics";
|
||||
import EventEmitter from "events";
|
||||
import { HoundConnection } from "./Connections/HoundConnection";
|
||||
|
||||
const log = new Logger("ConnectionManager");
|
||||
|
||||
@ -341,6 +343,10 @@ export class ConnectionManager extends EventEmitter {
|
||||
return this.connections.filter(c => c instanceof FeedConnection && c.feedUrl === url) as FeedConnection[];
|
||||
}
|
||||
|
||||
public getConnectionsForHoundChallengeId(challengeId: string): HoundConnection[] {
|
||||
return this.connections.filter(c => c instanceof HoundConnection && c.challengeId === challengeId) as HoundConnection[];
|
||||
}
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
public getAllConnectionsOfType<T extends IConnection>(typeT: new (...params : any[]) => T): T[] {
|
||||
return this.connections.filter((c) => (c instanceof typeT)) as T[];
|
||||
|
183
src/Connections/HoundConnection.ts
Normal file
@ -0,0 +1,183 @@
|
||||
import { Intent, StateEvent } from "matrix-bot-sdk";
|
||||
import markdownit from "markdown-it";
|
||||
import { BaseConnection } from "./BaseConnection";
|
||||
import { IConnection, IConnectionState } from ".";
|
||||
import { Connection, InstantiateConnectionOpts, ProvisionConnectionOpts } from "./IConnection";
|
||||
import { CommandError } from "../errors";
|
||||
|
||||
export interface HoundConnectionState extends IConnectionState {
|
||||
challengeId: string;
|
||||
}
|
||||
|
||||
export interface HoundPayload {
|
||||
activity: HoundActivity,
|
||||
challengeId: string,
|
||||
}
|
||||
|
||||
export interface HoundActivity {
|
||||
id: string;
|
||||
distance: number; // in meters
|
||||
duration: number;
|
||||
elevation: number;
|
||||
createdAt: string;
|
||||
activityType: string;
|
||||
activityName: string;
|
||||
user: {
|
||||
id: string;
|
||||
fullname: string;
|
||||
fname: string;
|
||||
lname: string;
|
||||
}
|
||||
}
|
||||
|
||||
export interface IChallenge {
|
||||
id: string;
|
||||
distance: number;
|
||||
duration: number;
|
||||
elevaion: number;
|
||||
}
|
||||
|
||||
export interface ILeader {
|
||||
id: string;
|
||||
fullname: string;
|
||||
duration: number;
|
||||
distance: number;
|
||||
elevation: number;
|
||||
}
|
||||
|
||||
function getEmojiForType(type: string) {
|
||||
switch (type) {
|
||||
case "run":
|
||||
return "🏃";
|
||||
case "virtualrun":
|
||||
return "👨💻🏃";
|
||||
case "ride":
|
||||
case "cycle":
|
||||
case "cycling":
|
||||
return "🚴";
|
||||
case "mountainbikeride":
|
||||
return "⛰️🚴";
|
||||
case "virtualride":
|
||||
return "👨💻🚴";
|
||||
case "walk":
|
||||
case "hike":
|
||||
return "🚶";
|
||||
case "skateboard":
|
||||
return "🛹";
|
||||
case "virtualwalk":
|
||||
case "virtualhike":
|
||||
return "👨💻🚶";
|
||||
case "alpineski":
|
||||
return "⛷️";
|
||||
case "swim":
|
||||
return "🏊";
|
||||
default:
|
||||
return "🕴️";
|
||||
}
|
||||
}
|
||||
|
||||
const md = markdownit();
|
||||
@Connection
|
||||
export class HoundConnection extends BaseConnection implements IConnection {
|
||||
static readonly CanonicalEventType = "uk.half-shot.matrix-hookshot.challengehound.activity";
|
||||
static readonly LegacyEventType = "uk.half-shot.matrix-challenger.activity"; // Magically import from matrix-challenger
|
||||
|
||||
static readonly EventTypes = [
|
||||
HoundConnection.CanonicalEventType,
|
||||
HoundConnection.LegacyEventType,
|
||||
];
|
||||
static readonly ServiceCategory = "challengehound";
|
||||
|
||||
public static getIdFromURL(url: string): string {
|
||||
const parts = new URL(url).pathname.split('/');
|
||||
return parts[parts.length-1];
|
||||
}
|
||||
|
||||
public static validateState(data: Record<string, unknown>): HoundConnectionState {
|
||||
// Convert URL to ID.
|
||||
if (!data.challengeId && data.url && data.url === "string") {
|
||||
data.challengeId = this.getIdFromURL(data.url);
|
||||
}
|
||||
|
||||
// Test for v1 uuid.
|
||||
if (!data.challengeId || typeof data.challengeId !== "string" || /^\w{8}(?:-\w{4}){3}-\w{12}$/.test(data.challengeId)) {
|
||||
throw Error('Missing or invalid id');
|
||||
}
|
||||
|
||||
return {
|
||||
challengeId: data.challengeId
|
||||
}
|
||||
}
|
||||
|
||||
public static createConnectionForState(roomId: string, event: StateEvent<Record<string, unknown>>, {config, intent}: InstantiateConnectionOpts) {
|
||||
if (!config.challengeHound) {
|
||||
throw Error('Challenge hound is not configured');
|
||||
}
|
||||
return new HoundConnection(roomId, event.stateKey, this.validateState(event.content), intent);
|
||||
}
|
||||
|
||||
static async provisionConnection(roomId: string, _userId: string, data: Record<string, unknown> = {}, {intent, config}: ProvisionConnectionOpts) {
|
||||
if (!config.challengeHound) {
|
||||
throw Error('Challenge hound is not configured');
|
||||
}
|
||||
const validState = this.validateState(data);
|
||||
// Check the event actually exists.
|
||||
const statusDataRequest = await fetch(`https://api.challengehound.com/challenges/${validState.challengeId}/status`);
|
||||
if (!statusDataRequest.ok) {
|
||||
throw new CommandError(`Fetch failed, status ${statusDataRequest.status}`, "Challenge could not be found. Is it active?");
|
||||
}
|
||||
const { challengeName } = await statusDataRequest.json() as {challengeName: string};
|
||||
const connection = new HoundConnection(roomId, validState.challengeId, validState, intent);
|
||||
await intent.underlyingClient.sendStateEvent(roomId, HoundConnection.CanonicalEventType, validState.challengeId, validState);
|
||||
return {
|
||||
connection,
|
||||
stateEventContent: validState,
|
||||
challengeName,
|
||||
};
|
||||
}
|
||||
|
||||
constructor(
|
||||
roomId: string,
|
||||
stateKey: string,
|
||||
private state: HoundConnectionState,
|
||||
private readonly intent: Intent) {
|
||||
super(roomId, stateKey, HoundConnection.CanonicalEventType)
|
||||
}
|
||||
|
||||
public isInterestedInStateEvent() {
|
||||
return false; // We don't support state-updates...yet.
|
||||
}
|
||||
|
||||
public get challengeId() {
|
||||
return this.state.challengeId;
|
||||
}
|
||||
|
||||
public get priority(): number {
|
||||
return this.state.priority || super.priority;
|
||||
}
|
||||
|
||||
public async handleNewActivity(payload: HoundActivity) {
|
||||
const distance = `${(payload.distance / 1000).toFixed(2)}km`;
|
||||
const emoji = getEmojiForType(payload.activityType);
|
||||
const body = `🎉 **${payload.user.fullname}** completed a ${distance} ${emoji} ${payload.activityType} (${payload.activityName})`;
|
||||
const content: any = {
|
||||
body,
|
||||
format: "org.matrix.custom.html",
|
||||
formatted_body: md.renderInline(body),
|
||||
};
|
||||
content["msgtype"] = "m.notice";
|
||||
content["uk.half-shot.matrix-challenger.activity.id"] = payload.id;
|
||||
content["uk.half-shot.matrix-challenger.activity.distance"] = Math.round(payload.distance);
|
||||
content["uk.half-shot.matrix-challenger.activity.elevation"] = Math.round(payload.elevation);
|
||||
content["uk.half-shot.matrix-challenger.activity.duration"] = Math.round(payload.duration);
|
||||
content["uk.half-shot.matrix-challenger.activity.user"] = {
|
||||
"name": payload.user.fullname,
|
||||
id: payload.user.id,
|
||||
};
|
||||
await this.intent.underlyingClient.sendMessage(this.roomId, content);
|
||||
}
|
||||
|
||||
public toString() {
|
||||
return `HoundConnection ${this.challengeId}`;
|
||||
}
|
||||
}
|
@ -1,4 +1,3 @@
|
||||
// We need to instantiate some functions which are not directly called, which confuses typescript.
|
||||
import { BotCommands, botCommand, compileBotCommands, HelpFunction } from "../BotCommands";
|
||||
import { CommandConnection } from "./CommandConnection";
|
||||
import { GenericHookConnection, GenericHookConnectionState, GitHubRepoConnection, JiraProjectConnection, JiraProjectConnectionState } from ".";
|
||||
@ -15,6 +14,7 @@ import { IConnection, IConnectionState, ProvisionConnectionOpts } from "./IConne
|
||||
import { ApiError, Logger } from "matrix-appservice-bridge";
|
||||
import { Intent } from "matrix-bot-sdk";
|
||||
import YAML from 'yaml';
|
||||
import { HoundConnection } from "./HoundConnection";
|
||||
const md = new markdown();
|
||||
const log = new Logger("SetupConnection");
|
||||
|
||||
@ -72,13 +72,13 @@ export class SetupConnection extends CommandConnection {
|
||||
this.includeTitlesInHelp = false;
|
||||
}
|
||||
|
||||
@botCommand("github repo", { help: "Create a connection for a GitHub repository. (You must be logged in with GitHub to do this.)", requiredArgs: ["url"], includeUserId: true, category: "github"})
|
||||
@botCommand("github repo", { help: "Create a connection for a GitHub repository. (You must be logged in with GitHub to do this.)", requiredArgs: ["url"], includeUserId: true, category: GitHubRepoConnection.ServiceCategory})
|
||||
public async onGitHubRepo(userId: string, url: string) {
|
||||
if (!this.provisionOpts.github || !this.config.github) {
|
||||
throw new CommandError("not-configured", "The bridge is not configured to support GitHub.");
|
||||
}
|
||||
|
||||
await this.checkUserPermissions(userId, "github", GitHubRepoConnection.CanonicalEventType);
|
||||
await this.checkUserPermissions(userId, GitHubRepoConnection.ServiceCategory, GitHubRepoConnection.CanonicalEventType);
|
||||
const octokit = await this.provisionOpts.tokenStore.getOctokitForUser(userId);
|
||||
if (!octokit) {
|
||||
throw new CommandError("User not logged in", "You are not logged into GitHub. Start a DM with this bot and use the command `github login`.");
|
||||
@ -93,13 +93,13 @@ export class SetupConnection extends CommandConnection {
|
||||
await this.client.sendNotice(this.roomId, `Room configured to bridge ${connection.org}/${connection.repo}`);
|
||||
}
|
||||
|
||||
@botCommand("gitlab project", { help: "Create a connection for a GitHub project. (You must be logged in with GitLab to do this.)", requiredArgs: ["url"], includeUserId: true, category: "gitlab"})
|
||||
@botCommand("gitlab project", { help: "Create a connection for a GitHub project. (You must be logged in with GitLab to do this.)", requiredArgs: ["url"], includeUserId: true, category: GitLabRepoConnection.ServiceCategory})
|
||||
public async onGitLabRepo(userId: string, url: string) {
|
||||
if (!this.config.gitlab) {
|
||||
throw new CommandError("not-configured", "The bridge is not configured to support GitLab.");
|
||||
}
|
||||
|
||||
await this.checkUserPermissions(userId, "gitlab", GitLabRepoConnection.CanonicalEventType);
|
||||
await this.checkUserPermissions(userId, GitLabRepoConnection.ServiceCategory, GitLabRepoConnection.CanonicalEventType);
|
||||
|
||||
const {name, instance} = this.config.gitlab.getInstanceByProjectUrl(url) || {};
|
||||
if (!instance || !name) {
|
||||
@ -126,7 +126,7 @@ export class SetupConnection extends CommandConnection {
|
||||
}
|
||||
}
|
||||
|
||||
private async getJiraProjectSafeUrl(userId: string, urlStr: string) {
|
||||
private async getJiraProjectSafeUrl(urlStr: string) {
|
||||
const url = new URL(urlStr);
|
||||
const urlParts = /\/projects\/(\w+)\/?(\w+\/?)*$/.exec(url.pathname);
|
||||
const projectKey = urlParts?.[1] || url.searchParams.get('projectKey');
|
||||
@ -136,22 +136,22 @@ export class SetupConnection extends CommandConnection {
|
||||
return `https://${url.host}/projects/${projectKey}`;
|
||||
}
|
||||
|
||||
@botCommand("jira project", { help: "Create a connection for a JIRA project. (You must be logged in with JIRA to do this.)", requiredArgs: ["url"], includeUserId: true, category: "jira"})
|
||||
@botCommand("jira project", { help: "Create a connection for a JIRA project. (You must be logged in with JIRA to do this.)", requiredArgs: ["url"], includeUserId: true, category: JiraProjectConnection.ServiceCategory})
|
||||
public async onJiraProject(userId: string, urlStr: string) {
|
||||
if (!this.config.jira) {
|
||||
throw new CommandError("not-configured", "The bridge is not configured to support Jira.");
|
||||
}
|
||||
|
||||
await this.checkUserPermissions(userId, "jira", JiraProjectConnection.CanonicalEventType);
|
||||
await this.checkUserPermissions(userId, JiraProjectConnection.ServiceCategory, JiraProjectConnection.CanonicalEventType);
|
||||
await this.checkJiraLogin(userId, urlStr);
|
||||
const safeUrl = await this.getJiraProjectSafeUrl(userId, urlStr);
|
||||
const safeUrl = await this.getJiraProjectSafeUrl(urlStr);
|
||||
|
||||
const res = await JiraProjectConnection.provisionConnection(this.roomId, userId, { url: safeUrl }, this.provisionOpts);
|
||||
this.pushConnections(res.connection);
|
||||
await this.client.sendNotice(this.roomId, `Room configured to bridge Jira project ${res.connection.projectKey}.`);
|
||||
}
|
||||
|
||||
@botCommand("jira list project", { help: "Show JIRA projects currently connected to.", category: "jira"})
|
||||
@botCommand("jira list project", { help: "Show JIRA projects currently connected to.", category: JiraProjectConnection.ServiceCategory})
|
||||
public async onJiraListProject() {
|
||||
const projects: JiraProjectConnectionState[] = await this.client.getRoomState(this.roomId).catch((err: any) => {
|
||||
if (err.body.errcode === 'M_NOT_FOUND') {
|
||||
@ -177,11 +177,11 @@ export class SetupConnection extends CommandConnection {
|
||||
}
|
||||
}
|
||||
|
||||
@botCommand("jira remove project", { help: "Remove a connection for a JIRA project.", requiredArgs: ["url"], includeUserId: true, category: "jira"})
|
||||
@botCommand("jira remove project", { help: "Remove a connection for a JIRA project.", requiredArgs: ["url"], includeUserId: true, category: JiraProjectConnection.ServiceCategory})
|
||||
public async onJiraRemoveProject(userId: string, urlStr: string) {
|
||||
await this.checkUserPermissions(userId, "jira", JiraProjectConnection.CanonicalEventType);
|
||||
await this.checkUserPermissions(userId, JiraProjectConnection.ServiceCategory, JiraProjectConnection.CanonicalEventType);
|
||||
await this.checkJiraLogin(userId, urlStr);
|
||||
const safeUrl = await this.getJiraProjectSafeUrl(userId, urlStr);
|
||||
const safeUrl = await this.getJiraProjectSafeUrl(urlStr);
|
||||
|
||||
const eventTypes = [
|
||||
JiraProjectConnection.CanonicalEventType,
|
||||
@ -207,7 +207,7 @@ export class SetupConnection extends CommandConnection {
|
||||
return this.client.sendHtmlNotice(this.roomId, md.renderInline(`Room no longer bridged to Jira project \`${safeUrl}\`.`));
|
||||
}
|
||||
|
||||
@botCommand("webhook", { help: "Create an inbound webhook.", requiredArgs: ["name"], includeUserId: true, category: "generic"})
|
||||
@botCommand("webhook", { help: "Create an inbound webhook.", requiredArgs: ["name"], includeUserId: true, category: GenericHookConnection.ServiceCategory})
|
||||
public async onWebhook(userId: string, name: string) {
|
||||
if (!this.config.generic?.enabled) {
|
||||
throw new CommandError("not-configured", "The bridge is not configured to support webhooks.");
|
||||
@ -234,7 +234,7 @@ export class SetupConnection extends CommandConnection {
|
||||
|
||||
|
||||
|
||||
@botCommand("webhook list", { help: "Show webhooks currently configured.", category: "generic"})
|
||||
@botCommand("webhook list", { help: "Show webhooks currently configured.", category: GenericHookConnection.ServiceCategory})
|
||||
public async onWebhookList() {
|
||||
const webhooks: GenericHookConnectionState[] = await this.client.getRoomState(this.roomId).catch((err: any) => {
|
||||
if (err.body.errcode === 'M_NOT_FOUND') {
|
||||
@ -263,9 +263,9 @@ export class SetupConnection extends CommandConnection {
|
||||
}
|
||||
}
|
||||
|
||||
@botCommand("webhook remove", { help: "Remove a webhook from the room.", requiredArgs: ["name"], includeUserId: true, category: "generic"})
|
||||
@botCommand("webhook remove", { help: "Remove a webhook from the room.", requiredArgs: ["name"], includeUserId: true, category: GenericHookConnection.ServiceCategory})
|
||||
public async onWebhookRemove(userId: string, name: string) {
|
||||
await this.checkUserPermissions(userId, "generic", GenericHookConnection.CanonicalEventType);
|
||||
await this.checkUserPermissions(userId, GenericHookConnection.ServiceCategory, GenericHookConnection.CanonicalEventType);
|
||||
|
||||
const event = await this.client.getRoomStateEvent(this.roomId, GenericHookConnection.CanonicalEventType, name).catch((err: any) => {
|
||||
if (err.body.errcode === 'M_NOT_FOUND') {
|
||||
@ -284,13 +284,13 @@ export class SetupConnection extends CommandConnection {
|
||||
return this.client.sendHtmlNotice(this.roomId, md.renderInline(`Removed webhook \`${name}\``));
|
||||
}
|
||||
|
||||
@botCommand("figma file", { help: "Bridge a Figma file to the room.", requiredArgs: ["url"], includeUserId: true, category: "figma"})
|
||||
@botCommand("figma file", { help: "Bridge a Figma file to the room.", requiredArgs: ["url"], includeUserId: true, category: FigmaFileConnection.ServiceCategory})
|
||||
public async onFigma(userId: string, url: string) {
|
||||
if (!this.config.figma) {
|
||||
throw new CommandError("not-configured", "The bridge is not configured to support Figma.");
|
||||
}
|
||||
|
||||
await this.checkUserPermissions(userId, "figma", FigmaFileConnection.CanonicalEventType);
|
||||
await this.checkUserPermissions(userId, FigmaFileConnection.ServiceCategory, FigmaFileConnection.CanonicalEventType);
|
||||
|
||||
const res = /https:\/\/www\.figma\.com\/file\/(\w+).+/.exec(url);
|
||||
if (!res) {
|
||||
@ -302,13 +302,13 @@ export class SetupConnection extends CommandConnection {
|
||||
return this.client.sendHtmlNotice(this.roomId, md.renderInline(`Room configured to bridge Figma file.`));
|
||||
}
|
||||
|
||||
@botCommand("feed", { help: "Bridge an RSS/Atom feed to the room.", requiredArgs: ["url"], optionalArgs: ["label"], includeUserId: true, category: "feeds"})
|
||||
@botCommand("feed", { help: "Bridge an RSS/Atom feed to the room.", requiredArgs: ["url"], optionalArgs: ["label"], includeUserId: true, category: FeedConnection.ServiceCategory})
|
||||
public async onFeed(userId: string, url: string, label?: string) {
|
||||
if (!this.config.feeds?.enabled) {
|
||||
throw new CommandError("not-configured", "The bridge is not configured to support feeds.");
|
||||
}
|
||||
|
||||
await this.checkUserPermissions(userId, "feed", FeedConnection.CanonicalEventType);
|
||||
await this.checkUserPermissions(userId,FeedConnection.ServiceCategory, FeedConnection.CanonicalEventType);
|
||||
|
||||
// provisionConnection will check it again, but won't give us a nice CommandError on failure
|
||||
try {
|
||||
@ -327,7 +327,7 @@ export class SetupConnection extends CommandConnection {
|
||||
return this.client.sendHtmlNotice(this.roomId, md.renderInline(`Room configured to bridge \`${url}\``));
|
||||
}
|
||||
|
||||
@botCommand("feed list", { help: "Show feeds currently subscribed to. Supported formats `json` and `yaml`.", optionalArgs: ["format"], category: "feeds"})
|
||||
@botCommand("feed list", { help: "Show feeds currently subscribed to. Supported formats `json` and `yaml`.", optionalArgs: ["format"], category: FeedConnection.ServiceCategory})
|
||||
public async onFeedList(format?: string) {
|
||||
const useJsonFormat = format?.toLowerCase() === 'json';
|
||||
const useYamlFormat = format?.toLowerCase() === 'yaml';
|
||||
@ -373,7 +373,7 @@ export class SetupConnection extends CommandConnection {
|
||||
|
||||
@botCommand("feed remove", { help: "Unsubscribe from an RSS/Atom feed.", requiredArgs: ["url"], includeUserId: true, category: "feeds"})
|
||||
public async onFeedRemove(userId: string, url: string) {
|
||||
await this.checkUserPermissions(userId, "feed", FeedConnection.CanonicalEventType);
|
||||
await this.checkUserPermissions(userId, FeedConnection.ServiceCategory, FeedConnection.CanonicalEventType);
|
||||
|
||||
const event = await this.client.getRoomStateEvent(this.roomId, FeedConnection.CanonicalEventType, url).catch((err: any) => {
|
||||
if (err.body.errcode === 'M_NOT_FOUND') {
|
||||
@ -389,6 +389,36 @@ export class SetupConnection extends CommandConnection {
|
||||
return this.client.sendHtmlNotice(this.roomId, md.renderInline(`Unsubscribed from \`${url}\``));
|
||||
}
|
||||
|
||||
@botCommand("challenghound add", { help: "Bridge a ChallengeHound challenge to the room.", requiredArgs: ["url"], includeUserId: true, category: "challengehound"})
|
||||
public async onChallengeHoundAdd(userId: string, url: string) {
|
||||
if (!this.config.challengeHound) {
|
||||
throw new CommandError("not-configured", "The bridge is not configured to support challengeHound.");
|
||||
}
|
||||
|
||||
await this.checkUserPermissions(userId, HoundConnection.ServiceCategory, HoundConnection.CanonicalEventType);
|
||||
const {connection, challengeName} = await HoundConnection.provisionConnection(this.roomId, userId, { url }, this.provisionOpts);
|
||||
this.pushConnections(connection);
|
||||
return this.client.sendHtmlNotice(this.roomId, md.renderInline(`Room configured to bridge ${challengeName}. Good luck!`));
|
||||
}
|
||||
|
||||
@botCommand("challenghound remove", { help: "Unbridge a ChallengeHound challenge.", requiredArgs: ["urlOrId"], includeUserId: true, category: HoundConnection.ServiceCategory})
|
||||
public async onChallengeHoundRemove(userId: string, urlOrId: string) {
|
||||
await this.checkUserPermissions(userId, HoundConnection.ServiceCategory, HoundConnection.CanonicalEventType);
|
||||
const id = urlOrId.startsWith('http') ? HoundConnection.getIdFromURL(urlOrId) : urlOrId;
|
||||
const event = await this.client.getRoomStateEvent(this.roomId, HoundConnection.CanonicalEventType, id).catch((err: any) => {
|
||||
if (err.body.errcode === 'M_NOT_FOUND') {
|
||||
return null; // not an error to us
|
||||
}
|
||||
throw err;
|
||||
});
|
||||
if (!event || Object.keys(event).length === 0) {
|
||||
throw new CommandError("Invalid feed URL", `Challenge "${id}" is not currently bridged to this room`);
|
||||
}
|
||||
|
||||
await this.client.sendStateEvent(this.roomId, FeedConnection.CanonicalEventType, id, {});
|
||||
return this.client.sendHtmlNotice(this.roomId, md.renderInline(`Unsubscribed from challenge`));
|
||||
}
|
||||
|
||||
@botCommand("setup-widget", {category: "widget", help: "Open the setup widget in the room"})
|
||||
public async onSetupWidget() {
|
||||
if (this.config.widgets?.roomSetupWidget === undefined) {
|
||||
|
@ -14,6 +14,7 @@ export class MemoryStorageProvider extends MSP implements IBridgeStorageProvider
|
||||
private storedFiles = new QuickLRU<string, string>({ maxSize: 128 });
|
||||
private gitlabDiscussionThreads = new Map<string, SerializedGitlabDiscussionThreads>();
|
||||
private feedGuids = new Map<string, Array<string>>();
|
||||
private houndActivityIds = new Map<string, Array<string>>();
|
||||
|
||||
constructor() {
|
||||
super();
|
||||
@ -108,4 +109,20 @@ export class MemoryStorageProvider extends MSP implements IBridgeStorageProvider
|
||||
public async setGitlabDiscussionThreads(connectionId: string, value: SerializedGitlabDiscussionThreads): Promise<void> {
|
||||
this.gitlabDiscussionThreads.set(connectionId, value);
|
||||
}
|
||||
|
||||
async storeHoundActivity(url: string, ...ids: string[]): Promise<void> {
|
||||
let set = this.houndActivityIds.get(url);
|
||||
if (!set) {
|
||||
set = []
|
||||
this.houndActivityIds.set(url, set);
|
||||
}
|
||||
set.unshift(...ids);
|
||||
while (set.length > MAX_FEED_ITEMS) {
|
||||
set.pop();
|
||||
}
|
||||
}
|
||||
async hasSeenHoundActivity(url: string, ...ids: string[]): Promise<string[]> {
|
||||
const existing = this.houndActivityIds.get(url);
|
||||
return existing ? ids.filter((existingGuid) => existing.includes(existingGuid)) : [];
|
||||
}
|
||||
}
|
||||
|
@ -29,6 +29,7 @@ const WIDGET_TOKENS = "widgets.tokens.";
|
||||
const WIDGET_USER_TOKENS = "widgets.user-tokens.";
|
||||
|
||||
const FEED_GUIDS = "feeds.guids.";
|
||||
const HOUND_IDS = "feeds.guids.";
|
||||
|
||||
const log = new Logger("RedisASProvider");
|
||||
|
||||
@ -240,4 +241,25 @@ export class RedisStorageProvider extends RedisStorageContextualProvider impleme
|
||||
}
|
||||
return guids.filter((_guid, index) => res[index][1] !== null);
|
||||
}
|
||||
|
||||
public async storeHoundActivity(url: string, ...guids: string[]): Promise<void> {
|
||||
const feedKey = `${HOUND_IDS}${url}`;
|
||||
await this.redis.lpush(feedKey, ...guids);
|
||||
await this.redis.ltrim(feedKey, 0, MAX_FEED_ITEMS);
|
||||
}
|
||||
|
||||
public async hasSeenHoundActivity(url: string, ...guids: string[]): Promise<string[]> {
|
||||
let multi = this.redis.multi();
|
||||
const feedKey = `${HOUND_IDS}${url}`;
|
||||
|
||||
for (const guid of guids) {
|
||||
multi = multi.lpos(feedKey, guid);
|
||||
}
|
||||
const res = await multi.exec();
|
||||
if (res === null) {
|
||||
// Just assume we've seen none.
|
||||
return [];
|
||||
}
|
||||
return guids.filter((_guid, index) => res[index][1] !== null);
|
||||
}
|
||||
}
|
||||
|
@ -28,4 +28,6 @@ export interface IBridgeStorageProvider extends IAppserviceStorageProvider, ISto
|
||||
storeFeedGuids(url: string, ...guids: string[]): Promise<void>;
|
||||
hasSeenFeed(url: string): Promise<boolean>;
|
||||
hasSeenFeedGuids(url: string, ...guids: string[]): Promise<string[]>;
|
||||
storeHoundActivity(id: string, ...guids: string[]): Promise<void>;
|
||||
hasSeenHoundActivity(id: string, ...guids: string[]): Promise<string[]>;
|
||||
}
|
@ -12,6 +12,7 @@ import { GithubInstance, GITHUB_CLOUD_URL } from "../github/GithubInstance";
|
||||
import { Logger } from "matrix-appservice-bridge";
|
||||
import { BridgeConfigCache } from "./sections/cache";
|
||||
import { BridgeConfigQueue } from "./sections";
|
||||
import { DefaultConfigRoot } from "./Defaults";
|
||||
|
||||
const log = new Logger("Config");
|
||||
|
||||
@ -450,6 +451,10 @@ export interface BridgeConfigSentry {
|
||||
environment?: string;
|
||||
}
|
||||
|
||||
export interface BridgeConfigChallengeHound {
|
||||
token?: string;
|
||||
}
|
||||
|
||||
|
||||
export interface BridgeConfigRoot {
|
||||
bot?: BridgeConfigBot;
|
||||
@ -473,6 +478,7 @@ export interface BridgeConfigRoot {
|
||||
serviceBots?: BridgeConfigServiceBot[];
|
||||
webhook?: BridgeConfigWebhook;
|
||||
widgets?: BridgeWidgetConfigYAML;
|
||||
challengeHound?: BridgeConfigChallengeHound;
|
||||
}
|
||||
|
||||
export class BridgeConfig {
|
||||
@ -510,6 +516,8 @@ export class BridgeConfig {
|
||||
public readonly figma?: BridgeConfigFigma;
|
||||
@configKey("Configure this to enable RSS/Atom feed support", true)
|
||||
public readonly feeds?: BridgeConfigFeeds;
|
||||
@configKey("Configure Challenge Hound support", true)
|
||||
public readonly challengeHound?: BridgeConfigChallengeHound;
|
||||
@configKey("Define profile information for the bot user", true)
|
||||
public readonly bot?: BridgeConfigBot;
|
||||
@configKey("Define additional bot users for specific services", true)
|
||||
@ -534,6 +542,8 @@ export class BridgeConfig {
|
||||
@hideKey()
|
||||
private readonly bridgePermissions: BridgePermissions;
|
||||
|
||||
|
||||
|
||||
constructor(configData: BridgeConfigRoot, env?: {[key: string]: string|undefined}) {
|
||||
this.bridge = configData.bridge;
|
||||
assert.ok(this.bridge);
|
||||
@ -554,6 +564,7 @@ export class BridgeConfig {
|
||||
this.bot = configData.bot;
|
||||
this.serviceBots = configData.serviceBots;
|
||||
this.metrics = configData.metrics;
|
||||
this.challengeHound = configData.challengeHound;
|
||||
|
||||
// TODO: Formalize env support
|
||||
if (env?.CFG_QUEUE_MONOLITHIC && ["false", "off", "no"].includes(env.CFG_QUEUE_MONOLITHIC)) {
|
||||
@ -756,6 +767,9 @@ remove "useLegacySledStore" from your configuration file, and restart Hookshot.
|
||||
if (this.jira) {
|
||||
services.push("jira");
|
||||
}
|
||||
if (this.challengeHound) {
|
||||
services.push("challengehound");
|
||||
}
|
||||
return services;
|
||||
}
|
||||
|
||||
|
129
src/hound/reader.ts
Normal file
@ -0,0 +1,129 @@
|
||||
import axios from "axios";
|
||||
import { ConnectionManager } from "../ConnectionManager";
|
||||
import { HoundConnection, HoundPayload, HoundActivity } from "../Connections/HoundConnection";
|
||||
import { MessageQueue } from "../MessageQueue";
|
||||
import { IBridgeStorageProvider } from "../Stores/StorageProvider";
|
||||
import { BridgeConfigChallengeHound } from "../config/Config";
|
||||
import { Logger } from "matrix-appservice-bridge";
|
||||
|
||||
const log = new Logger("HoundReader");
|
||||
|
||||
export class HoundReader {
|
||||
private connections: HoundConnection[];
|
||||
private challengeIds: string[];
|
||||
private timeout?: NodeJS.Timeout;
|
||||
private shouldRun = true;
|
||||
private readonly houndClient: axios.AxiosInstance;
|
||||
|
||||
get sleepingInterval() {
|
||||
return 60000 / (this.challengeIds.length || 1);
|
||||
}
|
||||
|
||||
constructor(
|
||||
config: BridgeConfigChallengeHound,
|
||||
private readonly connectionManager: ConnectionManager,
|
||||
private readonly queue: MessageQueue,
|
||||
private readonly storage: IBridgeStorageProvider,
|
||||
) {
|
||||
this.connections = this.connectionManager.getAllConnectionsOfType(HoundConnection);
|
||||
this.challengeIds = this.connections.map(c => c.challengeId);
|
||||
this.houndClient = axios.create({
|
||||
headers: {
|
||||
'Authorization': config.token,
|
||||
}
|
||||
});
|
||||
|
||||
connectionManager.on('new-connection', newConnection => {
|
||||
if (!(newConnection instanceof HoundConnection)) {
|
||||
return;
|
||||
}
|
||||
if (!this.challengeIds.includes(newConnection.challengeId)) {
|
||||
log.info(`Connection added, adding "${newConnection.challengeId}" to queue`);
|
||||
this.challengeIds.push(newConnection.challengeId);
|
||||
}
|
||||
});
|
||||
connectionManager.on('connection-removed', removed => {
|
||||
if (!(removed instanceof HoundConnection)) {
|
||||
return;
|
||||
}
|
||||
let shouldKeepUrl = false;
|
||||
this.connections = this.connections.filter(c => {
|
||||
// Cheeky reuse of iteration to determine if we should remove this URL.
|
||||
if (c.connectionId !== removed.connectionId) {
|
||||
shouldKeepUrl = shouldKeepUrl || c.challengeId === removed.challengeId;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
});
|
||||
if (shouldKeepUrl) {
|
||||
log.info(`Connection removed, but not removing "${removed.challengeId}" as it is still in use`);
|
||||
return;
|
||||
}
|
||||
log.info(`Connection removed, removing "${removed.challengeId}" from queue`);
|
||||
this.challengeIds = this.challengeIds.filter(u => u !== removed.challengeId)
|
||||
});
|
||||
|
||||
log.debug('Loaded challenge IDs:', [...this.challengeIds].join(', '));
|
||||
void this.pollChallenges();
|
||||
}
|
||||
|
||||
public stop() {
|
||||
this.shouldRun = false;
|
||||
if (this.timeout) {
|
||||
clearTimeout(this.timeout);
|
||||
}
|
||||
}
|
||||
|
||||
public async poll(challengeId: string) {
|
||||
const resAct = await this.houndClient.get(`https://api.challengehound.com/challenges/${challengeId}/activities?limit=10`);
|
||||
const activites = resAct.data as HoundActivity[];
|
||||
const seen = await this.storage.hasSeenHoundActivity(challengeId, ...activites.map(a => a.id));
|
||||
for (const activity of activites) {
|
||||
if (seen.includes(activity.id)) {
|
||||
continue;
|
||||
}
|
||||
this.queue.push<HoundPayload>({
|
||||
eventName: "hound.activity",
|
||||
sender: "HoundReader",
|
||||
data: {
|
||||
challengeId,
|
||||
activity: activity,
|
||||
}
|
||||
});
|
||||
}
|
||||
await this.storage.storeHoundActivity(challengeId, ...activites.map(a => a.id))
|
||||
}
|
||||
|
||||
public async pollChallenges(): Promise<void> {
|
||||
log.debug(`Checking for updates`);
|
||||
|
||||
const fetchingStarted = Date.now();
|
||||
|
||||
const challengeId = this.challengeIds.pop();
|
||||
let sleepFor = this.sleepingInterval;
|
||||
|
||||
if (challengeId) {
|
||||
try {
|
||||
await this.poll(challengeId);
|
||||
const elapsed = Date.now() - fetchingStarted;
|
||||
sleepFor = Math.max(this.sleepingInterval - elapsed, 0);
|
||||
log.debug(`Activity fetching took ${elapsed / 1000}s, sleeping for ${sleepFor / 1000}s`);
|
||||
|
||||
if (elapsed > this.sleepingInterval) {
|
||||
log.warn(`It took us longer to update the activities than the expected interval`);
|
||||
}
|
||||
} finally {
|
||||
this.challengeIds.splice(0, 0, challengeId);
|
||||
}
|
||||
} else {
|
||||
log.debug(`No activites available to poll`);
|
||||
}
|
||||
|
||||
this.timeout = setTimeout(() => {
|
||||
if (!this.shouldRun) {
|
||||
return;
|
||||
}
|
||||
void this.pollChallenges();
|
||||
}, sleepFor);
|
||||
}
|
||||
}
|