Skip to content

Commit

Permalink
postgres db
Browse files Browse the repository at this point in the history
  • Loading branch information
jesopo authored and Gnuxie committed Sep 30, 2022
1 parent 9828025 commit 839fe05
Show file tree
Hide file tree
Showing 4 changed files with 112 additions and 30 deletions.
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@
"matrix-appservice-bridge": "^5.0.0",
"nedb": "^1.8.0",
"parse-duration": "^1.0.2",
"pg": "^8.8.0",
"shell-quote": "^1.7.3",
"yaml": "^2.1.1"
},
Expand Down
67 changes: 43 additions & 24 deletions src/appservice/AppService.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,20 +16,23 @@
*/

import { randomUUID } from "crypto";
import { AppServiceRegistration, Bridge, Cli, Request, WeakEvent, BridgeContext, MatrixUser, UserBridgeStore, RemoteUser } from "matrix-appservice-bridge";
import { AppServiceRegistration, Bridge, Cli, Request, WeakEvent, BridgeContext, MatrixUser } from "matrix-appservice-bridge";
import { MatrixClient } from "matrix-bot-sdk";
// needed by appservice irc, though it looks completely dead.
import * as Datastore from "nedb";
import { MjolnirManager } from ".//MjolnirManager";
import { DataStore, PgDataStore } from ".//datastore";
import { Api } from "./Api";
// ts-node src/appservice/AppService.ts -r -u "http://localhost:9000" # remember to add the registration to homeserver.yaml! you probably want host.docker.internal as the hostname of the appservice if using mx-tester
// ts-node src/appservice/AppService -p 9000 # to start.

