Page MenuHomePhabricator

[native] Fix copying encrypted images on iOS
ClosedPublic

Authored by ashoat on Feb 27 2025, 12:42 AM.
Tags
None
Referenced Files
F5120047: D14414.diff
Thu, Apr 3, 6:06 AM
Unknown Object (File)
Tue, Apr 1, 7:32 AM
Unknown Object (File)
Thu, Mar 13, 3:58 AM
Unknown Object (File)
Tue, Mar 11, 4:13 PM
Unknown Object (File)
Sun, Mar 9, 10:26 AM
Unknown Object (File)
Sat, Mar 8, 6:23 PM
Unknown Object (File)
Sat, Mar 8, 7:36 AM
Unknown Object (File)
Sat, Mar 8, 3:51 AM
Subscribers

Details

Summary
Test Plan

Confirmed that copy of an encrypted blob-hosted image works on iOS using a Release build on a physical device

Diff Detail

Repository
rCOMM Comm
Lint
Lint Not Applicable
Unit
Tests Not Applicable

Event Timeline

native/components/full-screen-view-modal.react.js
732 ↗(On Diff #47284)

Decided the Platform check made more sense to happen in ImageModal

native/media/save-media.js
624 ↗(On Diff #47284)

Unfortunately, deleting the temp file after the copy, even on a significant delay, prevents the paste from working.

It seems like the file needs to be available the first time a paste is attempted.

It's stored in a /tmp folder within the app sandbox, so I'm hoping it's eventually deleted. @bartek, curious if you have any context on that.

An alternative approach, that doesn't require using temporary file:

  • Extract some logic from saveRemoteMediaToDisk() in`save-file.js`, with the following changes:
    • For encrypted media, call fetchAndDecryptMedia({ destination: 'data_uri' }) instead of destination: file
    • For non-encrypted, fetch base64 but don't save to file (skip last step)
  • call Clipboard.setImageFromBase64 - iOS implementation is in the patch, line 116
native/media/save-media.js
624 ↗(On Diff #47284)

This is the case on Android, but I'm surprised that it happens on iOS too, given in the patch below you explicitly read the file into NSData and create an image from it. After this operation, the file shouldn't be needed anymore.

An alternative approach, that doesn't require using temporary file:

  • Extract some logic from saveRemoteMediaToDisk() in`save-file.js`, with the following changes:
    • For encrypted media, call fetchAndDecryptMedia({ destination: 'data_uri' }) instead of destination: file
    • For non-encrypted, fetch base64 but don't save to file (skip last step)
  • call Clipboard.setImageFromBase64 - iOS implementation is in the patch, line 116

I'm worried about using Clipboard.setImageFromBase64 for large media files... I suspect it will be passing a massive string over the legacy React Native bridge, and so I feel obliged to do some performance testing on that, and I don't have the time to do that for an approach that I suspect might end up not working for us.

native/media/save-media.js
624 ↗(On Diff #47284)

I tested this a lot last night, including add an up to 30s delay on the file deletion. As long as the first paste occurred before the file deletion, all pastes would work. But if the file was deleted before the paste occurred, the paste would fail (even though the copy succeeded).

@bartek, can you respond to the question I asked here, specifically about whether you have any context on whether the file will be eventually deleted?

It's stored in a /tmp folder within the app sandbox, so I'm hoping it's eventually deleted. @bartek, curious if you have any context on that.

I'm worried about using Clipboard.setImageFromBase64 for large media files... I suspect it will be passing a massive string over the legacy React Native bridge, and so I feel obliged to do some performance testing on that, and I don't have the time to do that for an approach that I suspect might end up not working for us.

Makes sense, let's stay with file-based approach then.

native/media/save-media.js
624 ↗(On Diff #47284)

I'm sorry, I misunderstood your question, I thought you're asking me about context on clipboard behavior.

Regarding /tmp - yes, iOS periodically cleans up app sandbox temporary directories. It's unspecified when exactly, but it happens when the app is not running.

This revision is now accepted and ready to land.Feb 27 2025, 12:17 PM