2023-04-30 12:57:30 +00:00
// @ts-check
2023-05-10 10:15:20 +00:00
const assert = require ( "assert" )
2023-05-08 12:58:46 +00:00
const passthrough = require ( "../../passthrough" )
const { discord , sync , db } = passthrough
/** @type {import("../converters/message-to-event")} */
const messageToEvent = sync . require ( "../converters/message-to-event" )
/** @type {import("../../matrix/api")} */
const api = sync . require ( "../../matrix/api" )
/** @type {import("./register-user")} */
const registerUser = sync . require ( "./register-user" )
2023-05-08 20:03:57 +00:00
/** @type {import("../actions/create-room")} */
const createRoom = sync . require ( "../actions/create-room" )
2023-10-09 22:23:51 +00:00
/** @type {import("../../discord/utils")} */
const dUtils = sync . require ( "../../discord/utils" )
2023-04-30 12:57:30 +00:00
/ * *
* @ param { import ( "discord-api-types/v10" ) . GatewayMessageCreateDispatchData } message
2023-06-28 11:38:58 +00:00
* @ param { import ( "discord-api-types/v10" ) . APIGuild } guild
2023-04-30 12:57:30 +00:00
* /
2023-06-28 11:38:58 +00:00
async function sendMessage ( message , guild ) {
2023-05-08 20:03:57 +00:00
const roomID = await createRoom . ensureRoom ( message . channel _id )
2023-05-12 05:35:37 +00:00
2023-05-08 12:58:46 +00:00
let senderMxid = null
2023-10-09 22:23:51 +00:00
if ( ! dUtils . isWebhookMessage ( message ) ) {
2023-08-19 10:54:23 +00:00
if ( message . member ) { // available on a gateway message create event
senderMxid = await registerUser . syncUser ( message . author , message . member , message . guild _id , roomID )
} else { // well, good enough...
senderMxid = await registerUser . ensureSimJoined ( message . author , roomID )
}
2023-05-08 12:58:46 +00:00
}
2023-05-12 05:35:37 +00:00
2023-08-16 08:44:38 +00:00
const events = await messageToEvent . messageToEvent ( message , guild , { } , { api } )
2023-05-12 05:35:37 +00:00
const eventIDs = [ ]
let eventPart = 0 // 0 is primary, 1 is supporting
2023-08-28 05:32:55 +00:00
if ( events . length ) {
db . prepare ( "REPLACE INTO message_channel (message_id, channel_id) VALUES (?, ?)" ) . run ( message . id , message . channel _id )
2023-09-17 11:07:33 +00:00
if ( senderMxid ) api . sendTyping ( roomID , false , senderMxid )
2023-08-28 05:32:55 +00:00
}
2023-05-12 05:35:37 +00:00
for ( const event of events ) {
const eventType = event . $type
/** @type {Pick<typeof event, Exclude<keyof event, "$type">> & { $type?: string }} */
const eventWithoutType = { ... event }
delete eventWithoutType . $type
2023-08-28 04:20:16 +00:00
const useTimestamp = message [ "backfill" ] ? new Date ( message . timestamp ) . getTime ( ) : undefined
const eventID = await api . sendEvent ( roomID , eventType , eventWithoutType , senderMxid , useTimestamp )
2023-08-28 12:05:25 +00:00
db . prepare ( "INSERT INTO event_message (event_id, event_type, event_subtype, message_id, part, source) VALUES (?, ?, ?, ?, ?, 1)" ) . run ( eventID , eventType , event . msgtype || null , message . id , eventPart ) // source 1 = discord
2023-05-12 05:35:37 +00:00
2023-10-06 03:58:18 +00:00
// The primary event is part = 0 and has the most important and distinct information. It is used to provide reply previews, be pinned, and possibly future uses.
// The first event is chosen to be the primary part because it is usually the message text content and is more likely to be distinct.
// For example, "Reply to 'this meme made me think of you'" is more useful than "Replied to image".
eventPart = 1
2023-05-12 05:35:37 +00:00
eventIDs . push ( eventID )
}
return eventIDs
2023-04-30 12:57:30 +00:00
}
2023-05-09 05:13:59 +00:00
module . exports . sendMessage = sendMessage