import * as https from 'https';
import * as stream from 'stream';
import * as zlib from 'zlib';
import * as Bluebird from 'bluebird';
import { expect } from 'chai';
import * as sinon from 'sinon';
import { setTimeout } from 'timers/promises';

import * as config from '~/src/config';

describe('Logger', function () {
	let logger: typeof import('~/src/logger');
	let configStub: sinon.SinonStub;

	beforeEach(async function () {
		this._req = new stream.PassThrough();
		this._req.flushHeaders = sinon.spy();
		this._req.end = sinon.spy();

		this._req.body = '';
		this._req.pipe(zlib.createGunzip()).on('data', (chunk: Buffer) => {
			this._req.body += chunk;
		});

		this.requestStub = sinon.stub(https, 'request').returns(this._req);

		configStub = sinon.stub(config, 'getMany').returns(
			// @ts-expect-error this should actually work but the type system doesnt like it
			Bluebird.resolve({
				apiEndpoint: 'https://example.com',
				uuid: 'deadbeef',
				deviceApiKey: 'secretkey',
				unmanaged: false,
				loggingEnabled: true,
				localMode: false,
			}),
		);
		// delete the require cache for the logger module so we can force a refresh
		delete require.cache[require.resolve('~/src/logger')];
		logger = require('~/src/logger');
		await logger.initialized();
	});

	afterEach(function () {
		this.requestStub.restore();
		configStub.restore();
	});

	after(function () {
		delete require.cache[require.resolve('~/src/logger')];
	});

	it('waits the grace period before sending any logs', async function () {
		const clock = sinon.useFakeTimers();
		logger.log({ message: 'foobar', serviceId: 15 });
		clock.tick(4999);
		clock.restore();

		await setTimeout(100);
		expect(this._req.body).to.equal('');
	});

	it('tears down the connection after inactivity', async function () {
		const clock = sinon.useFakeTimers();
		logger.log({ message: 'foobar', serviceId: 15 });
		clock.tick(61000);
		clock.restore();

		await setTimeout(100);
		expect(this._req.end.calledOnce).to.be.true;
	});

	it('sends logs as gzipped ndjson', async function () {
		const timestamp = Date.now();
		logger.log({ message: 'foobar', serviceId: 15 });
		logger.log({ timestamp: 1337, message: 'foobar', serviceId: 15 });
		logger.log({ message: 'foobar' }); // shold be ignored

		await setTimeout(5500);
		expect(this.requestStub.calledOnce).to.be.true;
		const opts = this.requestStub.firstCall.args[0];

		expect(opts.href).to.equal(
			'https://example.com/device/v2/deadbeef/log-stream',
		);
		expect(opts.method).to.equal('POST');
		expect(opts.headers).to.deep.equal({
			Authorization: 'Bearer secretkey',
			'Content-Type': 'application/x-ndjson',
			'Content-Encoding': 'gzip',
		});

		const lines = this._req.body.split('\n');
		expect(lines.length).to.equal(3);
		expect(lines[2]).to.equal('');

		let msg = JSON.parse(lines[0]);
		expect(msg).to.have.property('message').that.equals('foobar');
		expect(msg).to.have.property('serviceId').that.equals(15);
		expect(msg).to.have.property('timestamp').that.is.at.least(timestamp);
		msg = JSON.parse(lines[1]);
		expect(msg).to.deep.equal({
			timestamp: 1337,
			message: 'foobar',
			serviceId: 15,
		});
	});

	it('allows logging system messages which are also reported to the eventTracker', async function () {
		const timestamp = Date.now();
		logger.logSystemMessage(
			'Hello there!',
			{ someProp: 'someVal' },
			'Some event name',
		);

		await setTimeout(5500);
		const lines = this._req.body.split('\n');
		expect(lines.length).to.equal(2);
		expect(lines[1]).to.equal('');

		const msg = JSON.parse(lines[0]);
		expect(msg).to.have.property('message').that.equals('Hello there!');
		expect(msg).to.have.property('isSystem').that.equals(true);
		expect(msg).to.have.property('timestamp').that.is.at.least(timestamp);
	});
});