pull/3/head
Zef Hemel 2022-04-29 13:37:31 +02:00
parent 9ef30d1f49
commit a4e127a6dd
7 changed files with 156 additions and 38 deletions

2
.gitignore vendored
View File

@ -6,3 +6,5 @@ node_modules
dist dist
generated generated
.yarnrc.yml .yarnrc.yml
*.test.js
*.js.map

5
package-lock.json generated
View File

@ -11,9 +11,6 @@
"workspaces": [ "workspaces": [
"packages/*" "packages/*"
], ],
"dependencies": {
"handlebars": "^4.7.7"
},
"devDependencies": { "devDependencies": {
"@parcel/core": "2.3.2", "@parcel/core": "2.3.2",
"@parcel/packager-raw-url": "2.3.2", "@parcel/packager-raw-url": "2.3.2",
@ -9930,6 +9927,7 @@
"@mattermost/client": "^6.7.0-0", "@mattermost/client": "^6.7.0-0",
"@mattermost/types": "^6.7.0-0", "@mattermost/types": "^6.7.0-0",
"@types/yaml": "^1.9.7", "@types/yaml": "^1.9.7",
"handlebars": "^4.7.7",
"markdown-it": "^12.3.2", "markdown-it": "^12.3.2",
"markdown-it-task-lists": "^2.1.1", "markdown-it-task-lists": "^2.1.1",
"yaml": "^2.0.0" "yaml": "^2.0.0"
@ -11526,6 +11524,7 @@
"@mattermost/types": "^6.7.0-0", "@mattermost/types": "^6.7.0-0",
"@types/markdown-it": "^12.2.3", "@types/markdown-it": "^12.2.3",
"@types/yaml": "^1.9.7", "@types/yaml": "^1.9.7",
"handlebars": "^4.7.7",
"markdown-it": "^12.3.2", "markdown-it": "^12.3.2",
"markdown-it-task-lists": "^2.1.1", "markdown-it-task-lists": "^2.1.1",
"yaml": "^2.0.0" "yaml": "^2.0.0"

View File

@ -30,8 +30,5 @@
}, },
"workspaces": [ "workspaces": [
"packages/*" "packages/*"
], ]
"dependencies": {
"handlebars": "^4.7.7"
}
} }

View File

@ -1,6 +1,10 @@
name: github name: github
functions: functions:
test: queryEvents:
path: ./github.ts:queryEvents path: ./github.ts:queryEvents
events: events:
- query:gh-events - query:gh-events
queryIssues:
path: ./github.ts:queryIssues
events:
- query:gh-issues

View File

