General code coverage
This commit is contained in:
@@ -127,8 +127,9 @@ async function channelToKState(channel, guild, di) {
|
||||
const everyoneCanMentionEveryone = dUtils.hasAllPermissions(everyonePermissions, ["MentionEveryone"])
|
||||
|
||||
const spacePowerDetails = await mUtils.getEffectivePower(guildSpaceID, [], di.api)
|
||||
spacePowerDetails.powerLevels.users ??= {}
|
||||
const spaceCreatorsAndFounders = spacePowerDetails.allCreators
|
||||
.concat(Object.entries(spacePowerDetails.powerLevels.users ?? {}).filter(([, power]) => power >= spacePowerDetails.tombstone).map(([mxid]) => mxid))
|
||||
.concat(Object.entries(spacePowerDetails.powerLevels.users).filter(([, power]) => power >= spacePowerDetails.tombstone).map(([mxid]) => mxid))
|
||||
|
||||
const globalAdmins = select("member_power", ["mxid", "power_level"], {room_id: "*"}).all()
|
||||
const globalAdminPower = globalAdmins.reduce((a, c) => (a[c.mxid] = c.power_level, a), {})
|
||||
|
||||
@@ -50,3 +50,15 @@ test("emoji: spy needs u+fe0f in the middle", async t => {
|
||||
test("emoji: couple needs u+fe0f in the middle", async t => {
|
||||
t.equal(await encodeEmoji("👩❤👩", null), "%F0%9F%91%A9%E2%80%8D%E2%9D%A4%EF%B8%8F%E2%80%8D%F0%9F%91%A9")
|
||||
})
|
||||
|
||||
test("emoji: exact known emojis are returned", async t => {
|
||||
t.equal(await encodeEmoji("mxc://cadence.moe/qWmbXeRspZRLPcjseyLmeyXC", "hippo"), "hippo%3A230201364309868544")
|
||||
})
|
||||
|
||||
test("emoji: inexact emojis are guessed by name", async t => {
|
||||
t.equal(await encodeEmoji("mxc://example.invalid/a", "hippo"), "hippo%3A230201364309868544")
|
||||
})
|
||||
|
||||
test("emoji: unknown custom emoji returns null", async t => {
|
||||
t.equal(await encodeEmoji("mxc://example.invalid/a", "silly"), null)
|
||||
})
|
||||
|
||||
@@ -18,21 +18,6 @@ class MatrixServerError extends Error {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {Response} res
|
||||
* @param {object} opts
|
||||
*/
|
||||
async function makeMatrixServerError(res, opts = {}) {
|
||||
delete opts.headers?.["Authorization"]
|
||||
if (res.headers.get("content-type") === "application/json") {
|
||||
return new MatrixServerError(await res.json(), opts)
|
||||
} else if (res.headers.get("content-type")?.startsWith("text/")) {
|
||||
return new MatrixServerError({errcode: "CX_SERVER_ERROR", error: `Server returned HTTP status ${res.status}`, message: await res.text()}, opts)
|
||||
} else {
|
||||
return new MatrixServerError({errcode: "CX_SERVER_ERROR", error: `Server returned HTTP status ${res.status}`, content_type: res.headers.get("content-type")}, opts)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {undefined | string | object | streamWeb.ReadableStream | stream.Readable} body
|
||||
* @returns {Promise<string | streamWeb.ReadableStream | stream.Readable | Buffer>}
|
||||
@@ -52,6 +37,21 @@ async function _convertBody(body) {
|
||||
|
||||
/* c8 ignore start */
|
||||
|
||||
/**
|
||||
* @param {Response} res
|
||||
* @param {object} opts
|
||||
*/
|
||||
async function makeMatrixServerError(res, opts = {}) {
|
||||
delete opts.headers?.["Authorization"]
|
||||
if (res.headers.get("content-type") === "application/json") {
|
||||
return new MatrixServerError(await res.json(), opts)
|
||||
} else if (res.headers.get("content-type")?.startsWith("text/")) {
|
||||
return new MatrixServerError({errcode: "CX_SERVER_ERROR", error: `Server returned HTTP status ${res.status}`, message: await res.text()}, opts)
|
||||
} else {
|
||||
return new MatrixServerError({errcode: "CX_SERVER_ERROR", error: `Server returned HTTP status ${res.status}`, content_type: res.headers.get("content-type")}, opts)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {string} method
|
||||
* @param {string} url
|
||||
|
||||
@@ -138,14 +138,17 @@ async function getViaServers(roomID, api) {
|
||||
candidates.push(reg.ooye.server_name)
|
||||
// Candidate 1: Highest joined non-sim non-bot power level user in the room
|
||||
// https://github.com/matrix-org/matrix-react-sdk/blob/552c65db98b59406fb49562e537a2721c8505517/src/utils/permalinks/Permalinks.ts#L172
|
||||
/* c8 ignore next */
|
||||
const call = "getEffectivePower" in api ? api.getEffectivePower(roomID, [bot], api) : getEffectivePower(roomID, [bot], api)
|
||||
const {allCreators, powerLevels} = await call
|
||||
const sorted = allCreators.concat(Object.entries(powerLevels.users ?? {}).sort((a, b) => b[1] - a[1]).map(([mxid]) => mxid)) // Highest...
|
||||
powerLevels.users ??= {}
|
||||
const sorted = allCreators.concat(Object.entries(powerLevels.users).sort((a, b) => b[1] - a[1]).map(([mxid]) => mxid)) // Highest...
|
||||
for (const mxid of sorted) {
|
||||
if (!(mxid in joined)) continue // joined...
|
||||
if (userRegex.some(r => mxid.match(r))) continue // non-sim non-bot...
|
||||
const match = mxid.match(/:(.*)/)
|
||||
assert(match)
|
||||
/* c8 ignore next - should be already covered by the userRegex test, but let's be explicit */
|
||||
if (candidates.includes(match[1])) continue // from a different server
|
||||
candidates.push(match[1])
|
||||
break
|
||||
|
||||
@@ -1,7 +1,8 @@
|
||||
// @ts-check
|
||||
|
||||
const {select} = require("../passthrough")
|
||||
const {test} = require("supertape")
|
||||
const {eventSenderIsFromDiscord, getEventIDHash, MatrixStringBuilder, getViaServers, roomHasAtLeastVersion} = require("./utils")
|
||||
const {eventSenderIsFromDiscord, getEventIDHash, MatrixStringBuilder, getViaServers, roomHasAtLeastVersion, removeCreatorsFromPowerLevels, setUserPower} = require("./utils")
|
||||
const util = require("util")
|
||||
|
||||
/** @param {string[]} mxids */
|
||||
@@ -201,4 +202,219 @@ test("getViaServers: only considers power levels of currently joined members", a
|
||||
t.deepEqual(result, ["cadence.moe", "tractor.invalid", "thecollective.invalid", "selfhosted.invalid"])
|
||||
})
|
||||
|
||||
test("roomHasAtLeastVersion: v9 < v11", t => {
|
||||
t.equal(roomHasAtLeastVersion("9", 11), false)
|
||||
})
|
||||
|
||||
test("roomHasAtLeastVersion: v12 >= v11", t => {
|
||||
t.equal(roomHasAtLeastVersion("12", 11), true)
|
||||
})
|
||||
|
||||
test("roomHasAtLeastVersion: v12 >= v12", t => {
|
||||
t.equal(roomHasAtLeastVersion("12", 12), true)
|
||||
})
|
||||
|
||||
test("roomHasAtLeastVersion: custom versions never match", t => {
|
||||
t.equal(roomHasAtLeastVersion("moe.cadence.silly", 11), false)
|
||||
})
|
||||
|
||||
test("removeCreatorsFromPowerLevels: removes the creator from a v12 room", t => {
|
||||
t.deepEqual(removeCreatorsFromPowerLevels({
|
||||
type: "m.room.create",
|
||||
state_key: "",
|
||||
sender: "@_ooye_bot:cadence.moe",
|
||||
room_id: "!example",
|
||||
event_id: "$create",
|
||||
origin_server_ts: 0,
|
||||
content: {
|
||||
room_version: "12"
|
||||
}
|
||||
}, {
|
||||
users: {
|
||||
"@_ooye_bot:cadence.moe": 100
|
||||
}
|
||||
}), {
|
||||
users: {
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
test("removeCreatorsFromPowerLevels: removes all creators from a v12 room", t => {
|
||||
t.deepEqual(removeCreatorsFromPowerLevels({
|
||||
type: "m.room.create",
|
||||
state_key: "",
|
||||
sender: "@_ooye_bot:cadence.moe",
|
||||
room_id: "!example",
|
||||
event_id: "$create",
|
||||
origin_server_ts: 0,
|
||||
content: {
|
||||
additional_creators: ["@cadence:cadence.moe"],
|
||||
room_version: "12"
|
||||
}
|
||||
}, {
|
||||
users: {
|
||||
"@_ooye_bot:cadence.moe": 100,
|
||||
"@cadence:cadence.moe": 100
|
||||
}
|
||||
}), {
|
||||
users: {
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
test("removeCreatorsFromPowerLevels: doesn't touch a v11 room", t => {
|
||||
t.deepEqual(removeCreatorsFromPowerLevels({
|
||||
type: "m.room.create",
|
||||
state_key: "",
|
||||
sender: "@_ooye_bot:cadence.moe",
|
||||
room_id: "!example",
|
||||
event_id: "$create",
|
||||
origin_server_ts: 0,
|
||||
content: {
|
||||
additional_creators: ["@cadence:cadence.moe"],
|
||||
room_version: "11"
|
||||
}
|
||||
}, {
|
||||
users: {
|
||||
"@_ooye_bot:cadence.moe": 100,
|
||||
"@cadence:cadence.moe": 100
|
||||
}
|
||||
}), {
|
||||
users: {
|
||||
"@_ooye_bot:cadence.moe": 100,
|
||||
"@cadence:cadence.moe": 100
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
test("set user power: no-op", async t => {
|
||||
let called = 0
|
||||
await setUserPower("!room", "@cadence:cadence.moe", 0, {
|
||||
async getStateEvent(roomID, type, key) {
|
||||
called++
|
||||
t.equal(roomID, "!room")
|
||||
t.equal(type, "m.room.power_levels")
|
||||
t.equal(key, "")
|
||||
return {}
|
||||
},
|
||||
async getStateEventOuter(roomID, type, key) {
|
||||
called++
|
||||
t.equal(roomID, "!room")
|
||||
t.equal(type, "m.room.create")
|
||||
t.equal(key, "")
|
||||
return {
|
||||
type: "m.room.create",
|
||||
state_key: "",
|
||||
sender: "@_ooye_bot:cadence.moe",
|
||||
room_id: "!room",
|
||||
origin_server_ts: 0,
|
||||
event_id: "$create",
|
||||
content: {
|
||||
room_version: "11"
|
||||
}
|
||||
}
|
||||
},
|
||||
/* c8 ignore next 4 */
|
||||
async sendState() {
|
||||
called++
|
||||
throw new Error("should not try to send state")
|
||||
}
|
||||
})
|
||||
t.equal(called, 2)
|
||||
})
|
||||
|
||||
test("set user power: bridge bot must promote unprivileged users", async t => {
|
||||
let called = 0
|
||||
await setUserPower("!room", "@cadence:cadence.moe", 100, {
|
||||
async getStateEvent(roomID, type, key) {
|
||||
called++
|
||||
t.equal(roomID, "!room")
|
||||
t.equal(type, "m.room.power_levels")
|
||||
t.equal(key, "")
|
||||
return {
|
||||
users: {"@_ooye_bot:cadence.moe": 100}
|
||||
}
|
||||
},
|
||||
async getStateEventOuter(roomID, type, key) {
|
||||
called++
|
||||
t.equal(roomID, "!room")
|
||||
t.equal(type, "m.room.create")
|
||||
t.equal(key, "")
|
||||
return {
|
||||
type: "m.room.create",
|
||||
state_key: "",
|
||||
sender: "@_ooye_bot:cadence.moe",
|
||||
room_id: "!room",
|
||||
origin_server_ts: 0,
|
||||
event_id: "$create",
|
||||
content: {
|
||||
room_version: "11"
|
||||
}
|
||||
}
|
||||
},
|
||||
async sendState(roomID, type, key, content, mxid) {
|
||||
called++
|
||||
t.equal(roomID, "!room")
|
||||
t.equal(type, "m.room.power_levels")
|
||||
t.equal(key, "")
|
||||
t.deepEqual(content, {
|
||||
users: {
|
||||
"@_ooye_bot:cadence.moe": 100,
|
||||
"@cadence:cadence.moe": 100
|
||||
}
|
||||
})
|
||||
t.equal(mxid, undefined)
|
||||
return "$sent"
|
||||
}
|
||||
})
|
||||
t.equal(called, 3)
|
||||
})
|
||||
|
||||
test("set user power: privileged users must demote themselves", async t => {
|
||||
let called = 0
|
||||
await setUserPower("!room", "@cadence:cadence.moe", 0, {
|
||||
async getStateEvent(roomID, type, key) {
|
||||
called++
|
||||
t.equal(roomID, "!room")
|
||||
t.equal(type, "m.room.power_levels")
|
||||
t.equal(key, "")
|
||||
return {
|
||||
users: {
|
||||
"@cadence:cadence.moe": 100,
|
||||
"@_ooye_bot:cadence.moe": 100
|
||||
}
|
||||
}
|
||||
},
|
||||
async getStateEventOuter(roomID, type, key) {
|
||||
called++
|
||||
t.equal(roomID, "!room")
|
||||
t.equal(type, "m.room.create")
|
||||
t.equal(key, "")
|
||||
return {
|
||||
type: "m.room.create",
|
||||
state_key: "",
|
||||
sender: "@_ooye_bot:cadence.moe",
|
||||
room_id: "!room",
|
||||
origin_server_ts: 0,
|
||||
event_id: "$create",
|
||||
content: {
|
||||
room_version: "11"
|
||||
}
|
||||
}
|
||||
},
|
||||
async sendState(roomID, type, key, content, mxid) {
|
||||
called++
|
||||
t.equal(roomID, "!room")
|
||||
t.equal(type, "m.room.power_levels")
|
||||
t.equal(key, "")
|
||||
t.deepEqual(content, {
|
||||
users: {"@_ooye_bot:cadence.moe": 100}
|
||||
})
|
||||
t.equal(mxid, "@cadence:cadence.moe")
|
||||
return "$sent"
|
||||
}
|
||||
})
|
||||
t.equal(called, 3)
|
||||
})
|
||||
|
||||
module.exports.mockGetEffectivePower = mockGetEffectivePower
|
||||
|
||||
@@ -31,14 +31,13 @@ function getSnow(event) {
|
||||
/** @type {Map<string, Promise<string>>} */
|
||||
const cache = new Map()
|
||||
|
||||
/** @param {string} url */
|
||||
/** @param {string | undefined} url */
|
||||
function timeUntilExpiry(url) {
|
||||
const params = new URL(url).searchParams
|
||||
const ex = params.get("ex")
|
||||
assert(ex) // refreshed urls from the discord api always include this parameter
|
||||
const time = parseInt(ex, 16)*1000 - Date.now()
|
||||
if (time > 0) return time
|
||||
return false
|
||||
}
|
||||
|
||||
function defineMediaProxyHandler(domain) {
|
||||
@@ -71,6 +70,7 @@ function defineMediaProxyHandler(domain) {
|
||||
refreshed = await promise
|
||||
const time = timeUntilExpiry(refreshed)
|
||||
assert(time) // the just-refreshed URL will always be in the future
|
||||
/* c8 ignore next 3 */
|
||||
setTimeout(() => {
|
||||
cache.delete(url)
|
||||
}, time).unref()
|
||||
|
||||
@@ -5,20 +5,6 @@ const {test} = require("supertape")
|
||||
const {router} = require("../../../test/web")
|
||||
const {MatrixServerError} = require("../../matrix/mreq")
|
||||
|
||||
const snow = {
|
||||
channel: {
|
||||
async refreshAttachmentURLs(attachments) {
|
||||
if (typeof attachments === "string") attachments = [attachments]
|
||||
return {
|
||||
refreshed_urls: attachments.map(a => ({
|
||||
original: a,
|
||||
refreshed: a + `?ex=${Math.floor(Date.now() / 1000 + 3600).toString(16)}`
|
||||
}))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
test("web download discord: access denied if not a known attachment", async t => {
|
||||
const [error] = await tryToCatch(() =>
|
||||
router.test("get", "/download/discordcdn/:channel_id/:attachment_id/:file_name", {
|
||||
@@ -27,7 +13,19 @@ test("web download discord: access denied if not a known attachment", async t =>
|
||||
attachment_id: "2",
|
||||
file_name: "image.png"
|
||||
},
|
||||
snow
|
||||
snow: {
|
||||
channel: {
|
||||
async refreshAttachmentURLs(attachments) {
|
||||
if (typeof attachments === "string") attachments = [attachments]
|
||||
return {
|
||||
refreshed_urls: attachments.map(a => ({
|
||||
original: a,
|
||||
refreshed: a + `?ex=${Math.floor(Date.now() / 1000 + 3600).toString(16)}`
|
||||
}))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
)
|
||||
t.ok(error)
|
||||
@@ -42,8 +40,43 @@ test("web download discord: works if a known attachment", async t => {
|
||||
file_name: "image.png"
|
||||
},
|
||||
event,
|
||||
snow
|
||||
snow: {
|
||||
channel: {
|
||||
async refreshAttachmentURLs(attachments) {
|
||||
if (typeof attachments === "string") attachments = [attachments]
|
||||
return {
|
||||
refreshed_urls: attachments.map(a => ({
|
||||
original: a,
|
||||
refreshed: a + `?ex=${Math.floor(Date.now() / 1000 + 3600).toString(16)}`
|
||||
}))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
t.equal(event.node.res.statusCode, 302)
|
||||
t.match(event.node.res.getHeader("location"), /https:\/\/cdn.discordapp.com\/attachments\/655216173696286746\/1314358913482621010\/image\.png\?ex=/)
|
||||
})
|
||||
|
||||
test("web download discord: uses cache", async t => {
|
||||
let notCalled = true
|
||||
const event = {}
|
||||
await router.test("get", "/download/discordcdn/:channel_id/:attachment_id/:file_name", {
|
||||
params: {
|
||||
channel_id: "655216173696286746",
|
||||
attachment_id: "1314358913482621010",
|
||||
file_name: "image.png"
|
||||
},
|
||||
event,
|
||||
snow: {
|
||||
channel: {
|
||||
// @ts-ignore
|
||||
async refreshAttachmentURLs(attachments) {
|
||||
notCalled = false
|
||||
throw new Error("tried to refresh when it should be in cache")
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
t.ok(notCalled)
|
||||
})
|
||||
|
||||
@@ -314,6 +314,13 @@ test("api invite: can invite to a moderated guild", async t => {
|
||||
guest_can_join: false,
|
||||
num_joined_members: 2,
|
||||
}
|
||||
yield {
|
||||
room_id: spaceID,
|
||||
children_state: [],
|
||||
guest_can_join: false,
|
||||
num_joined_members: 2,
|
||||
room_type: "m.space"
|
||||
}
|
||||
},
|
||||
async sendState(roomID, type, key, content) {
|
||||
called++
|
||||
|
||||
@@ -68,8 +68,7 @@ as.router.get("/api/message", defineEventHandler(async event => {
|
||||
}
|
||||
}
|
||||
if (!matrix_author.displayname) matrix_author.displayname = mxid
|
||||
if (matrix_author.avatar_url) matrix_author.avatar_url = mUtils.getPublicUrlForMxc(matrix_author.avatar_url)
|
||||
else matrix_author.avatar_url = null
|
||||
matrix_author.avatar_url = mUtils.getPublicUrlForMxc(matrix_author.avatar_url) || null
|
||||
matrix_author["mxid"] = mxid
|
||||
}
|
||||
|
||||
|
||||
@@ -148,7 +148,7 @@ test("web link space: check that inviting user has PL 50", async t => {
|
||||
t.equal(roomID, "!zTMspHVUBhFLLSdmnS:cadence.moe")
|
||||
t.equal(type, "m.room.power_levels")
|
||||
t.equal(key, "")
|
||||
return {users: {"@_ooye_bot:cadence.moe": 100}}
|
||||
return {users: {"@_ooye_bot:cadence.moe": 100}, events: {"m.room.tombstone": 150}}
|
||||
},
|
||||
async getStateEventOuter(roomID, type, key) {
|
||||
called++
|
||||
@@ -163,7 +163,7 @@ test("web link space: check that inviting user has PL 50", async t => {
|
||||
event_id: "$create",
|
||||
origin_server_ts: 0,
|
||||
content: {
|
||||
room_version: "11"
|
||||
room_version: "12"
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -194,7 +194,7 @@ test("web link space: successfully adds entry to database and loads page", async
|
||||
t.equal(roomID, "!zTMspHVUBhFLLSdmnS:cadence.moe")
|
||||
t.equal(type, "m.room.power_levels")
|
||||
t.equal(key, "")
|
||||
return {users: {"@_ooye_bot:cadence.moe": 100, "@cadence:cadence.moe": 50}}
|
||||
return {users: {"@cadence:cadence.moe": 50}}
|
||||
},
|
||||
async getStateEventOuter(roomID, type, key) {
|
||||
called++
|
||||
@@ -204,12 +204,12 @@ test("web link space: successfully adds entry to database and loads page", async
|
||||
return {
|
||||
type: "m.room.create",
|
||||
state_key: "",
|
||||
sender: "@creator:cadence.moe",
|
||||
sender: "@_ooye_bot:cadence.moe",
|
||||
room_id: "!zTMspHVUBhFLLSdmnS:cadence.moe",
|
||||
event_id: "$create",
|
||||
origin_server_ts: 0,
|
||||
content: {
|
||||
room_version: "11"
|
||||
room_version: "12"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
16
src/web/routes/password.test.js
Normal file
16
src/web/routes/password.test.js
Normal file
@@ -0,0 +1,16 @@
|
||||
// @ts-check
|
||||
|
||||
const tryToCatch = require("try-to-catch")
|
||||
const {test} = require("supertape")
|
||||
const {router} = require("../../../test/web")
|
||||
|
||||
test("web password: stores password", async t => {
|
||||
const event = {}
|
||||
await router.test("post", "/api/password", {
|
||||
body: {
|
||||
password: "password123"
|
||||
},
|
||||
event
|
||||
})
|
||||
t.equal(event.node.res.statusCode, 302)
|
||||
})
|
||||
@@ -140,6 +140,7 @@ file._actuallyUploadDiscordFileToMxc = function(url, res) { throw new Error(`Not
|
||||
require("../src/web/routes/info.test")
|
||||
require("../src/web/routes/link.test")
|
||||
require("../src/web/routes/log-in-with-matrix.test")
|
||||
require("../src/web/routes/password.test")
|
||||
require("../src/discord/utils.test")
|
||||
require("../src/matrix/kstate.test")
|
||||
require("../src/matrix/api.test")
|
||||
|
||||
Reference in New Issue
Block a user