"use strict";
var assert = require('assert');
var vows = require('vows');
var layouts = require('../lib/layouts');
var sandbox = require('sandboxed-module');
var LoggingEvent = require('../lib/logger').LoggingEvent;
var cluster = require('cluster');

vows.describe('log4js cluster appender').addBatch({
	'when in master mode': {
		topic: function() {

			var registeredClusterEvents = [];
			var loggingEvents = [];
			var onChildProcessForked;
			var onMasterReceiveChildMessage;

			// Fake cluster module, so no real cluster listeners be really added
			var fakeCluster = {

				on: function(event, callback) {
					registeredClusterEvents.push(event);
					onChildProcessForked = callback;
				},

				isMaster: true,
				isWorker: false,

			};
			var fakeWorker = {
				on: function(event, callback) {
					onMasterReceiveChildMessage = callback;
				},
				process: {
					pid: 123
				},
				id: 'workerid'
			};

			var fakeActualAppender = function(loggingEvent) {
				loggingEvents.push(loggingEvent);
			};

			// Load appender and fake modules in it
			var appenderModule = sandbox.require('../lib/appenders/clustered', {
				requires: {
					'cluster': fakeCluster,
				}
			});

			var masterAppender = appenderModule.appender({
				actualAppenders: [fakeActualAppender, fakeActualAppender, fakeActualAppender],
				appenders: [{}, {category: "test"}, {category: "wovs"}]
			});

			// Actual test - log message using masterAppender
			masterAppender(new LoggingEvent('wovs', 'Info', ['masterAppender test']));

			// Simulate a 'fork' event to register the master's message handler on our fake worker.
			onChildProcessForked(fakeWorker);
			// Simulate a cluster message received by the masterAppender.
			var simulatedLoggingEvent = new LoggingEvent(
				'wovs',
				'Error',
				[
					'message deserialization test',
					{stack: 'my wrapped stack'}
				]
			);
			onMasterReceiveChildMessage({
				type : '::log-message',
				event : JSON.stringify(simulatedLoggingEvent)
			});

			var returnValue = {
				registeredClusterEvents: registeredClusterEvents,
				loggingEvents: loggingEvents,
			};

			return returnValue;
		},

		"should register 'fork' event listener on 'cluster'": function(topic) {
			assert.equal(topic.registeredClusterEvents[0], 'fork');
		},

		"should log using actual appender": function(topic) {
			assert.equal(topic.loggingEvents.length, 4);
			assert.equal(topic.loggingEvents[0].data[0], 'masterAppender test');
			assert.equal(topic.loggingEvents[1].data[0], 'masterAppender test');
			assert.equal(topic.loggingEvents[2].data[0], 'message deserialization test');
			assert.equal(topic.loggingEvents[2].data[1], 'my wrapped stack');
			assert.equal(topic.loggingEvents[3].data[0], 'message deserialization test');
			assert.equal(topic.loggingEvents[3].data[1], 'my wrapped stack');
		},

	},

	'when in worker mode': {

		topic: function() {

			var registeredProcessEvents = [];

			// Fake cluster module, to fake we're inside a worker process
			var fakeCluster = {

				isMaster: false,
				isWorker: true,

			};

			var fakeProcess = {

				send: function(data) {
					registeredProcessEvents.push(data);
				},

			};

			// Load appender and fake modules in it
			var appenderModule = sandbox.require('../lib/appenders/clustered', {
				requires: {
					'cluster': fakeCluster,
				},
				globals: {
					'process': fakeProcess,
				}
			});

			var workerAppender = appenderModule.appender();

			// Actual test - log message using masterAppender
			workerAppender(new LoggingEvent('wovs', 'Info', ['workerAppender test']));
			workerAppender(new LoggingEvent('wovs', 'Info', [new Error('Error test')]));

			var returnValue = {
				registeredProcessEvents: registeredProcessEvents,
			};

			return returnValue;

		},

		"worker appender should call process.send" : function(topic) {
			assert.equal(topic.registeredProcessEvents[0].type, '::log-message');
			assert.equal(
				JSON.parse(topic.registeredProcessEvents[0].event).data[0],
				"workerAppender test"
			);
		},

		"worker should serialize an Error correctly" : function(topic) {
			assert.equal(topic.registeredProcessEvents[1].type, '::log-message');
			assert(JSON.parse(topic.registeredProcessEvents[1].event).data[0].stack);
			var actual = JSON.parse(topic.registeredProcessEvents[1].event).data[0].stack;
			var expectedRegex = /^Error: Error test/;
			assert(
				actual.match(expectedRegex),
				"Expected: \n\n " + actual + "\n\n to match " + expectedRegex
			);
		}

	}

}).exportTo(module);