@ -1,5 +1,7 @@
import { applyQuery, QueryProviderEvent, renderQuery } from "../query/engine"; import { applyQuery, QueryProviderEvent, renderQuery } from "../query/engine";
import { jsonToMDTable } from "../query/util"; import { readPage } from "@silverbulletmd/plugos-silverbullet-syscall/space";
import { parseMarkdown } from "@silverbulletmd/plugos-silverbullet-syscall/markdown";
import { extractMeta } from "../query/data";
type GithubEvent = { type GithubEvent = {
id: string; id: string;
@ -37,16 +39,53 @@ type ExposedEvent = {
repo: string; repo: string;
}; };
async function listEvents(username: string): Promise<GithubEvent[]> { class GithubApi {
let events = await fetch(`https://api.github.com/users/${username}/events`); constructor(private token?: string) {}
return await events.json();
}
async function listIssues(filter: string): Promise<any[]> { async apiCall(url: string, options: any = {}): Promise<any> {
let issues = await fetch( let res = await fetch(url, {
...options,
headers: {
Authorization: this.token ? `token ${this.token}` : undefined,
},
});
if (res.status !== 200) {
throw new Error(await res.text());
}
return res.json();
}
async listEvents(username: string): Promise<GithubEvent[]> {
return this.apiCall(
`https://api.github.com/users/${username}/events?per_page=100`
);
}
async listIssues(filter: string): Promise<any[]> {
return this.apiCall(
`https://api.github.com/issues?q=${encodeURIComponent(filter)}` `https://api.github.com/issues?q=${encodeURIComponent(filter)}`
); );
return await issues.json(); }
static async fromConfig(): Promise<GithubApi> {
return new GithubApi((await getConfig()).token);
}
}
type GithubConfig = {
token?: string;
};
async function getConfig(): Promise<GithubConfig> {
try {
let { text } = await readPage("github-config");
let parsedContent = await parseMarkdown(text);
let pageMeta = await extractMeta(parsedContent);
return pageMeta as GithubConfig;
} catch (e) {
console.error("No github-config page found, using default config");
return {};
}
} }
function mapEvent(event: GithubEvent): any { function mapEvent(event: GithubEvent): any {
@ -64,21 +103,50 @@ function mapEvent(event: GithubEvent): any {
export async function queryEvents({ export async function queryEvents({
query, query,
}: QueryProviderEvent): Promise<any[]> { }: QueryProviderEvent): Promise<any[]> {
let api = await GithubApi.fromConfig();
let usernameFilter = query.filter.find((f) => f.prop === "username"); let usernameFilter = query.filter.find((f) => f.prop === "username");
if (!usernameFilter) { if (!usernameFilter) {
throw Error("No 'username' filter specified, this is mandatory"); throw Error("No 'username' filter specified, this is mandatory");
} }
let username = usernameFilter.value; let usernames: string[] = [];
let allEvents = (await listEvents(username)).map(mapEvent); if (usernameFilter.op === "=") {
return applyQuery(query, allEvents); usernames = [usernameFilter.value];
} else if (usernameFilter.op === "in") {
usernames = usernameFilter.value;
} else {
throw new Error(`Unsupported operator ${usernameFilter.op}`);
}
let allEvents: GithubEvent[] = [];
for (let eventList of await Promise.all(
usernames.map((username) => api.listEvents(username))
)) {
allEvents.push(...eventList);
}
// console.log("Usernames", usernames, "Event list lenght", allEvents[0]);
return applyQuery(query, allEvents.map(mapEvent));
} }
// export async function queryIssues({ export async function queryIssues({
// query, query,
// }: QueryProviderEvent): Promise<string> { }: QueryProviderEvent): Promise<any[]> {
// let filter = query.filter.find((f) => f.prop === "filter"); let api = await GithubApi.fromConfig();
// if (!filter) { let filter = query.filter.find((f) => f.prop === "filter");
// throw Error("No 'filter' specified, this is mandatory"); if (!filter) {
// } throw Error("No 'filter' specified, this is mandatory");
// let username = filter.value; }
// } let queries: string[] = [];
if (filter.op === "=") {
queries = [filter.value];
} else if (filter.op === "in") {
queries = filter.value;
} else {
throw new Error(`Unsupported operator ${filter.op}`);
}
let allIssues: any[] = [];
for (let issuesList of await Promise.all(
queries.map((query) => api.listIssues(query))
)) {
allIssues.push(...issuesList);
}
return allIssues;
}

View File

@ -39,7 +39,8 @@
"@types/yaml": "^1.9.7", "@types/yaml": "^1.9.7",
"markdown-it": "^12.3.2", "markdown-it": "^12.3.2",
"markdown-it-task-lists": "^2.1.1", "markdown-it-task-lists": "^2.1.1",
"yaml": "^2.0.0" "yaml": "^2.0.0",
"handlebars": "^4.7.7"
}, },
"devDependencies": { "devDependencies": {
"@types/markdown-it": "^12.2.3" "@types/markdown-it": "^12.2.3"

View File

@ -1,4 +1,9 @@
import { addParentPointers, collectNodesMatching, ParseTree, renderToText } from "@silverbulletmd/common/tree"; import {
addParentPointers,
collectNodesMatching,
ParseTree,
renderToText,
} from "@silverbulletmd/common/tree";
export const queryRegex = export const queryRegex =
/(<!--\s*#query\s+(.+?)-->)(.+?)(<!--\s*#end\s*-->)/gs; /(<!--\s*#query\s+(.+?)-->)(.+?)(<!--\s*#end\s*-->)/gs;
@ -44,27 +49,69 @@ export function removeQueries(pt: ParseTree) {
}); });
} }
const maxWidth = 70;
// Nicely format an array of JSON objects as a Markdown table // Nicely format an array of JSON objects as a Markdown table
export function jsonToMDTable( export function jsonToMDTable(
jsonArray: any[], jsonArray: any[],
valueTransformer?: (k: string, v: any) => string | undefined valueTransformer: (k: string, v: any) => string = (k, v) => "" + v
): string { ): string {
let headers = new Set<string>(); let fieldWidths = new Map<string, number>();
for (let entry of jsonArray) { for (let entry of jsonArray) {
for (let k of Object.keys(entry)) { for (let k of Object.keys(entry)) {
headers.add(k); let fieldWidth = fieldWidths.get(k);
if (!fieldWidth) {
fieldWidth = valueTransformer(k, entry[k]).length;
} else {
fieldWidth = Math.max(valueTransformer(k, entry[k]).length, fieldWidth);
}
fieldWidths.set(k, fieldWidth);
} }
} }
let headerList = [...headers];
let fullWidth = 0;
for (let v of fieldWidths.values()) {
fullWidth += v + 1;
}
let headerList = [...fieldWidths.keys()];
let lines = []; let lines = [];
lines.push("|" + headerList.join("|") + "|"); lines.push(
lines.push("|" + headerList.map((title) => "----").join("|") + "|"); "|" +
headerList
.map(
(headerName) =>
headerName +
charPad(" ", fieldWidths.get(headerName)! - headerName.length)
)
.join("|") +
"|"
);
lines.push(
"|" +
headerList
.map((title) => charPad("-", fieldWidths.get(title)!))
.join("|") +
"|"
);
for (const val of jsonArray) { for (const val of jsonArray) {
let el = []; let el = [];
for (let prop of headerList) { for (let prop of headerList) {
el.push(valueTransformer ? valueTransformer(prop, val[prop]) : val[prop]); let s = valueTransformer(prop, val[prop]);
el.push(s + charPad(" ", fieldWidths.get(prop)! - s.length));
} }
lines.push("|" + el.join("|") + "|"); lines.push("|" + el.join("|") + "|");
} }
return lines.join("\n"); return lines.join("\n");
function charPad(ch: string, length: number) {
if (fullWidth > maxWidth && ch === "") {
return "";
} else if (fullWidth > maxWidth && ch === "-") {
return "--";
}
if (length < 1) {
return "";
}
return new Array(length + 1).join(ch);
}
} }