fosscord/scripts/notion/GithubNotionSync.js
2021-08-12 01:27:28 +02:00

225 lines
5.5 KiB
JavaScript

require("missing-native-js-functions");
const fetch = require("node-fetch");
const { Client } = require("@notionhq/client");
const bodyParser = require("body-parser");
const express = require("express");
const app = express();
class GithubNotionSync {
constructor({ github, notion }) {
this.githubAuth = github.token;
this.githubWebhookSecret = github.webhookSecret;
this.notionAuth = notion.token;
this.databaseID = notion.database_id;
this.org = github.organization;
this.notion = new Client({ auth: this.notionAuth });
this.urls = {
base: "https://api.github.com/",
};
}
async init() {
this.allNotionPages = await this.getAllNotionPages();
this.repos = await this.getAllIssueUrls();
app.use(bodyParser({}));
app.post("/github", this.handleWebhook.bind(this));
app.listen(3010, () => {
console.log("Github <-> Notion sync listening on :3010");
});
}
async handleWebhook(req, res) {
const { hook, issue } = req.body;
await this.addItemToDb(GithubNotionSync.convertIssue(issue));
res.sendStatus(200);
}
async execute() {
await this.init();
let issues = 0;
for (let repo of this.repos) {
for (let issue of await this.getAllIssuesPerRepo(repo)) {
await this.addItemToDb(issue);
issues++;
}
}
return issues;
}
/**
* @returns array of urls in the following form:
* https://api.github.com/repos/${this.org}/${repo_name}/issues
*/
async getAllIssueUrls() {
let repos = await fetch(`${this.urls.base}orgs/${this.org}/repos`, {
headers: {
Authorization: `token ${this.githubAuth}`,
"User-Agent": this.org,
},
}).then((r) => r.json());
return repos.map((repo) => `${this.urls.base}repos/${this.org}/${repo.name}/issues`);
}
/**
* @param repoIssueUrl element of array returned by `getAllIssueUrls()`
* @returns array of issues for the given repo in the following json form:
* ```
* {
* url: 'https://api.github.com/repos/fosscord/fosscord-api/issues/78',
* title: '[Route] /guilds/:id/regions',
* body: '- [ ] regions',
* number: 78,
* state: 'open',
* label: 'Route',
* assignee: 'Stylix58'
* }
* ```
*/
async getAllIssuesPerRepo(repoIssueUrl) {
var allIssues = [];
var page = 1;
do {
var issues = await fetch(`${repoIssueUrl}?state=all&direction=asc&per_page=100&page=${page}`, {
headers: {
Authorization: `token ${this.githubAuth}`,
"User-Agent": this.org,
},
}).then((r) => r.json());
issues = issues.filter((x) => !x.pull_request).map(GithubNotionSync.convertIssue);
page++;
allIssues = allIssues.concat(issues);
} while (issues.length);
return allIssues;
}
static convertIssue(x) {
return {
url: x?.html_url,
title: x?.title,
body: x?.body,
number: x?.number,
state: x?.state,
labels: x?.labels,
assignees: x?.assignees,
};
}
/**
* @returns {Promise<import("@notionhq/client/build/src/api-types").Page[]>}
*/
async getAllNotionPages() {
var allPages = [];
var start_cursor;
do {
var pages = await this.notion.databases.query({
database_id: this.databaseID,
page_size: 100,
...(start_cursor && { start_cursor }),
});
start_cursor = pages.next_cursor;
allPages = allPages.concat(pages.results);
} while (pages.has_more);
return allPages;
}
async addItemToDb(issue) {
const options = {
parent: { database_id: this.databaseID },
properties: {
Name: {
title: [
{
text: {
content: issue.title,
},
},
],
},
State: {
select: {
name: issue.state,
},
},
Repo: {
select: {
name: GithubNotionSync.getRepoNameFromUrl(issue.url),
},
},
Url: {
url: issue.url,
},
Number: { number: issue.number },
...(issue.assignees && {
Assignee: {
multi_select: issue.assignees.map((x) => ({ name: x.login })),
},
}),
...(issue.labels && {
Label: {
multi_select: issue.labels.map((x) => ({ name: x.name })),
},
}),
},
children: [
{
object: "block",
type: "paragraph",
paragraph: {
text: [
{
type: "text",
text: {
content: issue.body?.slice(0, 1990) || "",
},
},
],
},
},
],
};
const exists = this.allNotionPages.find(
(x) =>
x.properties.Number.number == issue.number &&
x.properties.Repo.select.name === GithubNotionSync.getRepoNameFromUrl(issue.url)
);
try {
if (exists) {
if (
exists.properties.Name?.title?.[0].plain_text !== issue.title ||
exists.properties.State?.select.name !== issue.state ||
JSON.stringify(issue.labels.map((x) => x.name)) !==
JSON.stringify(exists.properties.Label?.multi_select.map((x) => x.name)) ||
JSON.stringify(issue.assignees.map((x) => x.login)) !==
JSON.stringify(exists.properties.Assignee?.multi_select.map((x) => x.name))
) {
console.log("update existing one");
await this.notion.pages.update({ page_id: exists.id, properties: options.properties });
exists.properties = options.properties;
}
} else {
console.log("adding new one");
const index = this.allNotionPages.push(options) - 1; // directly insert it as they might be inserted twice if the webhook is triggered in very short amount of time
this.allNotionPages[index] = await this.notion.pages.create(options);
}
} catch (error) {
console.log(issue);
console.error(error.body, error);
}
}
static getRepoNameFromUrl(url) {
return url?.match(/fosscord\/(fosscord-)?([\w.-]+)/)[2];
}
}
module.exports = { GithubNotionSync };