2016-02-17 11:38:01 +01:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
import React from 'react';
|
|
|
|
|
|
|
|
import Glyphicon from 'react-bootstrap/lib/Glyphicon';
|
|
|
|
|
2016-06-13 14:35:02 +02:00
|
|
|
import S3Fetcher from '../../fetchers/s3_fetcher';
|
|
|
|
|
2016-02-17 11:38:01 +01:00
|
|
|
import AppConstants from '../../constants/application_constants';
|
|
|
|
|
2016-06-13 14:35:02 +02:00
|
|
|
import { getLangText } from '../../utils/lang';
|
2016-06-13 15:30:20 +02:00
|
|
|
import { parseQueryParamStr } from '../../utils/url';
|
2016-02-17 11:38:01 +01:00
|
|
|
|
|
|
|
|
|
|
|
const { string } = React.PropTypes;
|
|
|
|
|
|
|
|
const S3DownloadButton = React.createClass({
|
|
|
|
propTypes: {
|
|
|
|
url: string,
|
|
|
|
title: string,
|
|
|
|
artistName: string,
|
|
|
|
fileExtension: string
|
|
|
|
},
|
|
|
|
|
|
|
|
getInitialState() {
|
|
|
|
return {
|
2016-02-22 17:38:02 +01:00
|
|
|
downloadUrl: null,
|
|
|
|
signatureExpiryTimerId: null
|
2016-02-17 11:38:01 +01:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
componentDidMount() {
|
2016-03-10 12:06:58 +01:00
|
|
|
// Initially, we request a signed url from the backend
|
2016-02-17 11:38:01 +01:00
|
|
|
this.signUrl();
|
|
|
|
},
|
|
|
|
|
2016-02-22 17:38:02 +01:00
|
|
|
componentWillUnmount() {
|
2016-03-10 12:06:58 +01:00
|
|
|
window.clearTimeout(this.state.signatureExpiryTimerId);
|
2016-02-22 17:38:02 +01:00
|
|
|
},
|
|
|
|
|
2016-02-17 11:38:01 +01:00
|
|
|
transformS3UrlToS3Key(url) {
|
|
|
|
return url.replace(`https://${AppConstants.cloudfrontDomain}/`, '');
|
|
|
|
},
|
|
|
|
|
2016-03-01 12:55:27 +01:00
|
|
|
signUrl() {
|
2016-02-17 11:38:01 +01:00
|
|
|
const { url, title, artistName } = this.props;
|
|
|
|
|
|
|
|
S3Fetcher
|
|
|
|
.signUrl(this.transformS3UrlToS3Key(url), title, artistName)
|
2016-03-01 12:55:27 +01:00
|
|
|
.then(({ signed_url: downloadUrl }) => {
|
|
|
|
const { signatureExpiryTimerId } = this.state;
|
2016-03-10 12:06:58 +01:00
|
|
|
|
|
|
|
// The signed url, however can expire, which is why we need to set up a timer to
|
|
|
|
// renew it when it expires.
|
2016-06-13 15:30:20 +02:00
|
|
|
const expires = parseInt(parseQueryParamStr(downloadUrl).expires, 10);
|
2016-03-10 12:06:58 +01:00
|
|
|
const now = new Date().getTime() / 1000;
|
|
|
|
|
|
|
|
// Amazon uses seconds as their signature unix timestamp while `setTimeout` uses
|
|
|
|
// milliseconds. Therefore we need to multiply with 1000.
|
|
|
|
const interval = (expires - now) * 1000;
|
|
|
|
|
|
|
|
// Make sure to clear the previous timeout just in case it was not the one that
|
|
|
|
// invoked the refetching
|
|
|
|
if (signatureExpiryTimerId) {
|
|
|
|
window.clearTimeout(signatureExpiryTimerId);
|
2016-03-01 12:55:27 +01:00
|
|
|
}
|
2016-03-10 12:06:58 +01:00
|
|
|
|
|
|
|
this.setState({
|
|
|
|
downloadUrl,
|
|
|
|
// Substract 5s to make sure there is a big enough window to sign again before
|
|
|
|
// expiration
|
|
|
|
signatureExpiryTimerId: window.setTimeout(this.signUrl, interval - 5000)
|
|
|
|
});
|
2016-03-01 12:55:27 +01:00
|
|
|
})
|
2016-02-17 11:38:01 +01:00
|
|
|
.catch(console.logGlobal);
|
|
|
|
},
|
|
|
|
|
|
|
|
render() {
|
|
|
|
const { fileExtension, url } = this.props;
|
|
|
|
const { downloadUrl } = this.state;
|
|
|
|
|
|
|
|
return (
|
2016-02-22 12:18:40 +01:00
|
|
|
<a
|
|
|
|
ref="downloadButton"
|
|
|
|
download
|
|
|
|
className="btn btn-xs btn-default ascribe-margin-1px"
|
|
|
|
target="_blank"
|
|
|
|
href={downloadUrl || url}>
|
|
|
|
{/*
|
|
|
|
If it turns out that `fileExtension` is an empty string, we're just
|
|
|
|
using the label 'file'.
|
|
|
|
*/}
|
|
|
|
{getLangText('Download')} .{fileExtension || 'file'} <Glyphicon glyph="cloud-download" />
|
|
|
|
</a>
|
2016-02-17 11:38:01 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2016-02-22 12:18:40 +01:00
|
|
|
export default S3DownloadButton;
|