下载 mega.co.nz 之类的文件

今天我查看了 mega.co.nz,发现了一些令人兴奋的特性。例如在下载页面,它会在浏览器上下载文件,然后用 javascript 解密它们。

下载一个 png 文件:

Https://mega.co.nz/#!7jrgfjzj!efpjgwuphyczlexy19ex82nuwfs4sr_dg4jxddeclh4

在这个链接中,它会在浏览器中启动下载。我检查网络标签在检查元素它会下载部分文件与 AJAX 后,完成所有部分的文件,会保存在一个文件的计算机上自动所有!

我想知道他们是做什么的?你能像那样解释或链接到浏览器内部下载文件的一些资源吗?

也可以只使用 javascript 或者应该使用一些 flash 插件或类似的东西?

76474 次浏览

Mega uses several different methods to do this: (as of 27 Nov 2013)

  1. Filesystem API (Chrome/Firefox Extension polyfill)
  2. Adobe Flash SWF Filewriter (old browsers fallback)
  3. BlobBuilder (IE10/IE11)
  4. MEGA Firefox Extension (deprecated)
  5. Arraybuffer/Blob (in memory) + a[download] (for browsers that support a[download])
  6. MediaSource (experimental streaming solution)
  7. Blob stored in IndexedDB storage + a[download] (Firefox 20+, improvement over the in-memory Blob method)

(source: https://eu.static.mega.co.nz/js/download_6.js)

MEGAcmd

There is megacmd, the official command line interface. You can also build it from sources on github at https://github.com/meganz/MEGAcmd

megacmd is a wrapper around Mega SDK and if you decide to compile it on your own you'll need the same dependencies (on ubuntu) as the ones listed below for Mega SDK.

For details on usage see the MEGAcmd User Guide.

Mega SDK

Mega SDK which can be compiled by following the steps on the github page. It includes the megacli utility which is an interactive shell for synching and downloading/uploading.

## compilation steps for ubuntu
git clone --depth 1 https://github.com/meganz/sdk megasdk
cd megasdk
sudo apt install libcurl4-openssl-dev  libc-ares-dev libssl-dev libcrypto++-dev  zlib1g-dev libsqlite3-dev  libfreeimage-dev libswscale-dev
autogen.sh
./configure
make -j 8  ## pass the number of CPUs you have to speed up compilation
sudo make install

mega.py python module (deprecated)

For those who found this question searching for an actual recipe to download a link in text mode here is a simple python script that uses the mega.py module (install it with sudo pip install mega.py):

import sys
import getpass
#install the module with: 'sudo pip install mega.py'
from mega import Mega


email = '_your_megamail_@domain.com'
password = getpass.getpass(prompt='Mega password for {}:'.format(email))


mega = Mega({'verbose': True})
m = mega.login(email, password)
m.download_url(sys.argv[1])

The script works with python 2.7 and takes the URL of the mega.nz link.

getpass is used for securely entering the password in the console in order to avoid storing the password in the script — if you are comfortable hardcoding the password then set it in line #7.

megatools

On most Linux/posix boxes you can install megatools from standard repositories, i.e.

On ubuntu/debian:

apt install megatools

On MacOS:

brew install megatools

Once installed you will find a number of command line utilities, among which megadl which can download both shared files and your own files. See megadl -h for details.

A basic implementation of multipart in-browser downloader using Blob and URL APIs is brought here. It downloads a file on 4 concurrent requests and shows the progress also. Please note that it seems setting range header might generally not a good idea on XHR requests, have a look at this topic.

While downloading:

While downloading

After the download:

After the download

Another interesting topic would be implementing Pause/Resume functionality from Mega. XHR API of current browsers doesn't offer that capability so the only chance you have is to do multiple small sized chunks downloading and giving up on the downloaded part of your small chunks, the way it seems is done on Mega also. But fetch streaming feature can be used for that purpose, I didn't explore that yet well enough but it is documented here.

Btw, have a look at these awesome projects:

As of 2020, you can use the Service Workers for seamlessly integrating your custom code with the browser's built-in download manager: https://developers.google.com/web/updates/2016/06/sw-readablestreams

I also guess you'd have the following headers in order for a file to be downloaded instead of being viewed:

 headers: {
'Content-Type': 'application/octet-stream',
'Content-Disposition': 'attachment; filename="your_filename.bin"',
}

Personally I have found this approach to be working flawlessly in both Google Chrome in Firefox, and I'm already using it in production.