export class MjolnirAppService {

public readonly dataStore: DataStore;
public readonly bridge: Bridge;
public readonly mjolnirManager: MjolnirManager = new MjolnirManager();

public constructor() {
this.dataStore = new PgDataStore("foo bar baz");
new Api("http://localhost:8081", this).start(9001);
this.bridge = new Bridge({
homeserverUrl: "http://localhost:8081",
Expand All @@ -39,40 +42,56 @@ export class MjolnirAppService {
onUserQuery: this.onUserQuery.bind(this),
onEvent: this.onEvent.bind(this),
},
userStore: new UserBridgeStore(new Datastore()),
suppressEcho: false,
});
}

public async provisionNewMjolnir(requestingUserId: string): Promise<[string, string]> {
// FIXME: we need to restrict who can do it (special list? ban remote users?)
const issuedMjolnirs = await this.bridge.getUserStore()!.getRemoteUsersFromMatrixId(requestingUserId);
if (issuedMjolnirs.length === 0) {
public async initStoredMjolnirs(): Promise<void> {
for (var mjolnirRecord of await this.dataStore.list()) {
const [_mjolnirUserId, mjolnirClient] = await this.makeMatrixClient(mjolnirRecord.localPart);
await this.mjolnirManager.makeInstance(
mjolnirRecord.owner,
mjolnirRecord.managementRoom,
mjolnirClient,
);
}
}

public async makeMatrixClient(localPart: string): Promise<[string, MatrixClient]> {
// Now we need to make one of the transparent mjolnirs and add it to the monitor.
const mjIntent = await this.bridge.getIntentFromLocalpart(`mjolnir_${randomUUID()}`);
const mjIntent = await this.bridge.getIntentFromLocalpart(localPart);
await mjIntent.ensureRegistered();
// we're only doing this because it's complaining about missing profiles.
// actually the user id wasn't even right, so this might not be necessary anymore.
await mjIntent.ensureProfile('Mjolnir');
return [mjIntent.userId, mjIntent.matrixClient];
}

public async provisionNewMjolnir(requestingUserId: string): Promise<[string, string]> {
// FIXME: we need to restrict who can do it (special list? ban remote users?)
const provisionedMjolnirs = await this.dataStore.lookupByOwner(requestingUserId);
if (provisionedMjolnirs.length === 0) {
const mjolnirLocalPart = `mjolnir_${randomUUID()}`;
const [mjolnirUserId, mjolnirClient] = await this.makeMatrixClient(mjolnirLocalPart);

const managementRoomId = await mjolnirClient.createRoom({
preset: 'private_chat',
invite: [requestingUserId],
name: `${requestingUserId}'s mjolnir`
});

const mjolnir = await this.mjolnirManager.makeInstance(requestingUserId, managementRoomId, mjolnirClient);
await mjolnir.createFirstList(requestingUserId, "list");

const managementRoomId = (await mjIntent.createRoom({
createAsClient: true,
options: {
preset: 'private_chat',
invite: [requestingUserId],
name: `${requestingUserId}'s mjolnir`
}
})).room_id;
await this.dataStore.store({
localPart: mjolnirLocalPart,
owner: requestingUserId,
managementRoom: managementRoomId,
});

await this.mjolnirManager.createNew(requestingUserId, managementRoomId, mjIntent.matrixClient);
// Technically the mjolnir is a remote user, but also not because it's matrix-matrix.
//const mjAsRemote = new RemoteUser(mjIntent.userId)
//const bridgeStore = this.bridge.getUserStore()!;
//bridgeStore.setRemoteUser()
await this.bridge.getUserStore()!.linkUsers(new MatrixUser(requestingUserId), new RemoteUser(mjIntent.userId));
return [mjIntent.userId, managementRoomId];
return [mjolnirUserId, managementRoomId];
} else {
throw new Error(`User: ${requestingUserId} has already provisioned ${issuedMjolnirs.length} mjolnirs.`);
throw new Error(`User: ${requestingUserId} has already provisioned ${provisionedMjolnirs.length} mjolnirs.`);
}
}

Expand Down
8 changes: 2 additions & 6 deletions src/appservice/MjolnirManager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,14 +14,10 @@ export class MjolnirManager {
return config;
}

public async createNew(requestingUserId: string, managementRoomId: string, client: MatrixClient) {
// FIXME: We should be creating the intent here and generating the id surely?
// rather than externally...
// FIXME: We need to verify that we haven't stored a mjolnir already if we aren't doing the above.

public async makeInstance(requestingUserId: string, managementRoomId: string, client: MatrixClient): Promise<ManagedMjolnir> {
const managedMjolnir = new ManagedMjolnir(await Mjolnir.setupMjolnirFromConfig(client, this.getDefaultMjolnirConfig(managementRoomId)));
await managedMjolnir.createFirstList(requestingUserId, 'list')
this.mjolnirs.set(await client.getUserId(), managedMjolnir);
return managedMjolnir;
}

public onEvent(request: Request<WeakEvent>, context: BridgeContext) {
Expand Down
66 changes: 66 additions & 0 deletions src/appservice/datastore.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
import { Client } from "pg";

export interface MjolnirRecord {
localPart: string,
owner: string,
managementRoom: string,
}

export interface DataStore {
init(): Promise<void>;

list(): Promise<MjolnirRecord[]>;

store(mjolnirRecord: MjolnirRecord): Promise<void>;

lookupByOwner(owner: string): Promise<MjolnirRecord[]>;

lookupByMxid(mxid: string): Promise<MjolnirRecord[]>;
}

export class PgDataStore implements DataStore {
private pgClient: Client;

constructor(connectionString: string) {
this.pgClient = new Client({ connectionString: connectionString });
}

public async init(): Promise<void> {
await this.pgClient.connect();
}

public async list(): Promise<MjolnirRecord[]> {
const result = await this.pgClient.query<MjolnirRecord>("SELECT mxid, owner, managementRoom FROM mjolnir");

if (!result.rowCount) {
return [];
}

return result.rows;
}

public async store(mjolnirRecord: MjolnirRecord): Promise<void> {
await this.pgClient.query(
"INSERT INTO mjolnir (mxid, owner, managementRoom) VALUES ($1, $2, $3)",
[mjolnirRecord.mxid, mjolnirRecord.owner, mjolnirRecord.managementRoom],
);
}

public async lookupByOwner(owner: string): Promise<MjolnirRecord[]> {
const result = await this.pgClient.query<MjolnirRecord>(
"SELECT mxid, owner FROM mjolnir WHERE owner = $1",
[owner],
);

return result.rows;
}

public async lookupByMxid(mxid: string): Promise<MjolnirRecord[]> {
const result = await this.pgClient.query<MjolnirRecord>(
"SELECT mxid, owner, managementRoom FROM mjolnir WHERE mxid = $1",
[mxid],
);

return result.rows;
}
}

0 comments on commit 839fe05

Please sign in to comment.