Merge pull request #4068 from NoelDeMartin/MOBILE-4470
MOBILE-4470 storagemanager: Fix removing coursesmain
commit
7e989c93d5
|
@ -14,6 +14,7 @@
|
|||
|
||||
import { DownloadStatus } from '@/core/constants';
|
||||
import { Component, OnDestroy, OnInit } from '@angular/core';
|
||||
import { CoreQueueRunner } from '@classes/queue-runner';
|
||||
import { CoreCourse, CoreCourseProvider } from '@features/course/services/course';
|
||||
import { CoreCourseHelper } from '@features/course/services/course-helper';
|
||||
import { CoreCourseModulePrefetchDelegate } from '@features/course/services/module-prefetch-delegate';
|
||||
|
@ -50,6 +51,8 @@ export class AddonStorageManagerCoursesStoragePage implements OnInit, OnDestroy
|
|||
courseStatusObserver?: CoreEventObserver;
|
||||
siteId: string;
|
||||
|
||||
private downloadedCoursesQueue = new CoreQueueRunner();
|
||||
|
||||
constructor() {
|
||||
this.siteId = CoreSites.getCurrentSiteId();
|
||||
}
|
||||
|
@ -87,7 +90,7 @@ export class AddonStorageManagerCoursesStoragePage implements OnInit, OnDestroy
|
|||
}
|
||||
}
|
||||
|
||||
await this.setDownloadedCourses(downloadedCourses);
|
||||
await this.downloadedCoursesQueue.run(() => this.setDownloadedCourses(downloadedCourses));
|
||||
|
||||
this.loaded = true;
|
||||
}
|
||||
|
@ -124,7 +127,9 @@ export class AddonStorageManagerCoursesStoragePage implements OnInit, OnDestroy
|
|||
try {
|
||||
await Promise.all(deletedCourseIds.map((courseId) => CoreCourseHelper.deleteCourseFiles(courseId)));
|
||||
|
||||
await this.setDownloadedCourses(this.downloadedCourses.filter((course) => !deletedCourseIds.includes(course.id)));
|
||||
await this.downloadedCoursesQueue.run(async () => {
|
||||
await this.setDownloadedCourses(this.downloadedCourses.filter((course) => !deletedCourseIds.includes(course.id)));
|
||||
});
|
||||
} catch (error) {
|
||||
CoreDomUtils.showErrorModalDefault(error, Translate.instant('core.errordeletefile'));
|
||||
} finally {
|
||||
|
@ -160,7 +165,9 @@ export class AddonStorageManagerCoursesStoragePage implements OnInit, OnDestroy
|
|||
try {
|
||||
await CoreCourseHelper.deleteCourseFiles(course.id);
|
||||
|
||||
await this.setDownloadedCourses(CoreArray.withoutItem(this.downloadedCourses, course));
|
||||
await this.downloadedCoursesQueue.run(async () => {
|
||||
await this.setDownloadedCourses(CoreArray.withoutItem(this.downloadedCourses, course));
|
||||
});
|
||||
} catch (error) {
|
||||
CoreDomUtils.showErrorModalDefault(error, Translate.instant('core.errordeletefile'));
|
||||
} finally {
|
||||
|
@ -175,7 +182,7 @@ export class AddonStorageManagerCoursesStoragePage implements OnInit, OnDestroy
|
|||
*/
|
||||
private async onCourseUpdated(courseId: number, status: DownloadStatus): Promise<void> {
|
||||
if (courseId == CoreCourseProvider.ALL_COURSES_CLEARED) {
|
||||
await this.setDownloadedCourses([]);
|
||||
await this.downloadedCoursesQueue.run(() => this.setDownloadedCourses([]));
|
||||
|
||||
return;
|
||||
}
|
||||
|
@ -189,7 +196,7 @@ export class AddonStorageManagerCoursesStoragePage implements OnInit, OnDestroy
|
|||
course.isDownloading = status === DownloadStatus.DOWNLOADING;
|
||||
course.totalSize = await this.calculateDownloadedCourseSize(course.id);
|
||||
|
||||
await this.setDownloadedCourses(this.downloadedCourses);
|
||||
await this.downloadedCoursesQueue.run(() => this.setDownloadedCourses(this.downloadedCourses));
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -121,8 +121,17 @@ export class CoreQueueRunner {
|
|||
* @param options Options.
|
||||
* @returns Promise resolved when the function has been executed.
|
||||
*/
|
||||
run<T>(id: string, fn: CoreQueueRunnerFunction<T>, options?: CoreQueueRunnerAddOptions): Promise<T> {
|
||||
options = options || {};
|
||||
run<T>(fn: CoreQueueRunnerFunction<T>, options?: CoreQueueRunnerAddOptions): Promise<T>;
|
||||
run<T>(id: string, fn: CoreQueueRunnerFunction<T>, options?: CoreQueueRunnerAddOptions): Promise<T>;
|
||||
run<T>(
|
||||
idOrFn: string | CoreQueueRunnerFunction<T>,
|
||||
fnOrOptions?: CoreQueueRunnerFunction<T> | CoreQueueRunnerAddOptions,
|
||||
options: CoreQueueRunnerAddOptions = {},
|
||||
): Promise<T> {
|
||||
let id = typeof idOrFn === 'string' ? idOrFn : this.getUniqueId('anonymous');
|
||||
const fn = typeof idOrFn === 'function' ? idOrFn : fnOrOptions as CoreQueueRunnerFunction<T>;
|
||||
|
||||
options = typeof fnOrOptions === 'object' ? fnOrOptions : options;
|
||||
|
||||
if (id in this.queue) {
|
||||
if (!options.allowRepeated) {
|
||||
|
|
|
@ -0,0 +1,38 @@
|
|||
// (C) Copyright 2015 Moodle Pty Ltd.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
import { CoreQueueRunner } from '@classes/queue-runner';
|
||||
import { CoreUtils } from '@services/utils/utils';
|
||||
|
||||
describe('CoreQueueRunner', () => {
|
||||
|
||||
it('Locks threads launched synchronously', async () => {
|
||||
// Arrange
|
||||
const concurrency = 100;
|
||||
const range = Array.from({ length: concurrency }, (_, item) => item);
|
||||
const lock = new CoreQueueRunner();
|
||||
const items: string[] = [];
|
||||
|
||||
// Act
|
||||
await Promise.all(range.map((i) => lock.run(async () => {
|
||||
await CoreUtils.wait(Math.floor(Math.random() * 10));
|
||||
|
||||
items.push(`Item #${i}`);
|
||||
})));
|
||||
|
||||
// Assert
|
||||
expect(items).toEqual(range.map(i => `Item #${i}`));
|
||||
});
|
||||
|
||||
});
|
Loading…
Reference in New Issue