/*!
 * Copyright (C) 2017 AT&T Intellectual Property. All rights reserved.
 *
 * 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 {actionTypes} from './JSONSchemaReducerConstants.js';
import Validator from 'nfvo-utils/Validator.js';
import JSONSchema from 'nfvo-utils/json/JSONSchema.js';
import JSONPointer from 'nfvo-utils/json/JSONPointer.js';
import forOwn from 'lodash/forOwn.js';
import isArray from 'lodash/isArray.js';


function flattenData(data, result, pointer = '') {
	let newPointer = pointer;
	if (typeof data === 'object' && !isArray(data)) {
		for (let i in data) {
			newPointer = newPointer ? newPointer + '/' + i : i;
			flattenData(data[i], result, newPointer);
			newPointer = pointer;
		}
	} else {
		result[newPointer] = data;
	}
}

function updateSchemaDataAndValidateReducer (state = {}, action, questionnaireName) {
	let genericFieldInfoClone;
	switch (action.type) {
		case actionTypes.DATA_LOADED:
			if (questionnaireName !== action.qName) {return {...state};}
			const schema = action.payload.qschema;
			let schemaLoader = new JSONSchema();
			schemaLoader.setSchema(schema);
			schemaLoader.setSupportedValidationFunctions(Object.keys(Validator.globalValidationFunctions));
			let {genericFieldInfo} = schemaLoader.flattenSchema();

			let data = action.payload.qdata;
			let dataMap =  {};
			flattenData(data, dataMap);

			return {
				...state,
				qdata: action.payload.qdata, // the original hierarchical data. to be used for submit and save
				qgenericFieldInfo : genericFieldInfo, // information about the fields that the view will require and reducer will need, such as validations, enum to use, etc.
				dataMap //  flattened schema data for ease of use
			};

		case actionTypes.DATA_CHANGED:
			let changedData = action.deltaData;
			if (questionnaireName !== action.qName)  {return {...state};}

			genericFieldInfoClone = {...state.qgenericFieldInfo};
			let qDataClone = {...state.qdata};
			let dataMapClone = {...state.dataMap};

			forOwn(changedData,(value, key) => {
				if (state.qgenericFieldInfo[key]) {
					let result = Validator.validate(key, value, state.qgenericFieldInfo[key].validations, state, action.customValidations);
					genericFieldInfoClone[key] = {...genericFieldInfoClone[key], isValid: result.isValid, errorText: result.errorText};
					qDataClone = JSONPointer.setValue(state.qdata, '/' + key, value);
					dataMapClone[key] = value;
				}
			});

			return {
				...state,
				qdata: qDataClone,
				dataMap: dataMapClone,
				qgenericFieldInfo: genericFieldInfoClone
			};

		case actionTypes.VALIDATE_DATA:
			let specificFields = action.data;
			if (questionnaireName !== action.qName)  {return {...state};}
			genericFieldInfoClone = {...state.qgenericFieldInfo};
			forOwn(specificFields,(value, key) => {
				let result = Validator.validate(key, value, state.qgenericFieldInfo[key].validations, state, action.customValidations);
				genericFieldInfoClone[key] = {...genericFieldInfoClone[key], isValid: result.isValid, errorText: result.errorText};
			});
			return {
				...state,
				formReady: null,
				qgenericFieldInfo: genericFieldInfoClone
			};

		case actionTypes.VALIDATE_FORM:
			if (questionnaireName !== action.qName)  {return {...state};}
			genericFieldInfoClone = {...state.qgenericFieldInfo};
			let formReady = true;
			forOwn(state.qgenericFieldInfo,(value, key) => {
				let val = state.dataMap[key] ? state.dataMap[key] : '';
				let result = Validator.validate(key, val, state.qgenericFieldInfo[key].validations, state, action.customValidations);
				genericFieldInfoClone[key] = {...genericFieldInfoClone[key], isValid: result.isValid, errorText: result.errorText};
				if (!result.isValid) {
					formReady = false;
				}
			});
			return {
				...state,
				formReady,
				qgenericFieldInfo: genericFieldInfoClone
			};

		default:
			return state;
	}
};

export function createJSONSchemaReducer(questionnaireName) {
	return (state = {}, action) => {
		return updateSchemaDataAndValidateReducer(state, action, questionnaireName);
	};
};

export function createComposedJSONSchemaReducer(questionnaireName, additionalActionsReducer) {
	return (state = {}, action) => {
		if(action.type === actionTypes.VALIDATE_DATA ||
			action.type === actionTypes.VALIDATE_FORM ||
			action.type === actionTypes.DATA_CHANGED ||
			action.type === actionTypes.DATA_LOADED
		) {
			return updateSchemaDataAndValidateReducer(state, action, questionnaireName);
		} else {
			return additionalActionsReducer(state, action);
		}
	};
};