UNPKG

matrix-react-sdk

Version:
248 lines (244 loc) 40.6 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _matrix = require("matrix-js-sdk/src/matrix"); var _fileSaver = require("file-saver"); var _logger = require("matrix-js-sdk/src/logger"); var _sanitizeFilename = _interopRequireDefault(require("sanitize-filename")); var _exportUtils = require("./exportUtils"); var _DecryptFile = require("../DecryptFile"); var _Media = require("../../customisations/Media"); var _DateUtils = require("../../DateUtils"); var _EventUtils = require("../EventUtils"); var _languageHandler = require("../../languageHandler"); var _SdkConfig = _interopRequireDefault(require("../../SdkConfig")); function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; } /* Copyright 2024 New Vector Ltd. Copyright 2021, 2022 The Matrix.org Foundation C.I.C. SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only Please see LICENSE files in the repository root for full details. */ class Exporter { constructor(room, exportType, exportOptions, setProgressText) { (0, _defineProperty2.default)(this, "files", []); (0, _defineProperty2.default)(this, "cancelled", false); this.room = room; this.exportType = exportType; this.exportOptions = exportOptions; this.setProgressText = setProgressText; if (exportOptions.maxSize < 1 * 1024 * 1024 || // Less than 1 MB exportOptions.maxSize > 8000 * 1024 * 1024 || // More than 8 GB !!exportOptions.numberOfMessages && exportOptions.numberOfMessages > 10 ** 8 || exportType === _exportUtils.ExportType.LastNMessages && !exportOptions.numberOfMessages) { throw new Error("Invalid export options"); } window.addEventListener("beforeunload", this.onBeforeUnload); } get destinationFileName() { return this.makeFileNameNoExtension(_SdkConfig.default.get().brand) + ".zip"; } onBeforeUnload(e) { e.preventDefault(); return e.returnValue = (0, _languageHandler._t)("export_chat|unload_confirm"); } updateProgress(progress, log = true, show = true) { if (log) _logger.logger.log(progress); if (show) this.setProgressText(progress); } addFile(filePath, blob) { const file = { name: filePath, blob }; this.files.push(file); } makeFileNameNoExtension(brand = "matrix") { // First try to use the real name of the room, then a translated copy of a generic name, // then finally hardcoded default to guarantee we'll have a name. const safeRoomName = (0, _sanitizeFilename.default)(this.room.name ?? (0, _languageHandler._t)("common|unnamed_room")).trim() || "Unnamed Room"; const safeDate = (0, _DateUtils.formatFullDateNoDayISO)(new Date()).replace(/:/g, "-"); // ISO format automatically removes a lot of stuff for us const safeBrand = (0, _sanitizeFilename.default)(brand); return `${safeBrand} - ${safeRoomName} - Chat Export - ${safeDate}`; } async downloadZIP() { const filename = this.destinationFileName; const filenameWithoutExt = filename.substring(0, filename.lastIndexOf(".")); // take off the extension const { default: JSZip } = await Promise.resolve().then(() => _interopRequireWildcard(require("jszip"))); const zip = new JSZip(); // Create a writable stream to the directory if (!this.cancelled) this.updateProgress((0, _languageHandler._t)("export_chat|generating_zip"));else return this.cleanUp(); for (const file of this.files) zip.file(filenameWithoutExt + "/" + file.name, file.blob); const content = await zip.generateAsync({ type: "blob" }); (0, _fileSaver.saveAs)(content, filenameWithoutExt + ".zip"); } cleanUp() { _logger.logger.log("Cleaning up..."); window.removeEventListener("beforeunload", this.onBeforeUnload); return ""; } async cancelExport() { _logger.logger.log("Cancelling export..."); this.cancelled = true; } downloadPlainText(fileName, text) { const content = new Blob([text], { type: "text/plain" }); (0, _fileSaver.saveAs)(content, fileName); } setEventMetadata(event) { const roomState = this.room.currentState; const sender = event.getSender(); event.sender = !!sender && roomState?.getSentinelMember(sender) || null; if (event.getType() === "m.room.member") { event.target = roomState?.getSentinelMember(event.getStateKey()) ?? null; } return event; } getLimit() { let limit; switch (this.exportType) { case _exportUtils.ExportType.LastNMessages: // validated in constructor that numberOfMessages is defined // when export type is LastNMessages limit = this.exportOptions.numberOfMessages; break; default: limit = 10 ** 8; } return limit; } async getRequiredEvents() { const eventMapper = this.room.client.getEventMapper(); let prevToken = null; let events = []; if (this.exportType === _exportUtils.ExportType.Timeline) { events = this.room.getLiveTimeline().getEvents(); } else { let limit = this.getLimit(); while (limit) { const eventsPerCrawl = Math.min(limit, 1000); const res = await this.room.client.createMessagesRequest(this.room.roomId, prevToken, eventsPerCrawl, _matrix.Direction.Backward); if (this.cancelled) { this.cleanUp(); return []; } if (res.chunk.length === 0) break; limit -= res.chunk.length; const matrixEvents = res.chunk.map(eventMapper); for (const mxEv of matrixEvents) { // if (this.exportOptions.startDate && mxEv.getTs() < this.exportOptions.startDate) { // // Once the last message received is older than the start date, we break out of both the loops // limit = 0; // break; // } events.push(mxEv); } if (this.exportType === _exportUtils.ExportType.LastNMessages) { this.updateProgress((0, _languageHandler._t)("export_chat|fetched_n_events_with_total", { count: events.length, total: this.exportOptions.numberOfMessages })); } else { this.updateProgress((0, _languageHandler._t)("export_chat|fetched_n_events", { count: events.length })); } prevToken = res.end ?? null; } // Reverse the events so that we preserve the order events.reverse(); } const decryptionPromises = events.filter(event => event.isEncrypted()).map(event => { return this.room.client.decryptEventIfNeeded(event, { emit: false }); }); // Wait for all the events to get decrypted. await Promise.all(decryptionPromises); for (let i = 0; i < events.length; i++) this.setEventMetadata(events[i]); return events; } /** * Decrypts if necessary, and fetches media from a matrix event * @param event - matrix event with media event content * @resolves when media has been fetched * @throws if media was unable to be fetched */ async getMediaBlob(event) { let blob = undefined; try { const isEncrypted = event.isEncrypted(); const content = event.getContent(); const shouldDecrypt = isEncrypted && content.hasOwnProperty("file") && event.getType() !== "m.sticker"; if (shouldDecrypt) { blob = await (0, _DecryptFile.decryptFile)(content.file); } else { const media = (0, _Media.mediaFromContent)(content); if (!media.srcHttp) { throw new Error("Cannot fetch without srcHttp"); } const image = await fetch(media.srcHttp); blob = await image.blob(); } } catch (err) { _logger.logger.log("Error decrypting media"); } if (!blob) { throw new Error("Unable to fetch file"); } return blob; } splitFileName(file) { const lastDot = file.lastIndexOf("."); if (lastDot === -1) return [file, ""]; const fileName = file.slice(0, lastDot); const ext = file.slice(lastDot + 1); return [fileName, "." + ext]; } getFilePath(event) { const mediaType = event.getContent().msgtype; let fileDirectory; switch (mediaType) { case "m.image": fileDirectory = "images"; break; case "m.video": fileDirectory = "videos"; break; case "m.audio": fileDirectory = "audio"; break; default: fileDirectory = event.getType() === "m.sticker" ? "stickers" : "files"; } const fileDate = (0, _DateUtils.formatFullDateNoDay)(new Date(event.getTs())); let [fileName, fileExt] = this.splitFileName(event.getContent().body); if (event.getType() === "m.sticker") fileExt = ".png"; if ((0, _EventUtils.isVoiceMessage)(event)) fileExt = ".ogg"; return fileDirectory + "/" + fileName + "-" + fileDate + fileExt; } isReply(event) { const isEncrypted = event.isEncrypted(); // If encrypted, in_reply_to lies in event.event.content const content = isEncrypted ? event.event.content : event.getContent(); const relatesTo = content["m.relates_to"]; return !!(relatesTo && relatesTo["m.in_reply_to"]); } isAttachment(mxEv) { const attachmentTypes = ["m.sticker", "m.image", "m.file", "m.video", "m.audio"]; return mxEv.getType() === attachmentTypes[0] || attachmentTypes.includes(mxEv.getContent().msgtype); } } exports.default = Exporter; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_matrix","require","_fileSaver","_logger","_sanitizeFilename","_interopRequireDefault","_exportUtils","_DecryptFile","_Media","_DateUtils","_EventUtils","_languageHandler","_SdkConfig","_getRequireWildcardCache","e","WeakMap","r","t","_interopRequireWildcard","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","Exporter","constructor","room","exportType","exportOptions","setProgressText","_defineProperty2","maxSize","numberOfMessages","ExportType","LastNMessages","Error","window","addEventListener","onBeforeUnload","destinationFileName","makeFileNameNoExtension","SdkConfig","brand","preventDefault","returnValue","_t","updateProgress","progress","log","show","logger","addFile","filePath","blob","file","name","files","push","safeRoomName","sanitizeFilename","trim","safeDate","formatFullDateNoDayISO","Date","replace","safeBrand","downloadZIP","filename","filenameWithoutExt","substring","lastIndexOf","JSZip","Promise","resolve","then","zip","cancelled","cleanUp","content","generateAsync","type","saveAs","removeEventListener","cancelExport","downloadPlainText","fileName","text","Blob","setEventMetadata","event","roomState","currentState","sender","getSender","getSentinelMember","getType","target","getStateKey","getLimit","limit","getRequiredEvents","eventMapper","client","getEventMapper","prevToken","events","Timeline","getLiveTimeline","getEvents","eventsPerCrawl","Math","min","res","createMessagesRequest","roomId","Direction","Backward","chunk","length","matrixEvents","map","mxEv","count","total","end","reverse","decryptionPromises","filter","isEncrypted","decryptEventIfNeeded","emit","all","getMediaBlob","undefined","getContent","shouldDecrypt","decryptFile","media","mediaFromContent","srcHttp","image","fetch","err","splitFileName","lastDot","slice","ext","getFilePath","mediaType","msgtype","fileDirectory","fileDate","formatFullDateNoDay","getTs","fileExt","body","isVoiceMessage","isReply","relatesTo","isAttachment","attachmentTypes","includes","exports"],"sources":["../../../src/utils/exportUtils/Exporter.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2021, 2022 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport { Direction, MatrixEvent, Room } from \"matrix-js-sdk/src/matrix\";\nimport { MediaEventContent } from \"matrix-js-sdk/src/types\";\nimport { saveAs } from \"file-saver\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport sanitizeFilename from \"sanitize-filename\";\n\nimport { ExportType, IExportOptions } from \"./exportUtils\";\nimport { decryptFile } from \"../DecryptFile\";\nimport { mediaFromContent } from \"../../customisations/Media\";\nimport { formatFullDateNoDay, formatFullDateNoDayISO } from \"../../DateUtils\";\nimport { isVoiceMessage } from \"../EventUtils\";\nimport { _t } from \"../../languageHandler\";\nimport SdkConfig from \"../../SdkConfig\";\n\ntype BlobFile = {\n    name: string;\n    blob: Blob;\n};\n\nexport default abstract class Exporter {\n    protected files: BlobFile[] = [];\n    protected cancelled = false;\n\n    protected constructor(\n        protected room: Room,\n        protected exportType: ExportType,\n        protected exportOptions: IExportOptions,\n        protected setProgressText: React.Dispatch<React.SetStateAction<string>>,\n    ) {\n        if (\n            exportOptions.maxSize < 1 * 1024 * 1024 || // Less than 1 MB\n            exportOptions.maxSize > 8000 * 1024 * 1024 || // More than 8 GB\n            (!!exportOptions.numberOfMessages && exportOptions.numberOfMessages > 10 ** 8) ||\n            (exportType === ExportType.LastNMessages && !exportOptions.numberOfMessages)\n        ) {\n            throw new Error(\"Invalid export options\");\n        }\n        window.addEventListener(\"beforeunload\", this.onBeforeUnload);\n    }\n\n    public get destinationFileName(): string {\n        return this.makeFileNameNoExtension(SdkConfig.get().brand) + \".zip\";\n    }\n\n    protected onBeforeUnload(e: BeforeUnloadEvent): string {\n        e.preventDefault();\n        return (e.returnValue = _t(\"export_chat|unload_confirm\"));\n    }\n\n    protected updateProgress(progress: string, log = true, show = true): void {\n        if (log) logger.log(progress);\n        if (show) this.setProgressText(progress);\n    }\n\n    protected addFile(filePath: string, blob: Blob): void {\n        const file = {\n            name: filePath,\n            blob,\n        };\n        this.files.push(file);\n    }\n\n    protected makeFileNameNoExtension(brand = \"matrix\"): string {\n        // First try to use the real name of the room, then a translated copy of a generic name,\n        // then finally hardcoded default to guarantee we'll have a name.\n        const safeRoomName = sanitizeFilename(this.room.name ?? _t(\"common|unnamed_room\")).trim() || \"Unnamed Room\";\n        const safeDate = formatFullDateNoDayISO(new Date()).replace(/:/g, \"-\"); // ISO format automatically removes a lot of stuff for us\n        const safeBrand = sanitizeFilename(brand);\n        return `${safeBrand} - ${safeRoomName} - Chat Export - ${safeDate}`;\n    }\n\n    protected async downloadZIP(): Promise<string | void> {\n        const filename = this.destinationFileName;\n        const filenameWithoutExt = filename.substring(0, filename.lastIndexOf(\".\")); // take off the extension\n        const { default: JSZip } = await import(\"jszip\");\n\n        const zip = new JSZip();\n        // Create a writable stream to the directory\n        if (!this.cancelled) this.updateProgress(_t(\"export_chat|generating_zip\"));\n        else return this.cleanUp();\n\n        for (const file of this.files) zip.file(filenameWithoutExt + \"/\" + file.name, file.blob);\n\n        const content = await zip.generateAsync({ type: \"blob\" });\n        saveAs(content, filenameWithoutExt + \".zip\");\n    }\n\n    protected cleanUp(): string {\n        logger.log(\"Cleaning up...\");\n        window.removeEventListener(\"beforeunload\", this.onBeforeUnload);\n        return \"\";\n    }\n\n    public async cancelExport(): Promise<void> {\n        logger.log(\"Cancelling export...\");\n        this.cancelled = true;\n    }\n\n    protected downloadPlainText(fileName: string, text: string): void {\n        const content = new Blob([text], { type: \"text/plain\" });\n        saveAs(content, fileName);\n    }\n\n    protected setEventMetadata(event: MatrixEvent): MatrixEvent {\n        const roomState = this.room.currentState;\n        const sender = event.getSender();\n        event.sender = (!!sender && roomState?.getSentinelMember(sender)) || null;\n        if (event.getType() === \"m.room.member\") {\n            event.target = roomState?.getSentinelMember(event.getStateKey()!) ?? null;\n        }\n        return event;\n    }\n\n    public getLimit(): number {\n        let limit: number;\n        switch (this.exportType) {\n            case ExportType.LastNMessages:\n                // validated in constructor that numberOfMessages is defined\n                // when export type is LastNMessages\n                limit = this.exportOptions.numberOfMessages!;\n                break;\n            default:\n                limit = 10 ** 8;\n        }\n        return limit;\n    }\n\n    protected async getRequiredEvents(): Promise<MatrixEvent[]> {\n        const eventMapper = this.room.client.getEventMapper();\n\n        let prevToken: string | null = null;\n\n        let events: MatrixEvent[] = [];\n        if (this.exportType === ExportType.Timeline) {\n            events = this.room.getLiveTimeline().getEvents();\n        } else {\n            let limit = this.getLimit();\n            while (limit) {\n                const eventsPerCrawl = Math.min(limit, 1000);\n                const res = await this.room.client.createMessagesRequest(\n                    this.room.roomId,\n                    prevToken,\n                    eventsPerCrawl,\n                    Direction.Backward,\n                );\n\n                if (this.cancelled) {\n                    this.cleanUp();\n                    return [];\n                }\n\n                if (res.chunk.length === 0) break;\n\n                limit -= res.chunk.length;\n\n                const matrixEvents: MatrixEvent[] = res.chunk.map(eventMapper);\n\n                for (const mxEv of matrixEvents) {\n                    // if (this.exportOptions.startDate && mxEv.getTs() < this.exportOptions.startDate) {\n                    //     // Once the last message received is older than the start date, we break out of both the loops\n                    //     limit = 0;\n                    //     break;\n                    // }\n                    events.push(mxEv);\n                }\n\n                if (this.exportType === ExportType.LastNMessages) {\n                    this.updateProgress(\n                        _t(\"export_chat|fetched_n_events_with_total\", {\n                            count: events.length,\n                            total: this.exportOptions.numberOfMessages,\n                        }),\n                    );\n                } else {\n                    this.updateProgress(\n                        _t(\"export_chat|fetched_n_events\", {\n                            count: events.length,\n                        }),\n                    );\n                }\n\n                prevToken = res.end ?? null;\n            }\n            // Reverse the events so that we preserve the order\n            events.reverse();\n        }\n\n        const decryptionPromises = events\n            .filter((event) => event.isEncrypted())\n            .map((event) => {\n                return this.room.client.decryptEventIfNeeded(event, { emit: false });\n            });\n\n        // Wait for all the events to get decrypted.\n        await Promise.all(decryptionPromises);\n\n        for (let i = 0; i < events.length; i++) this.setEventMetadata(events[i]);\n\n        return events;\n    }\n\n    /**\n     * Decrypts if necessary, and fetches media from a matrix event\n     * @param event - matrix event with media event content\n     * @resolves when media has been fetched\n     * @throws if media was unable to be fetched\n     */\n    protected async getMediaBlob(event: MatrixEvent): Promise<Blob> {\n        let blob: Blob | undefined = undefined;\n        try {\n            const isEncrypted = event.isEncrypted();\n            const content = event.getContent<MediaEventContent>();\n            const shouldDecrypt = isEncrypted && content.hasOwnProperty(\"file\") && event.getType() !== \"m.sticker\";\n            if (shouldDecrypt) {\n                blob = await decryptFile(content.file);\n            } else {\n                const media = mediaFromContent(content);\n                if (!media.srcHttp) {\n                    throw new Error(\"Cannot fetch without srcHttp\");\n                }\n                const image = await fetch(media.srcHttp);\n                blob = await image.blob();\n            }\n        } catch (err) {\n            logger.log(\"Error decrypting media\");\n        }\n        if (!blob) {\n            throw new Error(\"Unable to fetch file\");\n        }\n        return blob;\n    }\n\n    public splitFileName(file: string): string[] {\n        const lastDot = file.lastIndexOf(\".\");\n        if (lastDot === -1) return [file, \"\"];\n        const fileName = file.slice(0, lastDot);\n        const ext = file.slice(lastDot + 1);\n        return [fileName, \".\" + ext];\n    }\n\n    public getFilePath(event: MatrixEvent): string {\n        const mediaType = event.getContent().msgtype;\n        let fileDirectory: string;\n        switch (mediaType) {\n            case \"m.image\":\n                fileDirectory = \"images\";\n                break;\n            case \"m.video\":\n                fileDirectory = \"videos\";\n                break;\n            case \"m.audio\":\n                fileDirectory = \"audio\";\n                break;\n            default:\n                fileDirectory = event.getType() === \"m.sticker\" ? \"stickers\" : \"files\";\n        }\n        const fileDate = formatFullDateNoDay(new Date(event.getTs()));\n        let [fileName, fileExt] = this.splitFileName(event.getContent().body);\n\n        if (event.getType() === \"m.sticker\") fileExt = \".png\";\n        if (isVoiceMessage(event)) fileExt = \".ogg\";\n\n        return fileDirectory + \"/\" + fileName + \"-\" + fileDate + fileExt;\n    }\n\n    protected isReply(event: MatrixEvent): boolean {\n        const isEncrypted = event.isEncrypted();\n        // If encrypted, in_reply_to lies in event.event.content\n        const content = isEncrypted ? event.event.content! : event.getContent();\n        const relatesTo = content[\"m.relates_to\"];\n        return !!(relatesTo && relatesTo[\"m.in_reply_to\"]);\n    }\n\n    protected isAttachment(mxEv: MatrixEvent): boolean {\n        const attachmentTypes = [\"m.sticker\", \"m.image\", \"m.file\", \"m.video\", \"m.audio\"];\n        return mxEv.getType() === attachmentTypes[0] || attachmentTypes.includes(mxEv.getContent().msgtype!);\n    }\n\n    public abstract export(): Promise<void>;\n}\n"],"mappings":"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","ignoreList":[]}