feat(plugins/sponsors): fetch more than 100 sponsors/past sponsors and display organizations (#1003) [skip ci]
This commit is contained in:
@@ -9,24 +9,59 @@ export default async function({login, q, imports, data, graphql, queries, accoun
|
|||||||
//Load inputs
|
//Load inputs
|
||||||
const {sections, past} = 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
|
//Query description and goal
|
||||||
console.debug(`metrics/compute/${login}/plugins > sponsors > querying 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 {[account]:{sponsorsListing:{fullDescription, activeGoal}}} = await graphql(queries.sponsors.description({login, account}))
|
||||||
const about = await imports.markdown(fullDescription, {mode:"multiline"})
|
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 goal = activeGoal ? {progress:activeGoal.percentComplete, title:activeGoal.title, description:await imports.markdown(activeGoal.description)} : null
|
||||||
let list = nodes.map(({sponsorEntity:{login, avatarUrl}, tier}) => ({login, avatarUrl, amount:tier?.monthlyPriceInDollars ?? null, past:false}))
|
const count = {active:{total:0, user:0, organization:0}, past:{total:0, user:0, organization:0}}
|
||||||
await Promise.all(list.map(async user => user.avatar = await imports.imgb64(user.avatarUrl)))
|
|
||||||
|
//Query active sponsors
|
||||||
|
let list = []
|
||||||
|
{
|
||||||
|
const fetched = []
|
||||||
|
let cursor = null
|
||||||
|
let pushed = 0
|
||||||
|
do {
|
||||||
|
console.debug(`metrics/compute/${login}/sponsors > retrieving sponsors after ${cursor}`)
|
||||||
|
const {[account]:{sponsorshipsAsMaintainer:{edges, nodes}}} = await graphql(queries.sponsors.active({login, account, after:cursor ? `after: "${cursor}"` : ""}))
|
||||||
|
cursor = edges?.[edges?.length - 1]?.cursor
|
||||||
|
fetched.push(...nodes)
|
||||||
|
pushed = nodes.length
|
||||||
|
console.debug(`metrics/compute/${login}/sponsors > retrieved ${pushed} sponsors after ${cursor}`)
|
||||||
|
} while ((pushed) && (cursor))
|
||||||
|
list.push(...fetched.map(({sponsorEntity:{login, avatarUrl, url:organization = null}, tier}) => ({login, avatarUrl, type:organization ? "organization" : "user", amount:tier?.monthlyPriceInDollars ?? null, past:false})))
|
||||||
|
await Promise.all(list.map(async user => user.avatar = await imports.imgb64(user.avatarUrl)))
|
||||||
|
count.active.total = list.length
|
||||||
|
count.active.user = list.filter(user => user.type === "user").length
|
||||||
|
count.active.organization = list.filter(user => user.type === "organization").length
|
||||||
|
}
|
||||||
|
|
||||||
//Query past sponsors
|
//Query past sponsors
|
||||||
if (past) {
|
if (past) {
|
||||||
console.debug(`metrics/compute/${login}/plugins > sponsors > querying past sponsors`)
|
console.debug(`metrics/compute/${login}/plugins > sponsors > querying past sponsors`)
|
||||||
const active = new Set(list.map(({login}) => login))
|
const active = new Set(list.map(({login}) => login))
|
||||||
const {[account]:{sponsorsActivities:{nodes:events}}} = await graphql(queries.sponsors.all({login, account}))
|
const users = []
|
||||||
const users = events.map(({sponsor:{login, avatarUrl}, sponsorsTier}) => ({login, avatarUrl, amount:sponsorsTier?.monthlyPriceInDollars ?? null, past:true}))
|
{
|
||||||
|
const fetched = []
|
||||||
|
let cursor = null
|
||||||
|
let pushed = 0
|
||||||
|
do {
|
||||||
|
console.debug(`metrics/compute/${login}/sponsors > retrieving sponsors events after ${cursor}`)
|
||||||
|
const {[account]:{sponsorsActivities:{edges, nodes}}} = await graphql(queries.sponsors.all({login, account, after:cursor ? `after: "${cursor}"` : ""}))
|
||||||
|
cursor = edges?.[edges?.length - 1]?.cursor
|
||||||
|
fetched.push(...nodes)
|
||||||
|
pushed = nodes.length
|
||||||
|
console.debug(`metrics/compute/${login}/sponsors > retrieved ${pushed} sponsors events after ${cursor}`)
|
||||||
|
} while ((pushed) && (cursor))
|
||||||
|
users.push(...fetched.map(({sponsor:{login, avatarUrl, url:organization = null}, sponsorsTier}) => ({login, avatarUrl, type:organization ? "organization" : "user", amount:sponsorsTier?.monthlyPriceInDollars ?? null, past:true})))
|
||||||
|
}
|
||||||
for (const user of users) {
|
for (const user of users) {
|
||||||
if (!active.has(user.login)) {
|
if (!active.has(user.login)) {
|
||||||
active.add(user.login)
|
active.add(user.login)
|
||||||
list.push({...user, avatar:await imports.imgb64(user.avatarUrl)})
|
list.push({...user, avatar:await imports.imgb64(user.avatarUrl)})
|
||||||
|
count.past.total++
|
||||||
|
count.past[user.type]++
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,21 +1,20 @@
|
|||||||
query SponsorsDefault {
|
query SponsorsActive {
|
||||||
$account(login: "$login") {
|
$account(login: "$login") {
|
||||||
sponsorsListing {
|
sponsorshipsAsMaintainer($after first: 100) {
|
||||||
fullDescription
|
edges {
|
||||||
activeGoal {
|
cursor
|
||||||
percentComplete
|
|
||||||
title
|
|
||||||
description
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
sponsorshipsAsMaintainer(first: 100) {
|
|
||||||
totalCount
|
|
||||||
nodes {
|
nodes {
|
||||||
sponsorEntity {
|
sponsorEntity {
|
||||||
... on User {
|
... on User {
|
||||||
login
|
login
|
||||||
avatarUrl(size: 36)
|
avatarUrl(size: 36)
|
||||||
}
|
}
|
||||||
|
... on Organization {
|
||||||
|
login
|
||||||
|
avatarUrl(size: 36)
|
||||||
|
url
|
||||||
|
}
|
||||||
}
|
}
|
||||||
tier {
|
tier {
|
||||||
monthlyPriceInDollars
|
monthlyPriceInDollars
|
||||||
@@ -23,4 +22,4 @@ query SponsorsDefault {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,12 +1,20 @@
|
|||||||
query SponsorsAll {
|
query SponsorsAll {
|
||||||
$account(login: "$login") {
|
$account(login: "$login") {
|
||||||
sponsorsActivities(last: 100, period: ALL) {
|
sponsorsActivities($after first: 100, period: ALL) {
|
||||||
|
edges {
|
||||||
|
cursor
|
||||||
|
}
|
||||||
nodes {
|
nodes {
|
||||||
sponsor {
|
sponsor {
|
||||||
... on User {
|
... on User {
|
||||||
avatarUrl
|
avatarUrl
|
||||||
login
|
login
|
||||||
}
|
}
|
||||||
|
... on Organization {
|
||||||
|
login
|
||||||
|
avatarUrl(size: 36)
|
||||||
|
url
|
||||||
|
}
|
||||||
}
|
}
|
||||||
sponsorsTier {
|
sponsorsTier {
|
||||||
monthlyPriceInDollars
|
monthlyPriceInDollars
|
||||||
|
|||||||
12
source/plugins/sponsors/queries/description.graphql
Normal file
12
source/plugins/sponsors/queries/description.graphql
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
query SponsorsDescription {
|
||||||
|
$account(login: "$login") {
|
||||||
|
sponsorsListing {
|
||||||
|
fullDescription
|
||||||
|
activeGoal {
|
||||||
|
percentComplete
|
||||||
|
title
|
||||||
|
description
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -34,14 +34,14 @@
|
|||||||
<% } %>
|
<% } %>
|
||||||
<div class="goal-text">
|
<div class="goal-text">
|
||||||
<span>
|
<span>
|
||||||
<% if (plugins.sponsors.count) { %>
|
<% if (plugins.sponsors.count.active.total) { %>
|
||||||
<%= plugins.sponsors.count %> sponsor<%= plugins.sponsors.count !== 1 ? "s are" : " is" %> funding <%= user.login %>'s work
|
<%= plugins.sponsors.count.active.total %> sponsor<%= plugins.sponsors.count.active.total !== 1 ? "s are" : " is" %> funding <%= user.login %>'s work
|
||||||
<% } %>
|
<% } %>
|
||||||
</span>
|
</span>
|
||||||
<span><%= plugins.sponsors.goal.title %></span>
|
<span><%= plugins.sponsors.goal.title %></span>
|
||||||
</div>
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<% for (const user of plugins.sponsors.list) { %><img class="avatar <%= user.past ? "past" : "" %>" src="<%= user.avatar %>" width="24" height="24" alt="" /><% } %>
|
<% for (const user of plugins.sponsors.list) { %><img class="avatar <%= user.type === "organization" ? "organization" : "" %> <%= user.past ? "past" : "" %>" src="<%= user.avatar %>" width="24" height="24" alt="" /><% } %>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,18 +1,19 @@
|
|||||||
/**Mocked data */
|
/**Mocked data */
|
||||||
export default function({ faker, query, login = faker.internet.userName() }) {
|
export default function({ faker, query, login = faker.internet.userName() }) {
|
||||||
console.debug("metrics/compute/mocks > mocking graphql api result > sponsors/default")
|
console.debug("metrics/compute/mocks > mocking graphql api result > sponsors/default")
|
||||||
return ({
|
return /after: "MOCKED_CURSOR"/m.test(query) ?
|
||||||
|
({
|
||||||
user: {
|
user: {
|
||||||
sponsorsListing: {
|
|
||||||
fullDescription: faker.lorem.sentences(),
|
|
||||||
activeGoal: {
|
|
||||||
percentComplete: faker.datatype.number(100),
|
|
||||||
title: faker.lorem.sentence(),
|
|
||||||
description: faker.lorem.sentence(),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
sponsorshipsAsMaintainer: {
|
sponsorshipsAsMaintainer: {
|
||||||
totalCount: faker.datatype.number(100),
|
edges:[],
|
||||||
|
nodes:[],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
: ({
|
||||||
|
user: {
|
||||||
|
sponsorshipsAsMaintainer: {
|
||||||
|
edges: new Array(10).fill("MOCKED_CURSOR"),
|
||||||
nodes: new Array(10).fill(null).map(_ => ({
|
nodes: new Array(10).fill(null).map(_ => ({
|
||||||
sponsorEntity: {
|
sponsorEntity: {
|
||||||
login: faker.internet.userName(),
|
login: faker.internet.userName(),
|
||||||
@@ -1,9 +1,19 @@
|
|||||||
/**Mocked data */
|
/**Mocked data */
|
||||||
export default function({ faker, query, login = faker.internet.userName() }) {
|
export default function({ faker, query, login = faker.internet.userName() }) {
|
||||||
console.debug("metrics/compute/mocks > mocking graphql api result > sponsors/all")
|
console.debug("metrics/compute/mocks > mocking graphql api result > sponsors/all")
|
||||||
return ({
|
return /after: "MOCKED_CURSOR"/m.test(query) ?
|
||||||
|
({
|
||||||
|
user: {
|
||||||
|
sponsorsActivities: {
|
||||||
|
edges:[],
|
||||||
|
nodes:[],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
: ({
|
||||||
user: {
|
user: {
|
||||||
sponsorsActivities: {
|
sponsorsActivities: {
|
||||||
|
edges: new Array(10).fill("MOCKED_CURSOR"),
|
||||||
nodes: new Array(10).fill(null).map(_ => ({
|
nodes: new Array(10).fill(null).map(_ => ({
|
||||||
sponsor: {
|
sponsor: {
|
||||||
login: faker.internet.userName(),
|
login: faker.internet.userName(),
|
||||||
|
|||||||
16
tests/mocks/api/github/graphql/sponsors.description.mjs
Normal file
16
tests/mocks/api/github/graphql/sponsors.description.mjs
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
/**Mocked data */
|
||||||
|
export default function({ faker, query, login = faker.internet.userName() }) {
|
||||||
|
console.debug("metrics/compute/mocks > mocking graphql api result > sponsors/default")
|
||||||
|
return ({
|
||||||
|
user: {
|
||||||
|
sponsorsListing: {
|
||||||
|
fullDescription: faker.lorem.sentences(),
|
||||||
|
activeGoal: {
|
||||||
|
percentComplete: faker.datatype.number(100),
|
||||||
|
title: faker.lorem.sentence(),
|
||||||
|
description: faker.lorem.sentence(),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user