Source: settings.js

/*
Copyright (C) 2016
Developed at University of Toronto

This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

const db = require('./db.js');
const logger = require('./log.js');
const common = require('./common.js');

var allSettings = {};

/**
 * reset the global settings to their default values
 *
 * @param {function} callback
 */
exports.resetAllSettings = function (callback) {
    db.resetAllSettings (function(err, object){
        if (err){
            return callback(err);
        }
        allSettings = object;
        return callback(null,'success');
    });
}

/**
 * check if the class is active
 *
 * @param {funtion} callback
 */
exports.getClassActive = function () {
    return allSettings.general.active;
}

/**
 * set class activation status
 *
 * @param {boolean} isActive
 * @param {funtion} callback
 */
exports.setClassActive = function (isActive, callback) {
    if (common.getVariableType(isActive) !== '[object Boolean]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'general.active': isActive}}, callback);
}

/**
 * get the true of false if the leader board is set to a limit
 *
 * @param {funtion} callback
 */
exports.getLeaderboardLimited = function () {
    return allSettings.general.leaderboardLimited;
}

/**
 * set leaderboard limit
 *
 * @param {number} limit
 * @param {funtion} callback
 */
exports.setLeaderboardLimited = function (limited, callback) {
    if (common.getVariableType(limited) !== '[object Boolean]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'general.leaderboardLimited': limit}}, callback);
}

/**
 * get the limit of rows on the leaderboard
 *
 * @param {funtion} callback
 */
exports.getLeaderboardLimit = function () {
    return allSettings.general.leaderboardLimit;
}

/**
 * set leaderboard limit
 *
 * @param {number} limit
 * @param {funtion} callback
 */
exports.setLeaderboardLimit = function (limit, callback) {
    if (common.getVariableType(limit) !== '[object Number]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'general.leaderboardLimit': limit}}, callback);
}

/**
 * check if the student can edit their first and last name
 *
 * @param {funtion} callback
 */
exports.getStudentEditNameEnabled = function () {
    return allSettings.student.editNames;
}

/**
 * set student edit name enabled
 *
 * @param {boolean} isActive
 * @param {funtion} callback
 */
exports.setStudentEditNameEnabled = function (isActive, callback) {
    if (common.getVariableType(isActive) !== '[object Boolean]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'student.editNames': isActive}}, callback);
}

/**
 * check if the student can edit their email
 *
 * @param {funtion} callback
 */
exports.getStudentEditEmailEnabled = function () {
    return allSettings.student.editEmail;
}

/**
 * set student edit email enabled
 *
 * @param {boolean} isActive
 * @param {funtion} callback
 */
exports.setStudentEditEmailEnabled = function (isActive, callback) {
    if (common.getVariableType(isActive) !== '[object Boolean]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'student.editEmail': isActive}}, callback);
}

/**
 * check if the student can edit their password
 *
 * @param {funtion} callback
 */
exports.getStudentEditPasswordEnabled = function () {
    return allSettings.student.editPassword;
}

/**
 * set student edit password enabled
 *
 * @param {boolean} isActive
 * @param {funtion} callback
 */
exports.setStudentEditPasswordEnabled = function (isActive, callback) {
    if (common.getVariableType(isActive) !== '[object Boolean]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'student.editPassword': isActive}}, callback);
}

/**
 * get question default topic
 *
 * @param {funtion} callback
 */
exports.getQuestionDefaultTopic = function () {
    return allSettings.question.defaultTopic;
}

/**
 * set question default topic
 *
 * @param {string} topic
 * @param {funtion} callback
 */
exports.setQuestionDefaultTopic = function (topic, callback) {
    if (common.getVariableType(topic) !== '[object String]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'question.defaultTopic': topic}}, callback);
}

/**
 * get question default min points
 *
 * @param {funtion} callback
 */
exports.getQuestionDefaultMinPoints = function () {
    return allSettings.question.defaultMinPoints;
}

/**
 * get question default min points
 *
 * @param {number} points
 * @param {funtion} callback
 */
exports.setQuestionDefaultMinPoints = function (points, callback) {
    if (common.getVariableType(points) !== '[object Number]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'question.defaultMinPoints': points}}, callback);
}

/**
 * get question default max points
 *
 * @param {funtion} callback
 */
exports.getQuestionDefaultMaxPoints = function () {
    return allSettings.question.defaultMaxPoints;
}

/**
 * get question default max points
 *
 * @param {number} points
 * @param {funtion} callback
 */
exports.setQuestionDefaultMinPoints = function (points, callback) {
    if (common.getVariableType(points) !== '[object Number]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'question.defaultMaxPoints': points}}, callback);
}

/**
 * get question timeout enabled
 *
 * @param {funtion} callback
 */
exports.getQuestionTimeoutEnabled = function () {
    return allSettings.question.timeoutEnabled;
}

/**
 * set question timeout enabled
 *
 * @param {boolean} isActive
 * @param {funtion} callback
 */
exports.setQuestionTimeoutEnabled = function (isActive, callback) {
    if (common.getVariableType(isActive) !== '[object Boolean]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'question.timeoutEnabled': isActive}}, callback);
}

/**
 * get question timeout period
 *
 * @param {funtion} callback
 */
exports.getQuestionTimeoutPeriod = function () {
    return allSettings.question.timeoutPeriod;
}

/**
 * set question timeout period
 *
 * @param {number} timeout
 * @param {funtion} callback
 */
exports.setQuestionTimeoutPeriod = function (timeout, callback) {
    if (common.getVariableType(timeout) !== '[object Number]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'question.timeoutPeriod': timeout}}, callback);
}

/**
 * get discussionboard visibility enabled
 *
 * @param {funtion} callback
 */
exports.getDiscussionboardVisibilityEnabled = function () {
    return allSettings.discussionboard.visibility;
}

/**
 * set discussionboard visibility enabled
 *
 * @param {boolean} isActive
 * @param {funtion} callback
 */
exports.setDiscussionboardVisibilityEnabled = function (isActive, callback) {
    if (common.getVariableType(isActive) !== '[object Boolean]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'discussionboard.visibility': isActive}}, callback);
}

/**
 * get discussionboard dislikes enabled
 *
 * @param {funtion} callback
 */
exports.getDiscussionboardDislikesEnabled = function () {
    return allSettings.discussionboard.dislikesEnabled;
}

/**
 * set discussionboard dislikes enabled
 *
 * @param {boolean} isActive
 * @param {funtion} callback
 */
exports.setDiscussionboardDislikesEnabled = function (isActive, callback) {
    if (common.getVariableType(isActive) !== '[object Boolean]') {
        return callback(common.getError(1011), null);
    }

    updateSettings({$set: {'discussionboard.dislikesEnabled': isActive}}, callback);
}

/**
 * get all settings object
 *
 * @param {function} callback
 */
exports.getAllSettings = function (callback) {
    return allSettings;
}

/**
 * get all settings object from database
 *
 * @param {function} callback
 */
var getAllSettings = function (callback) {
    db.getAllSettings(function (err, allSettings) {
        return callback(err ? err : null, err ? null : allSettings);
    });
}

/**
 * update multiple settings based on the passed params
 *
 * @param {object} updateObject
 * @param {function} callback
 */
exports.updateSettings = function (updateObject, callback) {
    var updateQuery = {$set: {}};

    if ('classActive' in updateObject
        && (updateObject.classActive === 'false'
            || updateObject.classActive === 'true')) {
        updateQuery.$set['general.active'] = allSettings.general.active = (updateObject.classActive === 'true');
    }

    if ('limitedLeaderboard' in updateObject
        && (updateObject.limitedLeaderboard === 'false'
        || updateObject.limitedLeaderboard === 'true')) {
        updateQuery.$set['general.leaderboardLimited'] = allSettings.general.leaderboardLimited = (updateObject.limitedLeaderboard === 'true');
    }

    if ('studentsOnLeaderboard' in updateObject
        && parseInt(updateObject.studentsOnLeaderboard)
        && parseInt(updateObject.studentsOnLeaderboard) >= 3) {
        updateQuery.$set['general.leaderboardLimit'] = allSettings.general.leaderboardLimit = parseInt(updateObject.studentsOnLeaderboard);
    }

    if ('allowEditName' in updateObject
        && (updateObject.allowEditName === 'false'
            || updateObject.allowEditName === 'true')) {
        updateQuery.$set['student.editNames'] = allSettings.student.editNames = (updateObject.allowEditName === 'true');
    }

    if ('allowEditEmail' in updateObject
        && (updateObject.allowEditEmail === 'false'
            || updateObject.allowEditEmail === 'true')) {
        updateQuery.$set['student.editEmail'] = allSettings.student.editEmail = (updateObject.allowEditEmail === 'true');
    }

    if ('allowEditPassword' in updateObject
        && (updateObject.allowEditPassword === 'false'
            || updateObject.allowEditPassword === 'true')) {
        updateQuery.$set['student.editPassword'] = allSettings.student.editPassword = (updateObject.allowEditPassword === 'true');
    }

    if ('topic' in updateObject && common.getVariableType(updateObject.topic) === '[object String]') {
        updateQuery.$set['question.defaultTopic'] = allSettings.question.defaultTopic = updateObject.topic;
    }

    if ('minPoints' in updateObject
        && parseInt(updateObject.minPoints)
        && parseInt(updateObject.minPoints) >= 0) {
        updateQuery.$set['question.defaultMinPoints'] = allSettings.question.defaultMinPoints = parseInt(updateObject.minPoints);
    }

    if ('maxPoints' in updateObject
        && parseInt(updateObject.maxPoints)
        && parseInt(updateObject.minPoints) >= 0) {
        updateQuery.$set['question.defaultMaxPoints'] = allSettings.question.defaultMaxPoints = parseInt(updateObject.maxPoints);
    }

    if ('minPoints' in updateObject && 'maxPoints' in updateObject){
        if (parseInt(updateObject.minPoints) > parseInt(updateObject.maxPoints)){
            return callback(common.getError(3024),null)
        };
    }

    if ('allowTimeout' in updateObject
        && (updateObject.allowTimeout === 'false'
            || updateObject.allowTimeout === 'true')) {
        updateQuery.$set['question.timeoutEnabled'] = allSettings.question.timeoutEnabled = (updateObject.allowTimeout === 'true');
    }

    if ('timeoutPeriod' in updateObject
        && parseInt(updateObject.timeoutPeriod)
        && parseInt(updateObject.timeoutPeriod) >= 1) {
        updateQuery.$set['question.timeoutPeriod'] = allSettings.question.timeoutPeriod = parseInt(updateObject.timeoutPeriod)*60000;
    }

    if ('discussionView' in updateObject
        && (updateObject.discussionView === common.discussionboardVisibility.NONE
            || updateObject.discussionView === common.discussionboardVisibility.ANSWERED
            || updateObject.discussionView === common.discussionboardVisibility.ALL)) {
        updateQuery.$set['discussionboard.visibility'] = allSettings.discussionboard.visibility = updateObject.discussionView;
    }

    if ('allowDislikes' in updateObject
        && (updateObject.allowDislikes === 'false'
            || updateObject.allowDislikes === 'true')) {
        updateQuery.$set['discussionboard.dislikesEnabled'] = allSettings.discussionboard.dislikesEnabled = (updateObject.allowDislikes === 'true');
    }

    updateSettings(updateQuery, callback);
}

/**
 * update settings object
 *
 * @param {object} updateQuery
 * @param {function} callback
 */
var updateSettings = function (updateQuery, callback) {
    db.updateSettings({}, updateQuery, callback);
}

/**
 * Initialize allSettings object
 *
 */
exports.initialize = function(callback){
    getAllSettings(function(err, allSettingsObj){
        if(err){
            return callback(err, null);
        } else {
            allSettings = allSettingsObj;
            return callback(null,'success');
        }
    });
}