4 * Copyright 2016 RIFT.IO Inc
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
10 * http://www.apache.org/licenses/LICENSE-2.0
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
21 import _
from 'lodash'
22 import alt
from '../alt'
23 import guid
from '../libraries/guid'
24 import numeral
from 'numeral'
25 import moment
from 'moment'
26 import utils
from '../libraries/utils'
27 import CatalogPackageManagerSource
from '../sources/CatalogPackageManagerSource'
28 import CatalogPackageManagerActions
from '../actions/CatalogPackageManagerActions'
29 import CatalogDataSource
from '../sources/CatalogDataSource'
31 import imgDownload
from '../../../node_modules/open-iconic/svg/cloud-download.svg'
32 import imgOnboard
from '../../../node_modules/open-iconic/svg/cloud-upload.svg'
33 import imgUpdate
from '../../../node_modules/open-iconic/svg/data-transfer-upload.svg'
43 message
: 'Requesting catalog package export...',
50 checkStatusDelayInSeconds
: 2,
51 downloadUrlTimeToLiveInMinutes
: 5
54 const exception
= function ignoreException() {};
56 const packagePropertyNames
= Object
.keys(defaults
.downloadPackage
);
58 function getCatalogPackageManagerServerOrigin() {
59 return utils
.getSearchParams(window
.location
).upload_server
+ ':4567';
62 function delayStatusCheck(statusCheckFunction
, catalogPackage
) {
63 if (!catalogPackage
.checkStatusTimeoutId
) {
64 const delayCallback = function () {
65 delete catalogPackage
.checkStatusTimeoutId
;
66 statusCheckFunction(catalogPackage
).catch(exception
);
68 catalogPackage
.checkStatusTimeoutId
= _
.delay(delayCallback
, defaults
.checkStatusDelayInSeconds
* 1000);
72 class CatalogPackageManagerStore
{
78 this.registerAsync(CatalogDataSource
);
79 this.registerAsync(CatalogPackageManagerSource
);
80 this.bindAction(CatalogPackageManagerActions
.REMOVE_CATALOG_PACKAGE
, this.removeCatalogPackage
);
81 this.bindAction(CatalogPackageManagerActions
.DOWNLOAD_CATALOG_PACKAGE
, this.downloadCatalogPackage
);
82 this.bindAction(CatalogPackageManagerActions
.DOWNLOAD_CATALOG_PACKAGE_STATUS_UPDATED
, this.onDownloadCatalogPackageStatusUpdated
);
83 this.bindAction(CatalogPackageManagerActions
.DOWNLOAD_CATALOG_PACKAGE_ERROR
, this.onDownloadCatalogPackageError
);
84 this.bindAction(CatalogPackageManagerActions
.UPLOAD_CATALOG_PACKAGE
, this.uploadCatalogPackage
);
85 this.bindAction(CatalogPackageManagerActions
.UPLOAD_CATALOG_PACKAGE_STATUS_UPDATED
, this.onUploadCatalogPackageStatusUpdated
);
86 this.bindAction(CatalogPackageManagerActions
.UPLOAD_CATALOG_PACKAGE_ERROR
, this.onUploadCatalogPackageError
);
90 addPackage(catalogPackage
) {
91 const packages
= [catalogPackage
].concat(this.packages
);
92 this.setState({packages
: packages
});
95 updatePackage(catalogPackage
) {
96 const packages
= this.packages
.map(d
=> {
97 if (d
.id
=== catalogPackage
.id
) {
98 return Object
.assign({}, d
, catalogPackage
);
102 this.setState({packages
: packages
});
105 removeCatalogPackage(catalogPackage
) {
106 const packages
= this.packages
.filter(d
=> d
.id
!== catalogPackage
.id
);
107 this.setState({packages
: packages
});
110 uploadCatalogPackage(file
) {
111 file
.id
= file
.id
|| guid();
112 const catalogPackage
= _
.pick(file
, packagePropertyNames
);
113 catalogPackage
.icon
= file
.riftAction
=== 'onboard' ? imgOnboard
: imgUpdate
;
114 catalogPackage
.type
= 'upload';
115 this.addPackage(catalogPackage
);
116 // note DropZone.js handles the async upload so we don't have to invoke any async action creators
119 onUploadCatalogPackageStatusUpdated(response
) {
120 const upload
= updateStatusInfo(response
);
121 this.updatePackage(upload
);
122 // if pending with no transaction id - do nothing
123 // bc DropZone.js will notify upload progress
124 if (upload
.pending
&& upload
.transactionId
) {
125 delayStatusCheck(this.getInstance().requestCatalogPackageUploadStatus
, upload
);
126 } else if (upload
.success
) {
127 this.getInstance().loadCatalogs();
131 onUploadCatalogPackageError(response
) {
132 console
.warn('onUploadCatalogPackageError', response
);
133 const catalogPackage
= updateStatusInfo(response
);
134 this.updatePackage(catalogPackage
);
137 downloadCatalogPackage(data
) {
138 let catalogItems
= data
['selectedItems'] || [];
139 let format
= data
['selectedFormat'] || 'mano';
140 let grammar
= data
['selectedGrammar'] || 'osm';
141 if (catalogItems
.length
) {
142 const catalogPackage
= Object
.assign({}, defaults
.downloadPackage
, {id
: guid()});
143 catalogPackage
.name
= catalogItems
[0].name
;
144 catalogPackage
.type
= 'download';
145 if (catalogItems
.length
> 1) {
146 catalogPackage
.name
+= ' (' + catalogItems
.length
+ ' items)';
148 catalogPackage
.ids
= catalogItems
.map(d
=> d
.id
).sort().toString();
149 catalogPackage
.catalogItems
= catalogItems
;
150 this.addPackage(catalogPackage
);
151 this.getInstance().requestCatalogPackageDownload(catalogPackage
, format
, grammar
).catch(exception
);
155 onDownloadCatalogPackageStatusUpdated(response
) {
156 const download
= updateStatusInfo(response
);
157 this.updatePackage(download
);
158 if (download
.pending
) {
159 delayStatusCheck(this.getInstance().requestCatalogPackageDownloadStatus
, download
);
163 onDownloadCatalogPackageError(response
) {
164 console
.warn('onDownloadCatalogPackageError', response
);
165 const catalogPackage
= updateStatusInfo(response
);
166 this.updatePackage(catalogPackage
);
171 function calculateUploadProgressMessage(size
= 0, progress
= 0, bytesSent
= 0) {
172 const amount
= parseFloat(progress
) || 0;
173 const loaded
= amount
=== 100 ? size
: size
* amount
/ 100;
175 if (amount
=== 100) {
176 progressText
= numeral(loaded
).format('0.0b') + ' loaded ';
177 } else if (typeof amount
=== 'number' && amount
!= 0) {
178 progressText
= numeral(bytesSent
).format('0.0b') + ' out of ' + numeral(size
).format('0.0b');
180 progressText
= progress
;
185 function updateStatusInfo(response
) {
186 // returns the catalogPackage object with the status fields updated based on the server response
192 const responseData
= response
.data
;
193 const catalogPackage
= response
.state
;
194 switch(responseData
.status
) {
195 case 'upload-progress':
196 statusInfo
.pending
= true;
197 statusInfo
.progress
= parseFloat(responseData
.progress
) || 0;
198 statusInfo
.message
= calculateUploadProgressMessage(catalogPackage
.size
, responseData
.progress
, responseData
.bytesSent
);
200 case 'upload-success':
201 statusInfo
.pending
= true;
202 statusInfo
.progress
= 100;
203 statusInfo
.message
= 'Upload completed.';
204 statusInfo
.transactionId
= responseData
.transaction_id
;
207 statusInfo
.error
= true;
208 statusInfo
.message
= responseData
.message
;
210 case 'download-requested':
211 statusInfo
.pending
= true;
212 statusInfo
.progress
= 25;
213 statusInfo
.transactionId
= responseData
.transaction_id
;
216 statusInfo
.pending
= true;
217 statusInfo
.progress
= 50;
218 statusInfo
.message
= responseData
.events
[responseData
.events
.length
- 1].text
;
221 statusInfo
.success
= true;
222 statusInfo
.progress
= 100;
223 statusInfo
.message
= responseData
.events
[responseData
.events
.length
- 1].text
;
224 if (catalogPackage
.type
=== 'download') {
225 statusInfo
.urlValidUntil
= moment().add(defaults
.downloadUrlTimeToLiveInMinutes
, 'minutes').toISOString();
226 if (responseData
.filename
) {
227 statusInfo
.url
= getCatalogPackageManagerServerOrigin() + '/api/export/' + responseData
.filename
;
229 statusInfo
.url
= getCatalogPackageManagerServerOrigin() + '/api/export/' + catalogPackage
.transactionId
+ '.tar.gz';
234 statusInfo
.error
= true;
235 statusInfo
.message
= responseData
.errors
[0].value
;
238 throw new ReferenceError('a status of "request", "success", "failure", "pending", "upload-completed", "upload-error", "download-requested", "upload-progress", "upload-action" is required');
240 return Object
.assign({}, catalogPackage
, statusInfo
);
243 export default alt
.createStore(CatalogPackageManagerStore
, 'CatalogPackageManagerStore');