diff --git a/source/plugins/sponsors/examples.yml b/source/plugins/sponsors/examples.yml index 0c513f07..5a3d6287 100644 --- a/source/plugins/sponsors/examples.yml +++ b/source/plugins/sponsors/examples.yml @@ -6,6 +6,7 @@ base: "" plugin_sponsors: yes plugin_sponsors_sections: goal + plugin_sponsors_past: yes - name: Sponsors introduction uses: lowlighter/metrics@latest diff --git a/source/plugins/sponsors/index.mjs b/source/plugins/sponsors/index.mjs index ffe89402..7ffe74d1 100644 --- a/source/plugins/sponsors/index.mjs +++ b/source/plugins/sponsors/index.mjs @@ -7,16 +7,33 @@ export default async function({login, q, imports, data, graphql, queries, accoun return null //Load inputs - const {sections} = await imports.metadata.plugins.sponsors.inputs({data, account, q}) + const {sections, past} = await imports.metadata.plugins.sponsors.inputs({data, account, q}) //Query sponsors and goal + console.debug(`metrics/compute/${login}/plugins > sponsors > querying sponsors and goal`) const {[account]:{sponsorsListing:{fullDescription, activeGoal}, sponsorshipsAsMaintainer:{nodes, totalCount:count}}} = await graphql(queries.sponsors({login, account})) const about = await imports.markdown(fullDescription, {mode:"multiline"}) const goal = activeGoal ? {progress:activeGoal.percentComplete, title:activeGoal.title, description:await imports.markdown(activeGoal.description)} : null - const list = nodes.map(({sponsorEntity:{login, avatarUrl}, tier}) => ({login, avatarUrl, amount:tier?.monthlyPriceInDollars})) + let list = nodes.map(({sponsorEntity:{login, avatarUrl}, tier}) => ({login, avatarUrl, amount:tier?.monthlyPriceInDollars ?? null, past:false})) await Promise.all(list.map(async user => user.avatar = await imports.imgb64(user.avatarUrl))) + //Query past sponsors + if (past) { + console.debug(`metrics/compute/${login}/plugins > sponsors > querying past sponsors`) + const active = new Set(list.map(({login}) => login)) + const {[account]:{sponsorsActivities:{nodes:events}}} = await graphql(queries.sponsors.all({login, account})) + console.log(JSON.stringify(await graphql(queries.sponsors.all({login, account})), null, 2)) + const users = events.map(({sponsor:{login, avatarUrl}, sponsorsTier}) => ({login, avatarUrl, amount:sponsorsTier?.monthlyPriceInDollars ?? null, past:true})) + for (const user of users) { + if (!active.has(user.login)) { + active.add(user.login) + list.push({...user, avatar:await imports.imgb64(user.avatarUrl)}) + } + } + } + //Results + list = list.sort((a, b) => a.past === b.past ? b.amount - a.amount : a.past - b.past) return {sections, about, list, count, goal} } //Handle errors diff --git a/source/plugins/sponsors/metadata.yml b/source/plugins/sponsors/metadata.yml index 446a22c3..aa3bac3c 100644 --- a/source/plugins/sponsors/metadata.yml +++ b/source/plugins/sponsors/metadata.yml @@ -30,4 +30,12 @@ inputs: example: goal, about values: - goal - - about \ No newline at end of file + - about + + plugin_sponsors_past: + description: | + Display past sponsorships + + This feature requires a token from target account, as past sponsorships are gathered from sponsors activity and is private data. + type: boolean + default: no \ No newline at end of file diff --git a/source/plugins/sponsors/queries/all.graphql b/source/plugins/sponsors/queries/all.graphql new file mode 100644 index 00000000..7d4719eb --- /dev/null +++ b/source/plugins/sponsors/queries/all.graphql @@ -0,0 +1,17 @@ +query SponsorsAll { + $account(login: "$login") { + sponsorsActivities(last: 100, period: ALL) { + nodes { + sponsor { + ... on User { + avatarUrl + login + } + } + sponsorsTier { + monthlyPriceInDollars + } + } + } + } +} \ No newline at end of file diff --git a/source/templates/classic/partials/sponsors.ejs b/source/templates/classic/partials/sponsors.ejs index 415fa791..22f3d7e3 100644 --- a/source/templates/classic/partials/sponsors.ejs +++ b/source/templates/classic/partials/sponsors.ejs @@ -41,7 +41,7 @@ <%= plugins.sponsors.goal.title %>