// (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 { Injectable } from '@angular/core';

import { CoreUtils } from '@services/utils/utils';
import { CoreLogger } from '@singletons/logger';
import { makeSingleton } from '@singletons/core.singletons';

/**
 * Interface that all init handlers must implement.
 */
export type CoreInitHandler = {
    /**
     * A name to identify the handler.
     */
    name: string;

    /**
     * The highest priority is executed first. You should use values lower than MAX_RECOMMENDED_PRIORITY.
     */
    priority?: number;

    /**
     * Set this to true when this process should be resolved before any following one.
     */
    blocking?: boolean;

    /**
     * Function to execute during the init process.
     *
     * @return Promise resolved when done.
     */
    load(): Promise<void>;
};

/*
 * Provider for initialisation mechanisms.
 */
@Injectable()
export class CoreInitDelegate {
    static readonly DEFAULT_PRIORITY = 100; // Default priority for init processes.
    static readonly MAX_RECOMMENDED_PRIORITY = 600;

    protected initProcesses = {};
    protected logger: CoreLogger;
    protected readiness;

    constructor() {
        this.logger = CoreLogger.getInstance('CoreInitDelegate');
    }

    /**
     * Executes the registered init processes.
     *
     * Reserved for core use, do not call directly.
     */
    executeInitProcesses(): void {
        let ordered = [];

        if (typeof this.readiness == 'undefined') {
            this.initReadiness();
        }

        // Re-ordering by priority.
        for (const name in this.initProcesses) {
            ordered.push(this.initProcesses[name]);
        }
        ordered.sort((a, b) => b.priority - a.priority);

        ordered = ordered.map((data: CoreInitHandler) => ({
            func: this.prepareProcess.bind(this, data),
            blocking: !!data.blocking,
        }));

        // Execute all the processes in order to solve dependencies.
        CoreUtils.instance.executeOrderedPromises(ordered).finally(this.readiness.resolve);
    }

    /**
     * Init the readiness promise.
     */
    protected initReadiness(): void {
        this.readiness = CoreUtils.instance.promiseDefer();
        this.readiness.promise.then(() => this.readiness.resolved = true);
    }

    /**
     * Instantly returns if the app is ready.
     *
     * @return Whether it's ready.
     */
    isReady(): boolean {
        return this.readiness.resolved;
    }

    /**
     * Convenience function to return a function that executes the process.
     *
     * @param data The data of the process.
     * @return Promise of the process.
     */
    protected async prepareProcess(data: CoreInitHandler): Promise<void> {
        this.logger.debug(`Executing init process '${data.name}'`);

        try {
            await data.load();
        } catch (e) {
            this.logger.error('Error while calling the init process \'' + data.name + '\'. ' + e);
        }
    }

    /**
     * Notifies when the app is ready. This returns a promise that is resolved when the app is initialised.
     *
     * @return Resolved when the app is initialised. Never rejected.
     */
    async ready(): Promise<void> {
        if (typeof this.readiness == 'undefined') {
            // Prevent race conditions if this is called before executeInitProcesses.
            this.initReadiness();
        }

        await this.readiness.promise;
    }

    /**
     * Registers an initialisation process.
     *
     * @description
     * Init processes can be used to add initialisation logic to the app. Anything that should block the user interface while
     * some processes are done should be an init process. It is recommended to use a priority lower than MAX_RECOMMENDED_PRIORITY
     * to make sure that your process does not happen before some essential other core processes.
     *
     * An init process should never change state or prompt user interaction.
     *
     * This delegate cannot be used by site plugins.
     *
     * @param instance The instance of the handler.
     */
    registerProcess(handler: CoreInitHandler): void {
        if (typeof handler.priority == 'undefined') {
            handler.priority = CoreInitDelegate.DEFAULT_PRIORITY;
        }

        if (typeof this.initProcesses[handler.name] != 'undefined') {
            this.logger.log(`Process '${handler.name}' already registered.`);

            return;
        }

        this.logger.log(`Registering process '${handler.name}'.`);
        this.initProcesses[handler.name] = handler;
    }
}

export class CoreInit extends makeSingleton(CoreInitDelegate) {}