feat(plugins/sponsors): fetch more than 100 sponsors/past sponsors and display organizations (#1003) [skip ci]

This commit is contained in:
Simon Lecoq
2022-04-23 19:15:17 +02:00
committed by GitHub
parent 683df66ffa
commit e9e73c8916
8 changed files with 113 additions and 32 deletions

View File

@@ -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]++
} }
} }
} }

View File

@@ -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 {
} }
} }
} }
} }

View File

@@ -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

View File

@@ -0,0 +1,12 @@
query SponsorsDescription {
$account(login: "$login") {
sponsorsListing {
fullDescription
activeGoal {
percentComplete
title
description
}
}
}
}

View File

@@ -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>

View File

@@ -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(),

View File

@@ -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(),

View 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(),
},
},
},
})
}