Clean
parent
9ef30d1f49
commit
a4e127a6dd
|
@ -6,3 +6,5 @@ node_modules
|
|||
dist
|
||||
generated
|
||||
.yarnrc.yml
|
||||
*.test.js
|
||||
*.js.map
|
|
@ -11,9 +11,6 @@
|
|||
"workspaces": [
|
||||
"packages/*"
|
||||
],
|
||||
"dependencies": {
|
||||
"handlebars": "^4.7.7"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@parcel/core": "2.3.2",
|
||||
"@parcel/packager-raw-url": "2.3.2",
|
||||
|
@ -9930,6 +9927,7 @@
|
|||
"@mattermost/client": "^6.7.0-0",
|
||||
"@mattermost/types": "^6.7.0-0",
|
||||
"@types/yaml": "^1.9.7",
|
||||
"handlebars": "^4.7.7",
|
||||
"markdown-it": "^12.3.2",
|
||||
"markdown-it-task-lists": "^2.1.1",
|
||||
"yaml": "^2.0.0"
|
||||
|
@ -11526,6 +11524,7 @@
|
|||
"@mattermost/types": "^6.7.0-0",
|
||||
"@types/markdown-it": "^12.2.3",
|
||||
"@types/yaml": "^1.9.7",
|
||||
"handlebars": "^4.7.7",
|
||||
"markdown-it": "^12.3.2",
|
||||
"markdown-it-task-lists": "^2.1.1",
|
||||
"yaml": "^2.0.0"
|
||||
|
|
|
@ -30,8 +30,5 @@
|
|||
},
|
||||
"workspaces": [
|
||||
"packages/*"
|
||||
],
|
||||
"dependencies": {
|
||||
"handlebars": "^4.7.7"
|
||||
}
|
||||
]
|
||||
}
|
||||
|
|
|
@ -1,6 +1,10 @@
|
|||
name: github
|
||||
functions:
|
||||
test:
|
||||
queryEvents:
|
||||
path: ./github.ts:queryEvents
|
||||
events:
|
||||
- query:gh-events
|
||||
queryIssues:
|
||||
path: ./github.ts:queryIssues
|
||||
events:
|
||||
- query:gh-issues
|
||||
|
|
|
@ -1,5 +1,7 @@
|
|||
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 = {
|
||||
id: string;
|
||||
|
@ -37,16 +39,53 @@ type ExposedEvent = {
|
|||
repo: string;
|
||||
};
|
||||
|
||||
async function listEvents(username: string): Promise<GithubEvent[]> {
|
||||
let events = await fetch(`https://api.github.com/users/${username}/events`);
|
||||
return await events.json();
|
||||
class GithubApi {
|
||||
constructor(private token?: string) {}
|
||||
|
||||
async apiCall(url: string, options: any = {}): Promise<any> {
|
||||
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)}`
|
||||
);
|
||||
}
|
||||
|
||||
static async fromConfig(): Promise<GithubApi> {
|
||||
return new GithubApi((await getConfig()).token);
|
||||
}
|
||||
}
|
||||
|
||||
async function listIssues(filter: string): Promise<any[]> {
|
||||
let issues = await fetch(
|
||||
`https://api.github.com/issues?q=${encodeURIComponent(filter)}`
|
||||
);
|
||||
return await issues.json();
|
||||
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 {
|
||||
|
@ -64,21 +103,50 @@ function mapEvent(event: GithubEvent): any {
|
|||
export async function queryEvents({
|
||||
query,
|
||||
}: QueryProviderEvent): Promise<any[]> {
|
||||
let api = await GithubApi.fromConfig();
|
||||
let usernameFilter = query.filter.find((f) => f.prop === "username");
|
||||
if (!usernameFilter) {
|
||||
throw Error("No 'username' filter specified, this is mandatory");
|
||||
}
|
||||
let username = usernameFilter.value;
|
||||
let allEvents = (await listEvents(username)).map(mapEvent);
|
||||
return applyQuery(query, allEvents);
|
||||
let usernames: string[] = [];
|
||||
if (usernameFilter.op === "=") {
|
||||
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({
|
||||
// query,
|
||||
// }: QueryProviderEvent): Promise<string> {
|
||||
// let filter = query.filter.find((f) => f.prop === "filter");
|
||||
// if (!filter) {
|
||||
// throw Error("No 'filter' specified, this is mandatory");
|
||||
// }
|
||||
// let username = filter.value;
|
||||
// }
|
||||
export async function queryIssues({
|
||||
query,
|
||||
}: QueryProviderEvent): Promise<any[]> {
|
||||
let api = await GithubApi.fromConfig();
|
||||
let filter = query.filter.find((f) => f.prop === "filter");
|
||||
if (!filter) {
|
||||
throw Error("No 'filter' specified, this is mandatory");
|
||||
}
|
||||
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;
|
||||
}
|
||||
|
|
|
@ -39,7 +39,8 @@
|
|||
"@types/yaml": "^1.9.7",
|
||||
"markdown-it": "^12.3.2",
|
||||
"markdown-it-task-lists": "^2.1.1",
|
||||
"yaml": "^2.0.0"
|
||||
"yaml": "^2.0.0",
|
||||
"handlebars": "^4.7.7"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/markdown-it": "^12.2.3"
|
||||
|
|
|
@ -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 =
|
||||
/(<!--\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
|
||||
export function jsonToMDTable(
|
||||
jsonArray: any[],
|
||||
valueTransformer?: (k: string, v: any) => string | undefined
|
||||
valueTransformer: (k: string, v: any) => string = (k, v) => "" + v
|
||||
): string {
|
||||
let headers = new Set<string>();
|
||||
let fieldWidths = new Map<string, number>();
|
||||
for (let entry of jsonArray) {
|
||||
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 = [];
|
||||
lines.push("|" + headerList.join("|") + "|");
|
||||
lines.push("|" + headerList.map((title) => "----").join("|") + "|");
|
||||
lines.push(
|
||||
"|" +
|
||||
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) {
|
||||
let el = [];
|
||||
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("|") + "|");
|
||||
}
|
||||
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);
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue