From f72aa4c5fa2e87a69bf819cb5621808891571ec0 Mon Sep 17 00:00:00 2001 From: balzack Date: Sun, 1 Dec 2024 13:34:31 -0800 Subject: [PATCH] implementing focus module --- app/client/web/src/MediaFiles.ts | 1 - .../src/conversation/useConversation.hook.ts | 11 +- app/sdk/src/contact.ts | 10 +- app/sdk/src/focus.ts | 152 ++++++++++-------- app/sdk/src/stream.ts | 10 +- app/sdk/src/types.ts | 2 +- 6 files changed, 108 insertions(+), 78 deletions(-) diff --git a/app/client/web/src/MediaFiles.ts b/app/client/web/src/MediaFiles.ts index d74553b1..70264947 100644 --- a/app/client/web/src/MediaFiles.ts +++ b/app/client/web/src/MediaFiles.ts @@ -31,7 +31,6 @@ export class MediaFiles implements Media { public async read(source: any): Promise<{ size: number, getData: (position: number, length: number)=>Promise, close: ()=>Promise }> { const data = await this.loadFileData(source); const size = data.byteLength; - const getData = async (position: number, length: number) => { if (position + length > data.byteLength) { throw new Error('invalid read request'); diff --git a/app/client/web/src/conversation/useConversation.hook.ts b/app/client/web/src/conversation/useConversation.hook.ts index d74b1ab1..2dc7ff1f 100644 --- a/app/client/web/src/conversation/useConversation.hook.ts +++ b/app/client/web/src/conversation/useConversation.hook.ts @@ -4,6 +4,9 @@ import { DisplayContext } from '../context/DisplayContext' import { Focus, Topic, AssetSource, HostingMode, TransformType } from 'databag-client-sdk' import { ContextType } from '../context/ContextType' +const img = + ''; + export function useConversation() { const app = useContext(AppContext) as ContextType const display = useContext(DisplayContext) as ContextType @@ -51,6 +54,7 @@ export function useConversation() { } }, add: async (file: File) => { + const { focus } = app.state; if (focus) { const asset = { @@ -58,12 +62,11 @@ export function useConversation() { mimeType: 'image', extension: 'jpg', source: file, - transforms: [ {type: TransformType.Thumb, appId: '1'}, {type: TransformType.Copy, appId: '2'}], + transforms: [ {type: TransformType.Thumb, thumb: ()=>(img), appId: '1'}, {type: TransformType.Copy, appId: '2'}], } - const topicId = await focus.addTopic(false, 'superbasictopic', (assets: {assetId: string, appId: string}[])=>{ - console.log("HERER!!!"); + const topicId = await focus.addTopic(true, 'sealedtopic', (assets: {assetId: string, appId: string}[])=>{ console.log(assets); - return { text: 'addedasset', assets: [{ image: { thumb: '0', full: '1' } }] }; + return { text: 'addedasset', assets: [{ encrypted: { type: 'image', thumb: '0', parts: '1' } }] }; }, [asset], (percent: number)=>{ console.log(percent); }); diff --git a/app/sdk/src/contact.ts b/app/sdk/src/contact.ts index a14c81d4..e13beba9 100644 --- a/app/sdk/src/contact.ts +++ b/app/sdk/src/contact.ts @@ -1029,9 +1029,13 @@ export class ContactModule implements Contact { } if (item.channelKey) { const { messageEncrypted, messageIv } = JSON.parse(item.summary.data); - const { data } = await this.crypto.aesDecrypt(messageEncrypted, messageIv, item.channelKey); - item.unsealedSummary = data; - return true; + if (!messageEncrypted || !messageIv) { + this.log.warn('invalid sealed summary'); + } else { + const { data } = await this.crypto.aesDecrypt(messageEncrypted, messageIv, item.channelKey); + item.unsealedSummary = data; + return true; + } } } catch (err) { this.log.warn(err); diff --git a/app/sdk/src/focus.ts b/app/sdk/src/focus.ts index 6477efac..04c166b3 100644 --- a/app/sdk/src/focus.ts +++ b/app/sdk/src/focus.ts @@ -221,7 +221,6 @@ export class FocusModule implements Focus { this.emitTopics(); } } - this.syncing = false; } } @@ -241,8 +240,12 @@ export class FocusModule implements Focus { xhr.setRequestHeader('Content-Type', 'text/plain'); xhr.progress = progress; xhr.onload = () => { - if (xhr.status >= 200 && xhr.status < 300 && xhr.response?.assetId) { - resolve(xhr.response.assetId) + if (xhr.status >= 200 && xhr.status < 300) { + try { + resolve(JSON.parse(xhr.response).assetId); + } catch (err) { + reject('invalid block response'); + } } else { reject(xhr.statusText) } @@ -271,7 +274,11 @@ export class FocusModule implements Focus { xhr.progress = progress; xhr.onload = () => { if (xhr.status >= 200 && xhr.status < 300) { - resolve(xhr.response) + try { + resolve(JSON.parse(xhr.response)); + } catch (err) { + reject('invalid asset response'); + } } else { reject(xhr.statusText) } @@ -343,7 +350,7 @@ export class FocusModule implements Focus { for (const transform of asset.transforms) { if (transform.type === TransformType.Thumb && transform.thumb) { const assetItem = { - assetId: `${assetItems.size}`, + assetId: `${assetItems.length}`, encrytped: true, hosting: HostingMode.Inline, inline: await transform.thumb(), @@ -351,18 +358,18 @@ export class FocusModule implements Focus { appAsset.push({appId: transform.appId, assetId: assetItem.assetId}); assetItems.push(assetItem); } else if (transform.type === TransformType.Copy) { - const media = await this.file.read(asset.soure); - const split = [] as { blockId: string, blockIv: string }[]; - for (let i = 0; media.size < i * ENCRYPT_BLOCK_SIZE; i++) { - const length = media.size - (i * ENCRYPT_BLOCK_SIZE) > ENCRYPT_BLOCK_SIZE ? ENCRYPT_BLOCK_SIZE : media.size - (i * ENCRYPT_BLOCK_SIZE); - const base64Data = await media.getData(i * ENCRYPT_BLOCK_SIZE, length); + const mediaFile = await this.media.read(asset.source); + const split = [] as { partId: string, blockIv: string }[]; + for (let i = 0; i * ENCRYPT_BLOCK_SIZE < mediaFile.size; i++) { + const length = mediaFile.size - (i * ENCRYPT_BLOCK_SIZE) > ENCRYPT_BLOCK_SIZE ? ENCRYPT_BLOCK_SIZE : mediaFile.size - (i * ENCRYPT_BLOCK_SIZE); + const base64Data = await mediaFile.getData(i * ENCRYPT_BLOCK_SIZE, length); const { ivHex } = await crypto.aesIv(); const { encryptedDataB64 } = await crypto.aesEncrypt(base64Data, ivHex, channelKey); - const blockId = await uploadBlock(encryptedDataB64, topicId, (percent: number) => { console.log(`percent: ${percent}`) }); - split.push({ blockId, blockIv: ivHex }); + const partId = await this.uploadBlock(encryptedDataB64, topicId, (percent: number) => { console.log(`percent: ${percent}`) }); + split.push({ partId, blockIv: ivHex }); } const assetItem = { - assetId: `${assetItems.size}`, + assetId: `${assetItems.length}`, encrypted: true, hosting: HostingMode.Split, split, @@ -406,7 +413,7 @@ export class FocusModule implements Focus { } else if (transform.type === TransformType.Copy && asset.mimeType === 'binary') { const assetId = await this.mirrorFile(asset.source, topicId, (percent: number)=>{console.log(`progress: ${percent}`)}); const assetItem = { - assetId: `${assetItems.size}`, + assetId: `${assetItems.length}`, encrytped: false, hosting: HostingMode.Basic, basic: assetId, @@ -421,7 +428,7 @@ export class FocusModule implements Focus { const transformAssets = await this.transformFile(asset.source, topicId, transforms, (percent: number)=>{console.log(`progress: ${percent}`)}); for (let transformAsset of transformAssets) { const assetItem = { - assetId: `${assetItems.size}`, + assetId: `${assetItems.length}`, encrypted: false, hosting: HostingMode.Basic, basic: transformAsset.assetId, @@ -440,6 +447,9 @@ export class FocusModule implements Focus { const getAsset = (assetId: string) => { const index = parseInt(assetId); const item = assetItems[index]; + if (!item) { + throw new Error('invalid assetId in subject'); + } if (item.hosting === HostingMode.Inline) { return item.inline; } @@ -451,36 +461,38 @@ export class FocusModule implements Focus { } } - const mapped = !assets ? [] : assets.map(asset => { - if (asset.image) { + const filtered = !assets ? [] : assets.filter(asset => { + if (sealed && asset.encrypted) { + return true; + } else if (!sealed && !asset.encrypted) { + return true; + } else { + return false; + } + }); + const mapped = filtered.map(asset => { + if (sealed) { + const { type, thumb, parts } = asset.encrypted; + return { encrypted: { type, thumb: getAsset(thumb), parts: getAsset(parts) } }; + } else if (asset.image) { const { thumb, full } = asset.image; return { image: { thumb: getAsset(thumb), full: getAsset(full) } }; - } - if (asset.video) { + } else if (asset.video) { const { thumb, lq, hd } = asset.video; return { video: { thumb: getAsset(thumb), lq: getAsset(lq), hd: getAsset(hd) } }; - } - if (asset.audio) { + } else if (asset.audio) { const { label, fulle } = asset.audio; return { audio: { label, full: getAsset(full) } }; - } - if (asset.binary) { + } else if (asset.binary) { const { label, extension, data } = asset.binary; return { binary: { label, extension, data: getAsset(data) } }; } - if (asset.encrypted) { - const { type, thumb, parts } = asset.encrypted; - return { encrypted: { type, thumb: getAsset(thumb), parts: getAsset(parts) } }; - } }); const updated = { text, textColor, textSize, assets: mapped }; - -console.log("OK UPDATED: ", updated); - if (sealed) { - const subjectString = JSON.stringify(updated); + const subjectString = JSON.stringify({ message: updated }); const { ivHex } = await crypto.aesIv(); - const { encryptedDataB64 } = await crypto.aesEncrypt(decryptedString, ivHex, channelKey); + const { encryptedDataB64 } = await crypto.aesEncrypt(subjectString, ivHex, channelKey); const data = { messageEncrypted: encryptedDataB64, messageIv: ivHex }; await this.setRemoteChannelTopicSubject(topicId, type, data); } else { @@ -489,7 +501,6 @@ console.log("OK UPDATED: ", updated); return topicId; } - } public async setTopicSubject(topicId: string, type: string, subject: (assets: {assetId: string, appId: string}[])=>any, files: AssetSource[], progress: (percent: number)=>boolean) { @@ -511,7 +522,7 @@ console.log("OK UPDATED: ", updated); for (const transform of asset.transforms) { if (transform.type === TransformType.Thumb && transform.thumb) { const assetItem = { - assetId: `${assetItems.size}`, + assetId: `${assetItems.length}`, encrytped: true, hosting: HostingMode.Inline, inline: await transform.thumb(), @@ -519,18 +530,18 @@ console.log("OK UPDATED: ", updated); appAsset.push({appId: transform.appId, assetId: assetItem.assetId}); assetItems.push(assetItem); } else if (transform.type === TransformType.Copy) { - const media = await this.file.read(asset.soure); - const split = [] as { blockId: string, blockIv: string }[]; - for (let i = 0; media.size < i * ENCRYPT_BLOCK_SIZE; i++) { - const length = media.size - (i * ENCRYPT_BLOCK_SIZE) > ENCRYPT_BLOCK_SIZE ? ENCRYPT_BLOCK_SIZE : media.size - (i * ENCRYPT_BLOCK_SIZE); - const base64Data = await media.getData(i * ENCRYPT_BLOCK_SIZE, length); + const mediaFile = await this.file.read(asset.source); + const split = [] as { partId: string, blockIv: string }[]; + for (let i = 0; i * ENCRYPT_BLOCK_SIZE < mediaFile.size; i++) { + const length = mediaFile.size - (i * ENCRYPT_BLOCK_SIZE) > ENCRYPT_BLOCK_SIZE ? ENCRYPT_BLOCK_SIZE : mediaFile.size - (i * ENCRYPT_BLOCK_SIZE); + const base64Data = await mediaFile.getData(i * ENCRYPT_BLOCK_SIZE, length); const { ivHex } = await crypto.aesIv(); const { encryptedDataB64 } = await crypto.aesEncrypt(base64Data, ivHex, channelKey); - const blockId = await uploadBlock(encryptedDataB64, topicId, (percent: number) => { console.log(`percent: ${percent}`) }); - split.push({ blockId, blockIv: ivHex }); + const partId = await this.uploadBlock(encryptedDataB64, topicId, (percent: number) => { console.log(`percent: ${percent}`) }); + split.push({ partId, blockIv: ivHex }); } const assetItem = { - assetId: `${assetItems.size}`, + assetId: `${assetItems.length}`, encrypted: true, hosting: HostingMode.Split, split, @@ -568,7 +579,7 @@ console.log("OK UPDATED: ", updated); } else if (transform.type === TransformType.Copy && asset.mimeType === 'binary') { const assetId = await this.mirrorFile(asset.source, topicId, (percent: number)=>{console.log(`progress: ${percent}`)}); const assetItem = { - assetId: `${assetItems.size}`, + assetId: `${assetItems.length}`, encrytped: false, hosting: HostingMode.Basic, basic: assetId, @@ -610,34 +621,39 @@ console.log("OK UPDATED: ", updated); } } - const mapped = assets.map(asset => { - if (asset.image) { - const { thumb, full } = asset.image; - return { image: { thumb: getAsset(thumb), full: getAsset(full) } }; + const filtered = !assets ? [] : assets.filter(asset => { + if (sealed && asset.encrypted) { + return true; + } else if (!sealed && !asset.encrypted) { + return true; + } else { + return false; } - if (asset.video) { - const { thumb, lq, hd } = asset.video; - return { video: { thumb: getAsset(thumb), lq: getAsset(lq), hd: getAsset(hd) } }; - } - if (asset.audio) { - const { label, fulle } = asset.audio; - return { audio: { label, full: getAsset(full) } }; - } - if (asset.binary) { - const { label, extension, data } = asset.binary; - return { binary: { label, extension, data: getAsset(data) } }; - } - if (asset.encrypted) { + }); + const mapped = filtered.map(asset => { + if (sealed) { const { type, thumb, parts } = asset.encrypted; return { encrypted: { type, thumb: getAsset(thumb), parts: getAsset(parts) } }; + } else if (asset.image) { + const { thumb, full } = asset.image; + return { image: { thumb: getAsset(thumb), full: getAsset(full) } }; + } else if (asset.video) { + const { thumb, lq, hd } = asset.video; + return { video: { thumb: getAsset(thumb), lq: getAsset(lq), hd: getAsset(hd) } }; + } else if (asset.audio) { + const { label, fulle } = asset.audio; + return { audio: { label, full: getAsset(full) } }; + } else if (asset.binary) { + const { label, extension, data } = asset.binary; + return { binary: { label, extension, data: getAsset(data) } }; } }); const updated = { text, textColor, textSize, assets: mapped }; if (sealed) { - const subjectString = JSON.stringify(updated); + const subjectString = JSON.stringify({ message: updated }); const { ivHex } = await crypto.aesIv(); - const { encryptedDataB64 } = await crypto.aesEncrypt(decryptedString, ivHex, channelKey); + const { encryptedDataB64 } = await crypto.aesEncrypt(subjectString, ivHex, channelKey); const data = { messageEncrypted: encryptedDataB64, messageIv: ivHex }; return await this.setRemoteChannelTopicSubject(topicId, type, data); } else { @@ -667,10 +683,14 @@ console.log("OK UPDATED: ", updated); if (item.detail.sealed && !item.unsealedDetail && this.sealEnabled && this.channelKey && this.crypto) { try { const { messageEncrypted, messageIv } = item.detail.data; - const { data } = await this.crypto.aesDecrypt(messageEncrypted, messageIv, this.channelKey); - const { message } = JSON.parse(data); - item.unsealedDetail = message; - return true; + if (!messageEncrypted || !messageIv) { + this.log.warn('invalid sealed topic'); + } else { + const { data } = await this.crypto.aesDecrypt(messageEncrypted, messageIv, this.channelKey); + const { message } = JSON.parse(data); + item.unsealedDetail = message; + return true; + } } catch (err) { this.log.warn(err); } diff --git a/app/sdk/src/stream.ts b/app/sdk/src/stream.ts index 2eb12648..62b50b59 100644 --- a/app/sdk/src/stream.ts +++ b/app/sdk/src/stream.ts @@ -552,9 +552,13 @@ export class StreamModule { } if (item.channelKey) { const { messageEncrypted, messageIv } = JSON.parse(item.summary.data); - const { data } = await this.crypto.aesDecrypt(messageEncrypted, messageIv, item.channelKey); - item.unsealedSummary = data; - return true; + if (!messageEncrypted || !messageIv) { + this.log.warn('invalid sealed summary'); + } else { + const { data } = await this.crypto.aesDecrypt(messageEncrypted, messageIv, item.channelKey); + item.unsealedSummary = data; + return true; + } } } catch (err) { this.log.warn(err); diff --git a/app/sdk/src/types.ts b/app/sdk/src/types.ts index 8d1441b2..2a8716bf 100644 --- a/app/sdk/src/types.ts +++ b/app/sdk/src/types.ts @@ -111,7 +111,7 @@ export type Tag = { }; export enum HostingMode { - Inline = 'inlien', // sealed or unsealed + Inline = 'inline', // sealed or unsealed Split = 'split', // sealed only, split file into blocks Basic = 'basic', // unsealed only, basic download }