'use strict';


var	async = require('async');
var assert = require('assert');
var db = require('../mocks/databasemock');

describe('List methods', function () {
	describe('listAppend()', function () {
		it('should append to a list', function (done) {
			db.listAppend('testList1', 5, function (err) {
				assert.ifError(err);
				assert.equal(arguments.length, 1);
				done();
			});
		});

		it('should not add anyhing if key is falsy', function (done) {
			db.listAppend(null, 3, function (err) {
				assert.ifError(err);
				done();
			});
		});
	});

	describe('listPrepend()', function () {
		it('should prepend to a list', function (done) {
			db.listPrepend('testList2', 3, function (err) {
				assert.equal(err, null);
				assert.equal(arguments.length, 1);
				done();
			});
		});

		it('should prepend 2 more elements to a list', function (done) {
			async.series([
				function (next) {
					db.listPrepend('testList2', 2, next);
				},
				function (next) {
					db.listPrepend('testList2', 1, next);
				},
			], function (err) {
				assert.equal(err, null);
				done();
			});
		});

		it('should not add anyhing if key is falsy', function (done) {
			db.listPrepend(null, 3, function (err) {
				assert.ifError(err);
				done();
			});
		});
	});

	describe('getListRange()', function () {
		before(function (done) {
			async.series([
				function (next) {
					db.listAppend('testList3', 7, next);
				},
				function (next) {
					db.listPrepend('testList3', 3, next);
				},
				function (next) {
					db.listAppend('testList4', 5, next);
				},
			], done);
		});

		it('should return an empty list', function (done) {
			db.getListRange('doesnotexist', 0, -1, function (err, list) {
				assert.equal(err, null);
				assert.equal(arguments.length, 2);
				assert.equal(Array.isArray(list), true);
				assert.equal(list.length, 0);
				done();
			});
		});

		it('should return a list with one element', function (done) {
			db.getListRange('testList4', 0, 0, function (err, list) {
				assert.equal(err, null);
				assert.equal(Array.isArray(list), true);
				assert.equal(list[0], 5);
				done();
			});
		});

		it('should return a list with 2 elements 3, 7', function (done) {
			db.getListRange('testList3', 0, -1, function (err, list) {
				assert.equal(err, null);
				assert.equal(Array.isArray(list), true);
				assert.equal(list.length, 2);
				assert.deepEqual(list, ['3', '7']);
				done();
			});
		});

		it('should not get anything if key is falsy', function (done) {
			db.getListRange(null, 0, -1, function (err, data) {
				assert.ifError(err);
				assert.equal(data, undefined);
				done();
			});
		});
	});

	describe('listRemoveLast()', function () {
		before(function (done) {
			async.series([
				function (next) {
					db.listAppend('testList7', 12, next);
				},
				function (next) {
					db.listPrepend('testList7', 9, next);
				},
			], done);
		});

		it('should remove the last element of list and return it', function (done) {
			db.listRemoveLast('testList7', function (err, lastElement) {
				assert.equal(err, null);
				assert.equal(arguments.length, 2);
				assert.equal(lastElement, '12');
				done();
			});
		});

		it('should not remove anyhing if key is falsy', function (done) {
			db.listRemoveLast(null, function (err) {
				assert.ifError(err);
				done();
			});
		});
	});

	describe('listRemoveAll()', function () {
		before(function (done) {
			async.series([
				async.apply(db.listAppend, 'testList5', 1),
				async.apply(db.listAppend, 'testList5', 1),
				async.apply(db.listAppend, 'testList5', 1),
				async.apply(db.listAppend, 'testList5', 2),
				async.apply(db.listAppend, 'testList5', 5),
			], done);
		});

		it('should remove all the matching elements of list', function (done) {
			db.listRemoveAll('testList5', '1', function (err) {
				assert.equal(err, null);
				assert.equal(arguments.length, 1);

				db.getListRange('testList5', 0, -1, function (err, list) {
					assert.equal(err, null);
					assert.equal(Array.isArray(list), true);
					assert.equal(list.length, 2);
					assert.equal(list.indexOf('1'), -1);
					done();
				});
			});
		});

		it('should not remove anyhing if key is falsy', function (done) {
			db.listRemoveAll(null, 3, function (err) {
				assert.ifError(err);
				done();
			});
		});
	});

	describe('listTrim()', function () {
		it('should trim list to a certain range', function (done) {
			var list = ['1', '2', '3', '4', '5'];
			async.eachSeries(list, function (value, next) {
				db.listAppend('testList6', value, next);
			}, function (err) {
				if (err) {
					return done(err);
				}

				db.listTrim('testList6', 0, 2, function (err) {
					assert.equal(err, null);
					assert.equal(arguments.length, 1);
					db.getListRange('testList6', 0, -1, function (err, list) {
						assert.equal(err, null);
						assert.equal(list.length, 3);
						assert.deepEqual(list, ['1', '2', '3']);
						done();
					});
				});
			});
		});

		it('should not add anyhing if key is falsy', function (done) {
			db.listTrim(null, 0, 3, function (err) {
				assert.ifError(err);
				done();
			});
		});
	});

	describe('listLength', function () {
		it('should get the length of a list', function (done) {
			db.listAppend('getLengthList', 1, function (err) {
				assert.ifError(err);
				db.listAppend('getLengthList', 2, function (err) {
					assert.ifError(err);
					db.listLength('getLengthList', function (err, length) {
						assert.ifError(err);
						assert.equal(length, 2);
						done();
					});
				});
			});
		});

		it('should return 0 if list does not have any elements', function (done) {
			db.listLength('doesnotexist', function (err, length) {
				assert.ifError(err);
				assert.strictEqual(length, 0);
				done();
			});
		});
	});
});