/**
 * @author [Tristan Valcke]{@link https://github.com/Itee}
 * @license [BSD-3-Clause]{@link https://opensource.org/licenses/BSD-3-Clause}
 *
 * @class TDatabaseController
 * @classdesc The TDatabaseController is the base class to perform CRUD operations on the database
 */
import {
    isArray,
    isBlankString,
    isDefined,
    isEmptyArray,
    isEmptyObject,
    isEmptyString,
    isNotArray,
    isNotBoolean,
    isNotDefined,
    isNotObject,
    isNotString,
    isNull,
    isObject,
    isUndefined
}                                   from 'itee-validators'
import { TAbstractResponder }       from '../databases/TAbstractResponder'
import { UnprocessableEntityError } from '../messages/http/UnprocessableEntityError'
/**
 * @class
 * @classdesc The TDatabaseController is the base class to perform CRUD operations on the database
 * @augments module:Databases/TAbstractResponder~TAbstractResponder
 */
class TAbstractDataController extends TAbstractResponder {
    /**
     * @constructor
     * @param {Object} [parameters={}] - An object containing all parameters to pass through the inheritance chain to initialize this instance
     * @param {external:Others~DatabaseDriver} parameters.driver Any official database driver that will be used internally by inherited class
     * @param {boolean} [parameters.useNext=false] A boolean flag to indicate that this instance should use "next()" function instead of return response to client.
     */
    constructor ( parameters ) {
        const _parameters = {
            ...{
                driver:  null,
                useNext: false
            },
            ...parameters
        }
        super( _parameters )
        /**
         * The database drive to use internally
         * @throws {TypeError} Will throw an error if the argument is null.
         * @throws {TypeError} Will throw an error if the argument is undefined.
         */
        this.driver = _parameters.driver
        this.useNext = _parameters.useNext
    }
    get useNext () {
        return this._useNext
    }
    set useNext ( value ) {
        if ( isNull( value ) ) { throw new TypeError( 'Driver cannot be null ! Expect a database driver.' ) }
        if ( isUndefined( value ) ) { throw new TypeError( 'Driver cannot be undefined ! Expect a database driver.' ) }
        if ( isNotBoolean( value ) ) { throw new TypeError( 'Driver cannot be undefined ! Expect a database driver.' ) }
        this._useNext = value
    }
    get driver () {
        return this._driver
    }
    set driver ( value ) {
        if ( isNull( value ) ) { throw new TypeError( 'Driver cannot be null ! Expect a database driver.' ) }
        if ( isUndefined( value ) ) { throw new TypeError( 'Driver cannot be undefined ! Expect a database driver.' ) }
        this._driver = value
    }
    //////////////////
    // CRUD Methods //
    //////////////////
    create ( request, response, next ) {
        const data = request.body
        if ( isNotDefined( data ) ) {
            TAbstractDataController.returnError(
                new UnprocessableEntityError( 'Le corps de la requete ne peut pas être null ou indefini.' ),
                ( this.useNext ) ? next : response
            )
        } else if ( isArray( data ) ) {
            if ( isEmptyArray( data ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `Le tableau d'objet de la requete ne peut pas être vide.` ),
                    ( this.useNext ) ? next : response
                )
            } else {
                this._createMany( data, response, next )
            }
        } else if ( isObject( data ) ) {
            if ( isEmptyObject( data ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `L'objet de la requete ne peut pas être vide.` ),
                    ( this.useNext ) ? next : response
                )
            } else {
                this._createOne( data, response, next )
            }
        } else {
            TAbstractDataController.returnError(
                new UnprocessableEntityError( `Le type de donnée de la requete est invalide. Les paramètres valides sont objet ou un tableau d'objet.` ),
                ( this.useNext ) ? next : response
            )
        }
    }
    _createOne ( /*data, response, next*/ ) {}
    _createMany ( /*datas, response, next*/ ) {}
    read ( request, response, next ) {
        const id          = request.params[ 'id' ]
        const requestBody = request.body
        const haveBody    = ( isDefined( requestBody ) )
        const ids         = ( haveBody ) ? requestBody.ids : null
        const query       = ( haveBody ) ? requestBody.query : null
        const projection  = ( haveBody ) ? requestBody.projection : null
        if ( isDefined( id ) ) {
            if ( isNotString( id ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `L'identifiant devrait être une chaine de caractères.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyString( id ) || isBlankString( id ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `L'identifiant ne peut pas être une chaine de caractères vide.` ),
                    ( this.useNext ) ? next : response
                )
            } else {
                this._readOne( id, projection, response, next )
            }
        } else if ( isDefined( ids ) ) {
            if ( isNotArray( ids ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `Le tableau d'identifiants devrait être un tableau de chaine de caractères.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyArray( ids ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `Le tableau d'identifiants ne peut pas être vide.` ),
                    ( this.useNext ) ? next : response
                )
            } else {
                this._readMany( ids, projection, response, next )
            }
        } else if ( isDefined( query ) ) {
            if ( isNotObject( query ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `La requete devrait être un objet javascript.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyObject( query ) ) {
                this._readAll( projection, response, next )
            } else {
                this._readWhere( query, projection, response, next )
            }
        } else {
            TAbstractDataController.returnError(
                new UnprocessableEntityError( `La requete ne peut pas être null.` ),
                ( this.useNext ) ? next : response
            )
        }
    }
    _readOne ( /*id, projection, response, next*/ ) {}
    _readMany ( /*ids, projection, response, next*/ ) {}
    _readWhere ( /*query, projection, response, next*/ ) {}
    _readAll ( /*projection, response, next*/ ) {}
    update ( request, response, next ) {
        const id          = request.params[ 'id' ]
        const requestBody = request.body
        const haveBody    = ( isDefined( requestBody ) )
        const ids         = ( haveBody ) ? requestBody.ids : null
        const query       = ( haveBody ) ? requestBody.query : null
        const update      = ( haveBody ) ? requestBody.update : null
        if ( isNotDefined( update ) ) {
            TAbstractDataController.returnError(
                new UnprocessableEntityError( `La mise à jour a appliquer ne peut pas être null ou indefini.` ),
                ( this.useNext ) ? next : response
            )
        } else if ( isDefined( id ) ) {
            if ( isNotString( id ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `L'identifiant devrait être une chaine de caractères.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyString( id ) || isBlankString( id ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `L'identifiant ne peut pas être une chaine de caractères vide.` ),
                    ( this.useNext ) ? next : response
                )
            } else {
                this._updateOne( id, update, response, next )
            }
        } else if ( isDefined( ids ) ) {
            if ( isNotArray( ids ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `Le tableau d'identifiants devrait être un tableau de chaine de caractères.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyArray( ids ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `Le tableau d'identifiants ne peut pas être vide.` ),
                    ( this.useNext ) ? next : response
                )
            } else {
                this._updateMany( ids, update, response, next )
            }
        } else if ( isDefined( query ) ) {
            if ( isNotObject( query ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `La requete devrait être un objet javascript.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyObject( query ) ) {
                this._updateAll( update, response, next )
            } else {
                this._updateWhere( query, update, response, next )
            }
        } else {
            TAbstractDataController.returnError(
                new UnprocessableEntityError( `La requete ne peut pas être vide.` ),
                ( this.useNext ) ? next : response
            )
        }
    }
    _updateOne ( /*id, update, response, next*/ ) {}
    _updateMany ( /*ids, updates, response, next*/ ) {}
    _updateWhere ( /*query, update, response, next*/ ) {}
    _updateAll ( /*update, response, next*/ ) {}
    delete ( request, response, next ) {
        const id          = request.params[ 'id' ]
        const requestBody = request.body
        const haveBody    = ( isDefined( requestBody ) )
        const ids         = ( haveBody ) ? requestBody.ids : null
        const query       = ( haveBody ) ? requestBody.query : null
        if ( isDefined( id ) ) {
            if ( isNotString( id ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `L'identifiant devrait être une chaine de caractères.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyString( id ) || isBlankString( id ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `L'identifiant ne peut pas être une chaine de caractères vide.` ),
                    ( this.useNext ) ? next : response
                )
            } else {
                this._deleteOne( id, response, next )
            }
        } else if ( isDefined( ids ) ) {
            if ( isNotArray( ids ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `Le tableau d'identifiants devrait être un tableau de chaine de caractères.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyArray( ids ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `Le tableau d'identifiants ne peut pas être vide.` ),
                    ( this.useNext ) ? next : response
                )
            } else {
                this._deleteMany( ids, response, next )
            }
        } else if ( isDefined( query ) ) {
            if ( isNotObject( query ) ) {
                TAbstractDataController.returnError(
                    new UnprocessableEntityError( `La requete devrait être un objet javascript.` ),
                    ( this.useNext ) ? next : response
                )
            } else if ( isEmptyObject( query ) ) {
                this._deleteAll( response, next )
            } else {
                this._deleteWhere( query, response, next )
            }
        } else {
            TAbstractDataController.returnError(
                new UnprocessableEntityError( `La requete ne peut pas être vide.` ),
                ( this.useNext ) ? next : response
            )
        }
    }
    _deleteOne ( /*id, response, next*/ ) {}
    _deleteMany ( /*ids, response, next*/ ) {}
    _deleteWhere ( /*query, response, next*/ ) {}
    _deleteAll ( /*response, next*/ ) {}
}
export { TAbstractDataController }