Skip to content

Commit

Permalink
fix: can not draft image payload in ff
Browse files Browse the repository at this point in the history
  • Loading branch information
guanbinrui committed Mar 16, 2020
1 parent 60f11b6 commit 03182fb
Show file tree
Hide file tree
Showing 9 changed files with 51 additions and 44 deletions.
17 changes: 10 additions & 7 deletions src/extension/background-script/SteganographyService.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import { EncodeOptions, DecodeOptions } from 'node-stego/es/stego'
import { getUrl, downloadUrl } from '../../utils/utils'
import { memoizePromise } from '../../utils/memoize'
import { getDimension } from '../../utils/image'
import { decodeArrayBuffer, encodeArrayBuffer } from '../../utils/type-transform/String-ArrayBuffer'

OnlyRunInContext('background', 'SteganographyService')

Expand Down Expand Up @@ -39,10 +40,11 @@ type EncodeImageOptions = {
template?: Template
} & PartialRequired<Required<EncodeOptions>, 'text' | 'pass'>

export async function encodeImage(buf: Uint8Array, options: EncodeImageOptions) {
export async function encodeImage(buf: string | ArrayBuffer, options: EncodeImageOptions) {
const { template } = options
return new Uint8Array(
await encode(buf.buffer, await getMaskBuf(), {
const _buf = typeof buf === 'string' ? decodeArrayBuffer(buf) : buf
return encodeArrayBuffer(
await encode(_buf, await getMaskBuf(), {
...defaultOptions,
fakeMaskPixels: template !== 'default',
cropEdgePixels: true,
Expand All @@ -56,20 +58,21 @@ export async function encodeImage(buf: Uint8Array, options: EncodeImageOptions)

type DecodeImageOptions = PartialRequired<Required<DecodeOptions>, 'pass'>

export async function decodeImage(buf: Uint8Array, options: DecodeImageOptions) {
const dimension = getDimension(buf)
export async function decodeImage(buf: string | ArrayBuffer, options: DecodeImageOptions) {
const _buf = typeof buf === 'string' ? decodeArrayBuffer(buf) : buf
const dimension = getDimension(_buf)
if (!dimensions.some(otherDimension => isSameDimension(dimension, otherDimension))) {
return ''
}
return decode(buf.buffer, await getMaskBuf(), {
return decode(_buf, await getMaskBuf(), {
...defaultOptions,
transformAlgorithm: TransformAlgorithm.FFT1D,
...options,
})
}

export async function decodeImageUrl(url: string, options: DecodeImageOptions) {
return decodeImage(new Uint8Array(await downloadUrl(url)), options)
return decodeImage(await downloadUrl(url), options)
}

export function downloadImage({ buffer }: Uint8Array) {
Expand Down
2 changes: 1 addition & 1 deletion src/extension/mock-service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ export const WelcomeService: Partial<typeof import('./background-script/WelcomeS
}
export const SteganographyService: Partial<typeof import('./background-script/SteganographyService')> = {
async encodeImage() {
return new Uint8Array()
return ''
},
async decodeImage() {
return ''
Expand Down
3 changes: 1 addition & 2 deletions src/social-network-provider/facebook.com/UI/collectPosts.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ import { getEmptyPostInfoByElement, PostInfo, SocialNetworkUI } from '../../../s
import { isMobileFacebook } from '../isMobile'
import { getProfileIdentifierAtFacebook } from '../getPersonIdentifierAtFacebook'
import Services from '../../../extension/service'
import { downloadUrl } from '../../../utils/utils'

const posts = new LiveSelector().querySelectorAll<HTMLDivElement>(
isMobileFacebook ? '.story_body_container ' : '.userContent, .userContent+*+div>div>div>div>div',
Expand Down Expand Up @@ -133,7 +132,7 @@ async function getSteganographyContent(node: DOMProxy) {
await Promise.all(
imgUrls.map(async url => {
try {
const content = await Services.Steganography.decodeImage(new Uint8Array(await downloadUrl(url)), {
const content = await Services.Steganography.decodeImageUrl(url, {
pass,
})
return content.indexOf('🎼') === 0 ? content : ''
Expand Down
22 changes: 12 additions & 10 deletions src/social-network-provider/facebook.com/tasks/uploadToPostBox.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import { SocialNetworkUI, getActivatedUI } from '../../../social-network/ui'
import { untilDocumentReady } from '../../../utils/dom'
import { getUrl, downloadUrl, pasteImageToActiveElements } from '../../../utils/utils'
import Services from '../../../extension/service'
import { decodeArrayBuffer } from '../../../utils/type-transform/String-ArrayBuffer'

export async function uploadToPostBoxFacebook(
text: string,
Expand All @@ -10,16 +11,17 @@ export async function uploadToPostBoxFacebook(
const { warningText, template = 'default' } = options
const { currentIdentity } = getActivatedUI()
const blankImage = await downloadUrl(getUrl(`/maskbook-steganography-${template}.png`))
const secretImage = await Services.Steganography.encodeImage(new Uint8Array(blankImage), {
text,
pass: currentIdentity.value ? currentIdentity.value.identifier.toText() : '',
template,
})

const image = new Uint8Array(secretImage)
await pasteImageToActiveElements(image)
const secretImage = new Uint8Array(
decodeArrayBuffer(
await Services.Steganography.encodeImage(new Uint8Array(blankImage), {
text,
pass: currentIdentity.value ? currentIdentity.value.identifier.toText() : '',
template,
}),
),
)
await pasteImageToActiveElements(secretImage)
await untilDocumentReady()

try {
// Need a better way to find whether the image is pasted into
// throw new Error('auto uploading is undefined')
Expand All @@ -30,7 +32,7 @@ export async function uploadToPostBoxFacebook(
async function uploadFail() {
console.warn('Image not uploaded to the post box')
if (confirm(warningText)) {
await Services.Steganography.downloadImage(image)
await Services.Steganography.downloadImage(secretImage)
}
}
}
23 changes: 12 additions & 11 deletions src/social-network-provider/twitter.com/ui/tasks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import { twitterEncoding } from '../encoding'
import { createTaskStartImmersiveSetupDefault } from '../../../social-network/defaults/taskStartImmersiveSetupDefault'
import { instanceOfTwitterUI } from '.'
import { ProfileIdentifier } from '../../../database/type'
import { encodeArrayBuffer, decodeArrayBuffer } from '../../../utils/type-transform/String-ArrayBuffer'

/**
* Wait for up to 5000 ms
Expand Down Expand Up @@ -80,17 +81,17 @@ const taskUploadToPostBox: SocialNetworkUI['taskUploadToPostBox'] = async (text,
const { warningText, template = 'default' } = options
const { currentIdentity } = getActivatedUI()
const blankImage = await downloadUrl(getUrl(`/maskbook-steganography-${template}.png`))
const secretImage = await Services.Steganography.encodeImage(new Uint8Array(blankImage), {
text,
pass: currentIdentity.value ? currentIdentity.value.identifier.toText() : '',
template,
})

const image = new Uint8Array(secretImage)

await pasteImageToActiveElements(image)
const secretImage = new Uint8Array(
decodeArrayBuffer(
await Services.Steganography.encodeImage(encodeArrayBuffer(blankImage), {
text,
pass: currentIdentity.value ? currentIdentity.value.identifier.toText() : '',
template,
}),
),
)
await pasteImageToActiveElements(secretImage)
await untilDocumentReady()

try {
// Need a better way to find whether the image is pasted into
// throw new Error('auto uploading is undefined')
Expand All @@ -101,7 +102,7 @@ const taskUploadToPostBox: SocialNetworkUI['taskUploadToPostBox'] = async (text,
async function uploadFail() {
console.warn('Image not uploaded to the post box')
if (confirm(warningText)) {
await Services.Steganography.downloadImage(image)
await Services.Steganography.downloadImage(secretImage)
}
}
}
Expand Down
2 changes: 1 addition & 1 deletion src/social-network-provider/twitter.com/utils/fetch.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { regexMatch, downloadUrl } from '../../../utils/utils'
import { regexMatch } from '../../../utils/utils'
import { notNullable } from '../../../utils/assert'
import { defaultTo } from 'lodash-es'
import { nthChild } from '../../../utils/dom'
Expand Down
4 changes: 2 additions & 2 deletions src/utils/__tests__/image.ts
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ function createJPEGBuffer(width: number, height: number) {
// EOI
0xff,
0xd9,
])
]).buffer
}

function createPNGBuffer(width: number, height: number) {
Expand All @@ -82,7 +82,7 @@ function createPNGBuffer(width: number, height: number) {
0, // compression method
0, // filter method
0, // interlace method
])
]).buffer
}

test('Get dimension of JPEG buffer', () => {
Expand Down
16 changes: 8 additions & 8 deletions src/utils/image.ts
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
/* eslint-disable no-bitwise */
import { imgType } from 'node-stego/es/helper'

export function getDimension(buf: Uint8Array) {
export function getDimension(buf: ArrayBuffer) {
const fallback = {
width: 0,
height: 0,
}
switch (imgType(buf)) {
switch (imgType(new Uint8Array(buf))) {
case 'image/jpeg':
return getDimensionAsJPEG(buf) ?? fallback
case 'image/png':
Expand All @@ -16,8 +16,8 @@ export function getDimension(buf: Uint8Array) {
}
}

function getDimensionAsPNG(buf: Uint8Array) {
const dataView = new DataView(buf.buffer, 0, 28)
function getDimensionAsPNG(buf: ArrayBuffer) {
const dataView = new DataView(buf, 0, 28)
return {
width: dataView.getInt32(16),
height: dataView.getInt32(20),
Expand All @@ -29,8 +29,8 @@ function getDimensionAsPNG(buf: Uint8Array) {
*
* @see http://vip.sugovica.hu/Sardi/kepnezo/JPEG%20File%20Layout%20and%20Format.htm
*/
function getDimensionAsJPEG(buf: Uint8Array) {
const dataView = new DataView(buf.buffer)
function getDimensionAsJPEG(buf: ArrayBuffer) {
const dataView = new DataView(buf)
let i = 0
if (
dataView.getUint8(i) === 0xff &&
Expand All @@ -47,9 +47,9 @@ function getDimensionAsJPEG(buf: Uint8Array) {
dataView.getUint8(i + 6) === 0x00
) {
let block_length = dataView.getUint8(i) * 256 + dataView.getUint8(i + 1)
while (i < buf.length) {
while (i < dataView.byteLength) {
i += block_length
if (i >= buf.length) return
if (i >= dataView.byteLength) return
if (dataView.getUint8(i) !== 0xff) return
if (
dataView.getUint8(i + 1) === 0xc0 || // SOF0 marker
Expand Down
6 changes: 4 additions & 2 deletions src/utils/type-transform/String-ArrayBuffer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,9 @@ export function decodeArrayBuffer(str: string): ArrayBuffer {
return new Uint8Array(uintArr).buffer
}
export function encodeArrayBuffer(buffer: ArrayBuffer) {
const x = [...new Uint8Array(buffer)]
const encodedString = String.fromCharCode.apply(null, x)
let encodedString = ''
for (const byte of new Uint8Array(buffer)) {
encodedString += String.fromCharCode(byte)
}
return btoa(encodedString)
}

0 comments on commit 03182fb

Please sign in to comment.