-
-
Notifications
You must be signed in to change notification settings - Fork 4.4k
Saving a remote file
FileSaver is built for saving client side generated content, but if the content is coming from a server then there is different ways to achieve the goal of saving the file downloaded from the cloud.
- Using Http Header
- Using a form element (Other than GET methods)
-
Using
a[download]
- Using Ajax + FileSaver
Content-Disposition
attachment header is the best preferred way to download files from the browser. It has better cross browser compatibility, won't have any memory limit and it doesn't require any JavaScript.
Content-Type: application/octet-stream
makes the browser incompatible to render the page so the fallback solution for browsers is to save the resource.
Content-Length
is optional and using it will let the user how much there is left in a progress bar.
Content-Type: 'application/octet-stream; charset=utf-8'
Content-Disposition: attachment; filename="filename.jpg"; filename*="filename.jpg"
Content-Length: <size in bytes>
filename
Is followed by a string containing the original name of the file transmitted. The filename is always optional and must not be used blindly by the application: path information should be stripped, and conversion to the server file system rules should be done. This parameter provides mostly indicative information. When used in combination with Content-Disposition: attachment, it is used as the default filename for an eventual 'Save As" dialog presented to the user.
filename*
The parameters "filename" and "filename*" differ only in that "filename*" uses the encoding defined in RFC 5987. When both "filename" and "filename*" are present in a single header field value, "filename*" is preferred over "filename" when both are present and understood.
If the file is generated using a POST request you may think you would need to use ajax to complete the request and then download the result that you have buffered up in the memory while creating your Blob.
That is a waste of time and memory. You can accomplish it with a regular <form>
submission and then respond with a content-disposition
attachment header. Ajax/JS isn't the solution to everything. It would be better to create a hidden form + fields and submit it using javascript then using ajax.
If you have no way to add the content-disposition
attachment header
a better way to save it directly to the hard drive would be to use the download attribute
<a href="uploads/screenshot.png" download="cat.png">download cat.png</a>
This attribute instructs browsers to download a URL instead of navigating to it, so the user will be prompted to save it as a local file. If the attribute has a value, it is used as the pre-filled file name in the Save prompt (the user can still change the file name if they want). There are no restrictions on allowed values, though / and \ are converted to underscores. Most file systems limit some punctuation in file names, and browsers will adjust the suggested name accordingly.
- This attribute can be used with blob: URLs and data: URLs to download content generated by JavaScript, such as pictures created in an image-editor Web app.
- If the HTTP header Content-Disposition: gives a different filename than this attribute, the HTTP header takes priority over this attribute.
- If Content-Disposition: is set to inline, Firefox prioritizes Content-Disposition, like the filename case, while Chrome prioritizes the download attribute.
If for some reason you need some headers (like authentication) for downloading the remote file. Then ajax will probably be the only way forward. Same goes if download attribute isn't supported in the browser you are targeting or content-disposition could not be added from the back-end server.
One way you could add request headers and modify the response header to include the content-disposition header is through Service Worker but I guess nearly newbody would go that route. Unless you use StreamSaver.js which is the core method of saving large files as a stream. but i won't go in to that, it requires the page to be https and have support for service workers.
The important thing in all request is to get the response as a blob. If you try to get the content they way you usually dose with text responses. This tells the browser not to parse the text content, and to let the bytes pass through unprocessed. Creating a blob from textContent is problematic if downloading binary data
Plain vanilla XMLHttpRequest
var xhr = new XMLHttpRequest()
xhr.open(method, url)
xhr.responseType = 'blob'
xhr.onload = function() {
FileSaver.saveAs(xhr.response, filename);
}
xhr.send()
Fetch API
// ES7
const res = await fetch(url)
const blob = await res.blob()
saveAs(blob, fileName)
// ES6
fetch(url)
.then(res => res.blob())
.then(blob => saveAs(blob, fileName))
// ES5
fetch(url)
.then(function(res) {
return res.blob()
})
.then(function(blob) {
saveAs(blob, fileName)
})
Angulars 1.x $http
$http({
url: "http://localhost:8080/filename.zip",
responseType: "blob"
}).then(function(response) {
saveAs(response.data, fileName)
})
Angulars http
import {ResponseContentType } from '@angular/http'
@Injectable()
export class AngularService {
constructor(private http: Http) {}
download(model: MyModel) { //get file from service
this.http.post("http://localhost/a2/pdf.php", JSON.stringify(model), {
method: RequestMethod.Post,
responseType: ResponseContentType.Blob,
headers: new Headers({'Content-Type', 'application/x-www-form-urlencoded'})
}).subscribe(
response => { // download file
var blob = new Blob([response.blob()], {type: 'application/pdf'});
var filename = 'file.pdf';
saveAs(blob, filename);
},
error => {
console.error(`Error: ${error.message}`);
}
);
}
}
jQuery's $.ajax()
Don't! They don't support changing the response type.