Skip to content

Commit 7754e7d

Browse files
[Fix] Hotfix (#333)
* allow parallel downloads for chunking * fix zoom range wrapper css * ui: add 'preparing' status * feat: base n of connections based on CDN cache --------- Co-authored-by: Flavio F Lima <flavioislima@gmail.com>
1 parent 58be10a commit 7754e7d

File tree

9 files changed

+79
-19
lines changed

9 files changed

+79
-19
lines changed

src/backend/downloadmanager/utils.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -54,10 +54,12 @@ async function installQueueElement(params: InstallParams): Promise<{
5454
}
5555
})
5656

57+
const startingStatus = runner === 'hyperplay' ? 'preparing' : 'installing'
58+
5759
sendFrontendMessage('gameStatusUpdate', {
5860
appName,
5961
runner,
60-
status: 'installing',
62+
status: startingStatus,
6163
folder: path
6264
})
6365

src/backend/storeManagers/hyperplay/games.ts

+11
Original file line numberDiff line numberDiff line change
@@ -197,6 +197,7 @@ async function downloadGame(
197197
destinationPath: string
198198
): Promise<void> {
199199
const appInfo = getGameInfo(appName)
200+
let downloadStarted = false
200201

201202
if (!appInfo || !appInfo.releaseMeta) {
202203
throw new Error('App not found in library')
@@ -227,6 +228,16 @@ async function downloadGame(
227228
platformInfo.downloadSize
228229
)
229230

231+
if (downloadedBytes > 0 && !downloadStarted) {
232+
downloadStarted = true
233+
sendFrontendMessage('gameStatusUpdate', {
234+
appName,
235+
status: 'installing',
236+
runner: 'hyperplay',
237+
folder: destinationPath
238+
})
239+
}
240+
230241
window.webContents.send(`progressUpdate-${appName}`, {
231242
appName,
232243
status: 'installing',

src/backend/utils.ts

+40-14
Original file line numberDiff line numberDiff line change
@@ -1220,6 +1220,18 @@ export interface ProgressCallback {
12201220
): void
12211221
}
12221222

1223+
/**
1224+
* Downloads a file from a given URL to a specified destination path.
1225+
* If there is cache on the CDN it will use 5 connections so the download will be faster.
1226+
* If there is no cache on the CDN it will use 1 connection, otherwise the download might fail to start.
1227+
*
1228+
* @param {string} url - The URL of the file to download.
1229+
* @param {string} dest - The destination path to save the downloaded file.
1230+
* @param {AbortController} abortController - The AbortController instance to cancel the download.
1231+
* @param {ProgressCallback} [progressCallback] - An optional callback function to track the download progress.
1232+
* @returns {Promise<void>} - A Promise that resolves when the download is complete.
1233+
* @throws {Error} - If the download fails or is incomplete.
1234+
*/
12231235
export async function downloadFile(
12241236
url: string,
12251237
dest: string,
@@ -1230,22 +1242,41 @@ export async function downloadFile(
12301242
let lastBytesWritten = 0
12311243
let fileSize = 0
12321244

1245+
let connections = 1
1246+
try {
1247+
const response = await axios.head(url)
1248+
const cdnCache = response.headers['cdn-cache']
1249+
const isCached = cdnCache === 'HIT' || cdnCache === 'STALE'
1250+
if (isCached) {
1251+
connections = 5
1252+
}
1253+
fileSize = parseInt(response.headers['content-length'], 10)
1254+
} catch (err) {
1255+
logError(
1256+
`Downloader: Failed to get headers for ${url}`,
1257+
LogPrefix.DownloadManager
1258+
)
1259+
throw new Error('Failed to get headers')
1260+
}
1261+
12331262
try {
12341263
const dl = new EasyDl(url, dest, {
12351264
existBehavior: 'overwrite',
1236-
maxRetry: 10,
1237-
retryDelay: 1000,
1238-
connections: 1,
1239-
chunkSize: 1024 * 1024 * 10
1265+
connections
12401266
}).start()
12411267

1242-
dl.on('metadata', (metadata) => {
1243-
fileSize = metadata.size
1244-
})
1245-
12461268
abortController.signal.addEventListener('abort', () => {
12471269
dl.destroy()
12481270
})
1271+
1272+
dl.on('error', (error) => {
1273+
logError(error, LogPrefix.HyperPlay)
1274+
})
1275+
1276+
dl.on('retry', (retry) => {
1277+
logInfo(`Retrying download: ${retry}`, LogPrefix.HyperPlay)
1278+
})
1279+
12491280
const throttledProgressCallback = throttle(
12501281
(
12511282
bytes: number,
@@ -1282,16 +1313,11 @@ export async function downloadFile(
12821313
}
12831314
})
12841315

1285-
dl.on('error', function (error) {
1286-
logError(`Downloader: Error: ${error}`, LogPrefix.DownloadManager)
1287-
throw error
1288-
})
1289-
12901316
const downloaded = await dl.wait()
12911317

12921318
if (!downloaded) {
12931319
logWarning(
1294-
`: Downloader: File ${url} not downloaded`,
1320+
`Downloader: Download stopped or paused`,
12951321
LogPrefix.DownloadManager
12961322
)
12971323
throw new Error('Download incomplete')

src/common/types.ts

+1
Original file line numberDiff line numberDiff line change
@@ -191,6 +191,7 @@ export type Status =
191191
| 'installed'
192192
| 'extracting'
193193
| 'paused'
194+
| 'preparing'
194195

195196
export interface GameStatus {
196197
appName: string

src/frontend/hooks/constants.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,8 @@ export function getStatusLabel({
3434
runner === 'sideload' ? '' : size
3535
}`,
3636
notInstalled: t('gamepage:status.notinstalled'),
37-
paused: t('gamepage:status.paused', 'Paused')
37+
paused: t('gamepage:status.paused', 'Paused'),
38+
preparing: t('gamepage:status.preparing', 'Preparing')
3839
}
3940

4041
return statusMap[status] || t('gamepage:status.notinstalled')

src/frontend/screens/Accessibility/index.css

+6
Original file line numberDiff line numberDiff line change
@@ -50,3 +50,9 @@ input[type='range'] {
5050
.Accessibility h1.headerTitle {
5151
margin-left: 0px;
5252
}
53+
54+
.rangeWrapper {
55+
display: flex;
56+
flex-direction: column;
57+
align-items: flex-start;
58+
}

src/frontend/screens/Game/GamePage/index.tsx

+6-1
Original file line numberDiff line numberDiff line change
@@ -118,6 +118,7 @@ export default React.memo(function GamePage(): JSX.Element | null {
118118
const isSyncing = status === 'syncing-saves'
119119
const isPaused = status === 'paused'
120120
const isExtracting = status === 'extracting'
121+
const isPreparing = status === 'preparing'
121122
const notAvailable = !gameAvailable && gameInfo.is_installed
122123
const notSupportedGame =
123124
gameInfo.runner !== 'sideload' && gameInfo.thirdPartyManagedApp === 'Origin'
@@ -650,6 +651,10 @@ export default React.memo(function GamePage(): JSX.Element | null {
650651
return t('status.paused', 'Paused')
651652
}
652653

654+
if (isPreparing) {
655+
return t('status.preparing', 'Preparing Download, please wait')
656+
}
657+
653658
if (notSupportedGame) {
654659
return t(
655660
'status.this-game-uses-third-party',
@@ -764,7 +769,7 @@ export default React.memo(function GamePage(): JSX.Element | null {
764769
if (isExtracting) {
765770
return t('status.extracting', 'Extracting files')
766771
}
767-
if (isInstalling) {
772+
if (isInstalling || isPreparing) {
768773
return t('button.queue.cancel', 'Cancel Download')
769774
}
770775
return t('button.install')

src/frontend/screens/Library/components/GameCard/constants.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ export function getCardStatus(
3131
const notSupportedGame = status === 'notSupportedGame'
3232
const syncingSaves = status === 'syncing-saves'
3333
const isPaused = status === 'paused'
34+
const isPreparing = status === 'preparing'
3435

3536
const haveStatus =
3637
isMoving ||
@@ -44,7 +45,9 @@ export function getCardStatus(
4445
isPlaying ||
4546
syncingSaves ||
4647
(isInstalled && layout !== 'grid') ||
47-
isPaused
48+
isPaused ||
49+
isPreparing
50+
4851
return {
4952
isInstalling,
5053
notSupportedGame,
@@ -54,6 +57,7 @@ export function getCardStatus(
5457
notAvailable,
5558
isUpdating,
5659
isPaused,
60+
isPreparing,
5761
haveStatus
5862
}
5963
}

src/frontend/state/GlobalState.tsx

+5-1
Original file line numberDiff line numberDiff line change
@@ -581,7 +581,11 @@ class GlobalState extends PureComponent<Props> {
581581
)
582582

583583
// in these cases we just add the new status
584-
if (['installing', 'updating', 'playing', 'extracting'].includes(status)) {
584+
if (
585+
['installing', 'updating', 'playing', 'extracting', 'preparing'].includes(
586+
status
587+
)
588+
) {
585589
currentApp.status = status
586590
newLibraryStatus.push(currentApp)
587591
this.setState({ libraryStatus: newLibraryStatus })

0 commit comments

Comments
 (0)