2021-02-13 13:40:18 +01:00
|
|
|
// external lib
|
2021-02-14 17:07:23 +01:00
|
|
|
import Parser from "rss-parser";
|
2021-02-07 17:50:54 +01:00
|
|
|
|
2021-02-13 13:40:18 +01:00
|
|
|
// tested class
|
|
|
|
import {
|
2021-02-14 15:00:33 +01:00
|
|
|
ListenerRSSInfos as ListenerRRSInfo,
|
2021-02-13 13:40:18 +01:00
|
|
|
ListenerRss as Listeners,
|
2021-02-14 15:00:33 +01:00
|
|
|
} from "./../src/index";
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
// Unit test
|
2021-02-14 15:00:33 +01:00
|
|
|
import * as chai from "chai";
|
|
|
|
import * as sinon from "ts-sinon";
|
|
|
|
|
|
|
|
const sinonChai = require("sinon-chai");
|
|
|
|
|
|
|
|
chai.use(sinonChai);
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
const expect = chai.expect;
|
|
|
|
|
|
|
|
describe("test class RSS: jsonfile", function () {
|
|
|
|
let myListener: Listeners | undefined = undefined;
|
|
|
|
|
2021-02-14 17:07:23 +01:00
|
|
|
const infosListener: ListenerRRSInfo = {
|
|
|
|
name: "my-test-service",
|
|
|
|
address: "fake.rss.service",
|
|
|
|
timeloop: 15,
|
2021-02-14 17:37:03 +01:00
|
|
|
customfields: {
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
},
|
2021-02-14 17:07:23 +01:00
|
|
|
};
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
// parseURL tests
|
2021-02-25 13:14:34 +01:00
|
|
|
let stubListener: sinon.StubbedInstance<Listeners>;
|
2021-02-14 15:00:33 +01:00
|
|
|
let stubParser: sinon.StubbedInstance<Parser>;
|
2021-02-25 13:14:34 +01:00
|
|
|
|
2021-02-25 14:02:02 +01:00
|
|
|
const mockedRSSOutput: Parser.Output<{
|
|
|
|
"media:group": { [key: string]: string | [any] };
|
|
|
|
}> = {
|
2021-02-13 13:40:18 +01:00
|
|
|
items: [
|
|
|
|
{
|
|
|
|
title: "my title 00",
|
|
|
|
"media:group": {
|
|
|
|
"media:description": "my description 00",
|
|
|
|
"media:thumbnail": [
|
|
|
|
{ $: { height: 360, width: 420, url: "my_image00.jpg" } },
|
|
|
|
],
|
|
|
|
},
|
|
|
|
link: "my_url_00.com",
|
|
|
|
pubDate: "myDate00",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "my title 01",
|
|
|
|
"media:group": {
|
|
|
|
"media:description": "my description 01",
|
|
|
|
"media:thumbnail": [
|
|
|
|
{ $: { height: 360, width: 420, url: "my_image01.jpg" } },
|
|
|
|
],
|
|
|
|
},
|
|
|
|
link: "my_url_01.com",
|
|
|
|
pubDate: "myDate01",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "my title 02",
|
|
|
|
"media:group": {
|
|
|
|
"media:description": "my description 02",
|
|
|
|
"media:thumbnail": [
|
|
|
|
{ $: { height: 360, width: 420, url: "my_image02.jpg" } },
|
|
|
|
],
|
|
|
|
},
|
|
|
|
link: "my_url_02.com",
|
|
|
|
pubDate: "myDate02",
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
|
2021-02-25 13:14:34 +01:00
|
|
|
/**
|
|
|
|
* The function create my Stubs for my Listener and my Parser
|
|
|
|
*/
|
|
|
|
const fun_initStub: () => void = () => {
|
|
|
|
if (myListener !== undefined && myListener.parser !== undefined) {
|
|
|
|
stubListener = sinon.stubObject<Listeners>(myListener, [
|
|
|
|
"setParser",
|
|
|
|
"fetchRSS",
|
|
|
|
]);
|
|
|
|
stubListener.setParser.callsFake(() => {
|
|
|
|
if (stubListener.parser !== undefined) {
|
|
|
|
stubParser = sinon.stubObject<Parser>(stubListener.parser, [
|
|
|
|
"parseURL",
|
|
|
|
]);
|
|
|
|
stubParser.parseURL
|
|
|
|
.withArgs(infosListener.address)
|
|
|
|
.resolves(mockedRSSOutput);
|
|
|
|
stubParser.parseURL
|
|
|
|
.withArgs("bad.rss.service")
|
|
|
|
.rejects(new Error("connect ECONNREFUSED 127.0.0.1:80"));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
stubListener.setParser();
|
2021-02-25 14:02:02 +01:00
|
|
|
stubListener.fetchRSS.returns(stubParser.parseURL(stubListener.address));
|
2021-02-25 13:14:34 +01:00
|
|
|
} else throw new Error("myListener need to be initiliaze before the stub");
|
|
|
|
};
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
afterEach(function () {
|
|
|
|
// restore stubs
|
2021-02-25 13:14:34 +01:00
|
|
|
myListener = undefined;
|
2021-02-13 13:40:18 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("Building Ytb listener", function () {
|
|
|
|
it("The build without issues (infosListener parameters)", function () {
|
|
|
|
myListener = new Listeners(infosListener);
|
|
|
|
|
|
|
|
// assertions
|
|
|
|
// myListener data
|
|
|
|
expect(myListener.timeloop).to.eql(15);
|
|
|
|
expect(myListener.name).to.eql("my-test-service");
|
|
|
|
expect(myListener.address).to.eql("fake.rss.service");
|
2021-02-14 17:37:03 +01:00
|
|
|
expect(myListener.customfields).to.eql({
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
});
|
2021-02-25 13:14:34 +01:00
|
|
|
expect(myListener.parser)
|
|
|
|
.to.have.property("options")
|
|
|
|
.to.have.property("customFields")
|
|
|
|
.to.be.eql({
|
|
|
|
feed: [],
|
|
|
|
item: ["media:group", "media:group"],
|
|
|
|
});
|
2021-02-13 13:40:18 +01:00
|
|
|
});
|
2021-02-21 00:14:04 +01:00
|
|
|
it("The build without issues (raw infos : 4 params)", function () {
|
|
|
|
myListener = new Listeners({
|
|
|
|
name: "my-test-service",
|
|
|
|
address: "fake.rss.service",
|
|
|
|
timeloop: 15,
|
|
|
|
customfields: {
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
// assertions
|
|
|
|
// myListener data
|
|
|
|
expect(myListener.timeloop).to.eql(15);
|
|
|
|
expect(myListener.name).to.eql("my-test-service");
|
|
|
|
expect(myListener.address).to.eql("fake.rss.service");
|
|
|
|
expect(myListener.customfields).to.eql({
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
});
|
|
|
|
expect(myListener.parser)
|
|
|
|
.to.have.property("options")
|
|
|
|
.to.have.property("customFields")
|
|
|
|
.to.be.eql({
|
|
|
|
feed: [],
|
|
|
|
item: ["media:group", "media:group"],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it("The build without issues (raw infos : just 2 params)", function () {
|
|
|
|
myListener = new Listeners({
|
|
|
|
name: "my-test-service",
|
|
|
|
address: "fake.rss.service",
|
|
|
|
});
|
|
|
|
|
|
|
|
// assertions
|
|
|
|
// myListener data
|
|
|
|
expect(myListener.timeloop).to.eql(5 * 60);
|
|
|
|
expect(myListener.name).to.eql("my-test-service");
|
|
|
|
expect(myListener.address).to.eql("fake.rss.service");
|
|
|
|
expect(myListener.customfields).to.eql(undefined);
|
|
|
|
expect(myListener.parser)
|
|
|
|
.to.have.property("options")
|
|
|
|
.to.have.property("customFields")
|
|
|
|
.to.be.eql({
|
|
|
|
feed: [],
|
|
|
|
item: [],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it("The build without issues (raw infos : just 3 params (no custom fields))", function () {
|
|
|
|
myListener = new Listeners({
|
|
|
|
name: "my-test-service",
|
|
|
|
address: "fake.rss.service",
|
|
|
|
timeloop: 15,
|
|
|
|
});
|
|
|
|
|
|
|
|
// assertions
|
|
|
|
// myListener data
|
|
|
|
expect(myListener.timeloop).to.eql(15);
|
|
|
|
expect(myListener.name).to.eql("my-test-service");
|
|
|
|
expect(myListener.address).to.eql("fake.rss.service");
|
|
|
|
expect(myListener.customfields).to.eql(undefined);
|
|
|
|
expect(myListener.parser)
|
|
|
|
.to.have.property("options")
|
|
|
|
.to.have.property("customFields")
|
|
|
|
.to.be.eql({
|
|
|
|
feed: [],
|
|
|
|
item: [],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it("The build without issues (raw infos : just 3 params (no timeloop))", function () {
|
|
|
|
myListener = new Listeners({
|
|
|
|
name: "my-test-service",
|
|
|
|
address: "fake.rss.service",
|
|
|
|
customfields: {
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
// assertions
|
|
|
|
// myListener data
|
|
|
|
expect(myListener.timeloop).to.eql(5 * 60);
|
|
|
|
expect(myListener.name).to.eql("my-test-service");
|
|
|
|
expect(myListener.address).to.eql("fake.rss.service");
|
|
|
|
expect(myListener.customfields).to.eql({
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
});
|
|
|
|
expect(myListener.parser)
|
|
|
|
.to.have.property("options")
|
|
|
|
.to.have.property("customFields")
|
|
|
|
.to.be.eql({
|
|
|
|
feed: [],
|
|
|
|
item: ["media:group", "media:group"],
|
|
|
|
});
|
2021-02-13 13:40:18 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("fetch some data", function () {
|
|
|
|
it("fetch without issues", function () {
|
|
|
|
myListener = new Listeners(infosListener);
|
2021-02-25 13:14:34 +01:00
|
|
|
fun_initStub();
|
|
|
|
|
|
|
|
expect(myListener).to.not.be.undefined;
|
|
|
|
if (myListener !== undefined) {
|
|
|
|
// fetch
|
|
|
|
let res = stubListener.fetchRSS();
|
|
|
|
|
|
|
|
//assertion
|
|
|
|
// calls
|
|
|
|
expect(stubParser.parseURL).to.have.been.calledOnce;
|
|
|
|
expect(stubParser.parseURL).to.have.been.calledWith(
|
|
|
|
infosListener.address
|
|
|
|
);
|
|
|
|
|
|
|
|
res
|
|
|
|
.then((obj: any) => {
|
|
|
|
expect(obj).to.be.eql(mockedRSSOutput);
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
expect(err).to.be.undefined;
|
|
|
|
});
|
|
|
|
} else throw new Error("Error into the before instruction");
|
2021-02-13 13:40:18 +01:00
|
|
|
});
|
2021-02-25 13:14:34 +01:00
|
|
|
|
2021-02-13 13:40:18 +01:00
|
|
|
it("fetch with bad address", function () {
|
2021-02-14 17:07:23 +01:00
|
|
|
myListener = new Listeners({
|
|
|
|
name: "my-test-service",
|
|
|
|
address: "bad.rss.service",
|
2021-02-14 17:37:03 +01:00
|
|
|
customfields: {
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
},
|
2021-02-14 17:07:23 +01:00
|
|
|
});
|
2021-02-25 13:14:34 +01:00
|
|
|
fun_initStub();
|
2021-02-13 13:40:18 +01:00
|
|
|
// fetch
|
2021-02-25 13:14:34 +01:00
|
|
|
let res = stubListener.fetchRSS();
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
//assertion
|
|
|
|
// calls
|
2021-02-14 15:00:33 +01:00
|
|
|
expect(stubParser.parseURL).to.have.been.calledOnce;
|
|
|
|
expect(stubParser.parseURL).to.have.been.calledWith("bad.rss.service");
|
2021-02-13 13:40:18 +01:00
|
|
|
// Promise
|
2021-02-21 00:14:04 +01:00
|
|
|
res
|
|
|
|
.then((obj: any) => {
|
|
|
|
expect(obj).to.be.undefined;
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
expect(err).to.be.eql(new Error("connect ECONNREFUSED 127.0.0.1:80"));
|
|
|
|
});
|
2021-02-13 13:40:18 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-02-21 00:14:04 +01:00
|
|
|
describe("start", function () {
|
2021-02-25 13:14:34 +01:00
|
|
|
it("Let's start the timer", async function () {
|
|
|
|
let clock: sinon.default.SinonFakeTimers = sinon.default.useFakeTimers();
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
// classic build
|
2021-02-14 17:07:23 +01:00
|
|
|
myListener = new Listeners({
|
|
|
|
name: "my-test-service",
|
|
|
|
address: "fake.rss.service",
|
2021-02-25 13:14:34 +01:00
|
|
|
timeloop: 60,
|
2021-02-14 17:37:03 +01:00
|
|
|
customfields: {
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
},
|
2021-02-14 17:07:23 +01:00
|
|
|
});
|
2021-02-25 13:14:34 +01:00
|
|
|
fun_initStub();
|
|
|
|
stubListener.fetchRSS.reset();
|
|
|
|
stubListener.fetchRSS.resolves(mockedRSSOutput);
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
//spy
|
2021-02-25 13:14:34 +01:00
|
|
|
let fun_spy: sinon.default.SinonSpy = sinon.default.spy((obj, err) => {
|
2021-02-21 00:14:04 +01:00
|
|
|
expect(obj).to.be.eql(mockedRSSOutput);
|
2021-02-25 13:14:34 +01:00
|
|
|
expect(err).to.be.eql(undefined);
|
2021-02-21 00:14:04 +01:00
|
|
|
});
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
// start timer
|
2021-02-25 13:14:34 +01:00
|
|
|
stubListener.start(fun_spy);
|
2021-02-13 13:40:18 +01:00
|
|
|
|
2021-02-25 13:14:34 +01:00
|
|
|
// wait and assertion
|
|
|
|
// After 1ms
|
|
|
|
await clock.tickAsync(1);
|
|
|
|
expect(stubListener.fetchRSS).to.have.been.calledOnce;
|
2021-02-21 00:14:04 +01:00
|
|
|
expect(fun_spy).to.have.been.calledOnce;
|
2021-02-13 13:40:18 +01:00
|
|
|
|
2021-02-25 13:14:34 +01:00
|
|
|
// After 60s
|
|
|
|
await clock.tickAsync(59999);
|
|
|
|
expect(stubListener.fetchRSS).to.have.been.calledTwice;
|
|
|
|
expect(fun_spy).to.have.been.calledTwice;
|
2021-02-13 13:40:18 +01:00
|
|
|
|
2021-02-25 13:14:34 +01:00
|
|
|
stubListener.stop();
|
2021-02-13 13:40:18 +01:00
|
|
|
});
|
2021-02-21 00:14:04 +01:00
|
|
|
|
2021-02-25 13:14:34 +01:00
|
|
|
it("Let's start the timer (with a bad address)", async function () {
|
|
|
|
let clock: sinon.default.SinonFakeTimers = sinon.default.useFakeTimers();
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
// classic build
|
2021-02-14 17:07:23 +01:00
|
|
|
myListener = new Listeners({
|
|
|
|
name: "my-test-service",
|
|
|
|
address: "fake.rss.service",
|
2021-02-25 13:14:34 +01:00
|
|
|
timeloop: 60,
|
2021-02-14 17:37:03 +01:00
|
|
|
customfields: {
|
|
|
|
description: ["media:group", "media:description"],
|
|
|
|
icon: ["media:group", "media:thumbnail"],
|
|
|
|
},
|
2021-02-14 17:07:23 +01:00
|
|
|
});
|
2021-02-25 13:14:34 +01:00
|
|
|
fun_initStub();
|
|
|
|
stubListener.fetchRSS.reset();
|
|
|
|
stubListener.fetchRSS.rejects(
|
|
|
|
new Error("connect ECONNREFUSED 127.0.0.1:80")
|
|
|
|
);
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
//spy
|
2021-02-25 13:14:34 +01:00
|
|
|
let fun_spy: sinon.default.SinonSpy = sinon.default.spy((obj, err) => {
|
|
|
|
expect(obj).to.be.eql(undefined);
|
|
|
|
expect(err).to.not.be.eql(undefined);
|
|
|
|
});
|
2021-02-13 13:40:18 +01:00
|
|
|
|
|
|
|
// start timer
|
2021-02-25 13:14:34 +01:00
|
|
|
stubListener.start(fun_spy);
|
2021-02-13 13:40:18 +01:00
|
|
|
|
2021-02-25 13:14:34 +01:00
|
|
|
// wait and assertion
|
|
|
|
// After 1ms
|
|
|
|
await clock.tickAsync(1);
|
|
|
|
expect(stubListener.fetchRSS).to.have.been.calledOnce;
|
|
|
|
expect(fun_spy).to.have.been.calledOnce;
|
2021-02-13 13:40:18 +01:00
|
|
|
|
2021-02-25 13:14:34 +01:00
|
|
|
// After 60s
|
|
|
|
await clock.tickAsync(59999);
|
|
|
|
expect(stubListener.fetchRSS).to.have.been.calledTwice;
|
|
|
|
expect(fun_spy).to.have.been.calledTwice;
|
|
|
|
|
|
|
|
stubListener.stop();
|
2021-02-07 17:50:54 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|