-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnews.js
160 lines (149 loc) · 4.61 KB
/
news.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
const fs = require("fs");
const RssParser = require("rss-parser");
const fetch = require("node-fetch");
const { DynamoDB } = require("@aws-sdk/client-dynamodb");
const { KMS } = require("@aws-sdk/client-kms");
const RateLimiter = require("limiter").RateLimiter;
const { ArgumentParser } = require("argparse");
const yaml = require("js-yaml");
const SENT_MESSAGES_TABLE_NAME = "news-sent-messages";
const SENT_MESSAGES_TABLE_PARTITION_KEY = "sourceId";
const SENT_MESSAGES_TABLE_SORT_KEY = "guid";
const dynamodb = new DynamoDB();
const kms = new KMS();
const discordLimiter = new RateLimiter(1, 500);
async function sendToDiscord(source, webhookUrl, item) {
const requestOpts = {
body: JSON.stringify({
username: "Halo CE News",
content: `${source.icon} | **${source.title}:** ${item.title} ${item.link}`
}),
method: "post",
headers: {
"Content-Type": "application/json"
}
};
await new Promise((resolve, _reject) => {
discordLimiter.removeTokens(1, resolve);
});
await fetch(webhookUrl, requestOpts);
}
function getSentMessagesForSource(sourceId) {
const requestParams = {
TableName: SENT_MESSAGES_TABLE_NAME,
KeyConditionExpression: `${SENT_MESSAGES_TABLE_PARTITION_KEY} = :sourceId`,
ExpressionAttributeValues: {
":sourceId": {S: sourceId}
}
};
return new Promise((resolve, reject) => {
dynamodb.query(requestParams, (err, data) => {
if (err) {
reject(err);
} else {
resolve(data.Items.map(item => item.guid.S));
}
});
});
}
function setMessageSent(sourceId, guid) {
const requestParams = {
TableName: SENT_MESSAGES_TABLE_NAME,
Item: {
[SENT_MESSAGES_TABLE_PARTITION_KEY]: {S: sourceId},
[SENT_MESSAGES_TABLE_SORT_KEY]: {S: guid}
}
};
return new Promise((resolve, reject) => {
dynamodb.putItem(requestParams, (err) => {
if (err) {
reject(err);
} else {
resolve();
}
});
});
}
async function handleFeed(source, webhookUrl, feed, opts) {
const sentGuids = await getSentMessagesForSource(source.id);
for (const item of feed.items) {
if (!item.guid || !item.title || !item.link) {
console.error(`Malformed item found at ${feed.url}`);
continue;
}
if (sentGuids.includes(item.guid)) {
continue;
}
console.log(`Found news ${item.guid} from ${source.id}: ${item.title}`);
try {
if (!opts.nosend) {
await sendToDiscord(source, webhookUrl, item);
}
if (!opts.nosave) {
await setMessageSent(source.id, item.guid);
}
} catch (err) {
console.error(`Failed to handle ${item.guid} from ${source.id}. Skipping`, err);
}
}
}
async function checkNews(sources, opts) {
console.log(`Checking for news (${Object.entries(opts).map(([k, v]) => `${k}=${v}`).join(",")})`);
const parser = new RssParser();
await Promise.all(sources.sources.map(async (source) => {
try {
console.log(`Checking source ${source.url}`);
const feed = await parser.parseURL(source.url);
await handleFeed(source, sources.webhooks[source.webhook ?? "default"], feed, opts);
} catch (err) {
console.error(`Failed to get feed from ${source.url}`, err);
}
}));
}
async function decryptSecret(ciphertext) {
if (!ciphertext.startsWith("KMS:")) {
return ciphertext;
}
return new Promise((resolve, reject) => {
kms.decrypt({
KeyId: "alias/news-secrets",
CiphertextBlob: Buffer.from(ciphertext.substring(4), "base64"),
}, (err, data) => {
if (err) {
reject(err);
} else {
resolve(new TextDecoder().decode(data.Plaintext));
}
})
});
}
(async () => {
const parser = new ArgumentParser({
description: "Gather news via RSS and submit to Discord",
});
parser.add_argument("--nosend", {
action: "store_true",
help: "Prevents messages from being sent to Discord"
});
parser.add_argument("--nosave", {
action: "store_true",
help: "Prevents messages from being saved to the idempotency table"
});
parser.add_argument("--sources", {
type: "str",
default: "sources.yml",
help: "Path to the news sources YAML file. Defaults to sources.yml"
});
const args = parser.parse_args();
let sources;
try {
sources = yaml.safeLoad(fs.readFileSync(args.sources, "utf8"));
await Promise.all(Object.entries(sources.webhooks).map(async ([key, value]) => {
sources.webhooks[key] = await decryptSecret(value);
}));
} catch (e) {
console.log(`Failed to load YAML file ${args.sources}: ${e.message}`);
process.exit(1);
}
await checkNews(sources, {nosend: args.nosend, nosave: args.nosave});
})();