libs/clipboard/README.md
Copy files and text through network. Main lowlevel logic from FreeRDP.
To enjoy file copy and paste feature on Linux/OSX,
please build with unix-file-copy-paste feature.
TODO: Move this lib to a separate project.
Terminologies:
localThe main algorithm of copying and pasting files is from Remote Desktop Protocol: Clipboard Virtual Channel Extension, and could be concluded as:
local and remote notify each other that it's ready.
local subscribes/listening to the system's clipboard for file copy
local once got file copy event, notice the remote
remote confirms receive and try pulls the file list
local updates its file-list, the remote flushes pulled file list to the clipboard
remote OS or desktop manager initiates a paste, making other programs reading clipboard files. Convert those reading requests to RPCs
finishing pasting all files one by one.
In a perspective of network data transferring:
sequenceDiagram
participant l as Local
participant r as Remote
note over l, r: Initialize
l ->> r: Monitor Ready
r ->> l: Monitor Ready
loop Get clipboard update
l ->> r: Format List (I got update)
r ->> l: Format List Response (notified)
r ->> l: Format Data Request (requests file list)
activate l
note left of l: Retrieve file list from system clipboard
l ->> r: Format Data Response (containing file list)
deactivate l
note over r: Update system clipboard with received file list
end
loop Some application requests copied files
note right of r: application reads file from x to x+y
note over r: the file is the a-th file on list
r ->> l: File Contents Request (read file a offset x size y)
activate l
note left of l: Find a-th file on list, read from x to x+y
l ->> r: File Contents Response (contents of file a offset x size y)
deactivate l
end
Note: In actual implementation, both sides could play send clipboard update and request file contents. There is no such limitation that only local can update clipboard and copy files to remote.
The protocol was originally designed as an extension of the Windows RDP, so the specific message packages fits windows well.
When starting cliprdr, a thread is spawned to create an invisible window
and to subscribe to OLE clipboard events.
The window's callback (see cliprdr_proc in src/windows/wf_cliprdr.c) was
set to handle a variety of events.
Detailed implementation is shown in pictures above.
The Cliprdr Server implementation has mainly 3 parts:
The clipboard client has a thread polling for file urls on clipboard.
If the client found any updates of file urls, after filtering out those pointing to our FUSE directory or duplicated, send format list directly to remote.
The cliprdr server also uses clipboard client for setting clipboard, or retrieve paths from system.
The local file list is a temporary list of file metadata. When receiving file contents PDU from peer, the server picks out the file requested and open it for reading if necessary.
Also when receiving Format Data Request PDU from remote asking for file list, the local file list should be rebuilt from file list retrieved from Clipboard Client.
Some caching and preloading could be done on it since applications are likely to read on the list sequentially.
The FUSE server could convert POSIX file reading request to File Contents Request/Response RPCs.
When received file list from remote, the FUSE server will figure out the file system tree and rearrange its content.
The protocol was originally implemented for windows, so paths in PDU will all be converted to DOS formats in UTF-16 LE encoding, and datetimes will be converted to LDAP timestamp instead of unix timestamp
UNIX
/usr/bin/rustdesk
->
DOS
\usr\bin\rustdesk
To better fit for preserving permissions on unix-like platforms, a reserved area of FileDescriptor PDU
you may notice
the mountpoint is still occupied after the application quits.
That's because the FUSE server was not mounted with AUTO_UNMOUNT.
AUTO_UNMOUNT was not enabled by default and requires enable
user_allow_other in configure. Letting users edit such global
configuration to use this feature might not be a good idea.umount()
syscall to unmount will also require that option.umount
program to unmount dangling FUSE server. It worked perfectly for now.