feat(plugins/sponsorships): add plugin (#1358)

This commit is contained in:
Simon Lecoq
2023-01-16 22:28:33 -05:00
committed by GitHub
parent cdf2c03b2a
commit 220deb05e3
13 changed files with 268 additions and 0 deletions

View File

@@ -1,6 +1,7 @@
deno deno
gpgarmor gpgarmor
github github
githubassets
https https
leetcode leetcode
pgn pgn

View File

@@ -458,6 +458,26 @@
}, },
}) })
: null), : null),
//Sponsorships
...(set.plugins.enabled.sponsorships
? ({
sponsorships: {
sections: options["sponsorships.sections"].split(",").map(x => x.trim()),
amount: faker.datatype.number(1000),
list: new Array(2+faker.datatype.number(8)).fill(null).map(_ => ({
login: faker.internet.userName(),
avatar: "",
type: "user",
tier: `$${faker.datatype.number(100) * 10} per month`,
private: false,
past: faker.datatype.boolean(),
})),
size: Number(options["sponsorships.size"]),
image: "",
started: faker.date.recent(),
},
})
: null),
//Languages //Languages
...(set.plugins.enabled.languages ...(set.plugins.enabled.languages
? ({ ? ({

View File

@@ -0,0 +1,12 @@
<!--header-->
<!--/header-->
## ➡️ Available options
<!--options-->
<!--/options-->
## Examples workflows
<!--examples-->
<!--/examples-->

View File

@@ -0,0 +1,7 @@
- name: 💝 GitHub sponsorships
uses: lowlighter/metrics@latest
with:
filename: metrics.plugin.sponsorships.svg
token: ${{ secrets.METRICS_TOKEN }}
base: ""
plugin_sponsorships: yes

View File

@@ -0,0 +1,51 @@
//Setup
export default async function({login, q, imports, data, graphql, queries, account}, {enabled = false, extras = false} = {}) {
//Plugin execution
try {
//Check if plugin is enabled and requirements are met
if ((!q.sponsorships) || (!imports.metadata.plugins.sponsorships.enabled(enabled, {extras})))
return null
//Load inputs
let {sections, size} = await imports.metadata.plugins.sponsorships.inputs({data, account, q})
//Query description and goal
let amount = NaN, image = null, started = null
if (sections.includes("amount")) {
console.debug(`metrics/compute/${login}/plugins > sponsorships > querying total amount spend`)
const {totalSponsorshipAmountAsSponsorInCents, sponsorshipsAsSponsor} = (await graphql(queries.sponsorships({login, account})))[account]
amount = totalSponsorshipAmountAsSponsorInCents/100
image = "https://github.githubassets.com/images/icons/emoji/hearts_around.png"
started = sponsorshipsAsSponsor.nodes[0]?.createdAt ? new Date(sponsorshipsAsSponsor.nodes[0]?.createdAt) : null
}
image = await imports.imgb64(image)
//Query sponsorships
const list = []
if (sections.includes("sponsorships")) {
console.debug(`metrics/compute/${login}/plugins > sponsorships > querying sponsorships`)
{
const fetched = []
let cursor = null
let pushed = 0
do {
console.debug(`metrics/compute/${login}/sponsorships > retrieving sponsorships after ${cursor}`)
const {[account]: {sponsorshipsAsSponsor: {edges, nodes}}} = await graphql(queries.sponsorships.all({login, account, after: cursor ? `after: "${cursor}"` : "", size: Math.round(size * 1.5)}))
cursor = edges?.[edges?.length - 1]?.cursor
fetched.push(...nodes)
pushed = nodes.length
console.debug(`metrics/compute/${login}/sponsorships > retrieved ${pushed} sponsorships events after ${cursor}`)
} while ((pushed) && (cursor))
list.push(...fetched.map(({sponsorable: {login, avatarUrl, url: organization = null}, tier:{name:tier}, privacyLevel:privacy, isActive:active}) => ({login, avatarUrl, type: organization ? "organization" : "user", tier, private: privacy !== "PUBLIC", past:!active})))
}
await Promise.all(list.map(async user => user.avatar = await imports.imgb64(user.avatarUrl)))
}
//Results
return {amount, list, sections, size, image, started}
}
//Handle errors
catch (error) {
throw imports.format.error(error)
}
}

View File

@@ -0,0 +1,41 @@
name: 💝 GitHub Sponsorships
category: github
description: |
This plugin displays sponsorships funded through [GitHub sponsors](https://github.com/sponsors/).
examples:
default: https://github.com/lowlighter/metrics/blob/examples/metrics.plugin.sponsorships.svg
supports:
- user
- organization
scopes:
- read:user
- read:org
inputs:
plugin_sponsorships:
description: |
Enable sponsorships plugin
type: boolean
default: no
plugin_sponsorships_sections:
description: |
Displayed sections
- `amount`: display total amount sponsored
- `sponsorships`: display GitHub sponsorships
type: array
format: comma-separated
default: amount, sponsorships
example: amount, sponsorships
values:
- amount
- sponsorships
plugin_sponsorships_size:
description: |
Profile picture display size
type: number
default: 24
min: 8
max: 64

View File

@@ -0,0 +1,29 @@
query SponsorshipsAll {
$account(login: "$login") {
sponsorshipsAsSponsor($after first: 100, activeOnly: false, orderBy: {field: CREATED_AT, direction: DESC}) {
edges {
cursor
}
nodes {
createdAt
isActive
isOneTimePayment
tier {
name
}
privacyLevel
sponsorable {
... on User {
avatarUrl(size: $size)
login
}
... on Organization {
login
avatarUrl(size: $size)
url
}
}
}
}
}
}

View File

@@ -0,0 +1,10 @@
query SponsorshipsDefault {
$account(login: "$login") {
totalSponsorshipAmountAsSponsorInCents
sponsorshipsAsSponsor(first: 1, activeOnly: false, orderBy: {field: CREATED_AT, direction: ASC}) {
nodes {
createdAt
}
}
}
}

View File

@@ -39,6 +39,7 @@
"code", "code",
"chess", "chess",
"sponsors", "sponsors",
"sponsorships",
"poopmap", "poopmap",
"fortune", "fortune",
"splatoon" "splatoon"

View File

@@ -0,0 +1,40 @@
<% if (plugins.sponsorships) { %>
<section>
<h2 class="field">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16"><path d="M10.586 1C12.268 1 13.5 2.37 13.5 4.25c0 1.745-.996 3.359-2.622 4.831-.166.15-.336.297-.509.438l1.116 5.584a.75.75 0 0 1-.991.852l-2.409-.876a.25.25 0 0 0-.17 0l-2.409.876a.75.75 0 0 1-.991-.852L5.63 9.519a13.78 13.78 0 0 1-.51-.438C3.497 7.609 2.5 5.995 2.5 4.25 2.5 2.37 3.732 1 5.414 1c.963 0 1.843.403 2.474 1.073L8 2.198l.112-.125a3.385 3.385 0 0 1 2.283-1.068L10.586 1Zm-3.621 9.495-.718 3.594 1.155-.42a1.75 1.75 0 0 1 1.028-.051l.168.051 1.154.42-.718-3.592c-.199.13-.37.235-.505.314l-.169.097a.75.75 0 0 1-.72 0 9.54 9.54 0 0 1-.515-.308l-.16-.105ZM10.586 2.5c-.863 0-1.611.58-1.866 1.459-.209.721-1.231.721-1.44 0C7.025 3.08 6.277 2.5 5.414 2.5 4.598 2.5 4 3.165 4 4.25c0 1.23.786 2.504 2.128 3.719.49.443 1.018.846 1.546 1.198l.325.21.076-.047.251-.163a13.341 13.341 0 0 0 1.546-1.198C11.214 6.754 12 5.479 12 4.25c0-1.085-.598-1.75-1.414-1.75Z"></path></svg>
Sponsorships
</h2>
<% if (plugins.sponsorships.error) { %>
<div class="row fill-width">
<section>
<div class="field error">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16"><path fill-rule="evenodd" d="M2.343 13.657A8 8 0 1113.657 2.343 8 8 0 012.343 13.657zM6.03 4.97a.75.75 0 00-1.06 1.06L6.94 8 4.97 9.97a.75.75 0 101.06 1.06L8 9.06l1.97 1.97a.75.75 0 101.06-1.06L9.06 8l1.97-1.97a.75.75 0 10-1.06-1.06L8 6.94 6.03 4.97z"></path></svg>
<%= plugins.sponsorships.error.message %>
</div>
</section>
</div>
<% } else { %>
<% for (const section of plugins.sponsorships.sections) { %>
<div class="row fill-width">
<% if (section === "amount") { %>
<section class="sponsorships sponsors goal">
<img src="<%= plugins.sponsorships.image %>" alt="" />
<div><span class="bold"><%= user.login %></span> has given a total of <span class="bold"><%= new Intl.NumberFormat("en", {style:"currency", currency:"USD"}).format(plugins.sponsorships.amount) %></span> to open source software<% if (plugins.sponsorships.started) { %> since <span class="bold"><%= f.date(plugins.sponsorships.started, { date:true, timeZone:config.timezone?.name}) %></span><% } %>.</div>
</section>
<% } else if (section === "sponsorships") { %>
<section class="sponsors goal">
<div class="goal-text">
<span>
<%= user.login %> helped funding the work of <%= plugins.sponsorships.list.length %> users and organizations.
</span>
</div>
<div class="row">
<% for (const user of plugins.sponsorships.list) { %><img class="avatar <%= user.past ? 'past' : '' %> <%= user.type === "organization" ? "organization" : "" %>" src="<%= user.avatar %>" width="<%= 0.8*plugins.sponsorships.size %>" height="<%= 0.8*plugins.sponsorships.size %>" alt="" /><% } %>
</div>
</section>
<% } %>
</div>
<% } %>
<% } %>
</section>
<% } %>

View File

@@ -1161,6 +1161,17 @@
.sponsors .avatar { .sponsors .avatar {
margin: 2px; margin: 2px;
} }
.sponsorships.goal {
display: flex;
align-items: center;
}
.sponsorships.goal .bold {
font-weight: bold;
}
.sponsorships.goal img {
width: 38px;
margin-right: 8px;
}
/* Stackoverflow */ /* Stackoverflow */
.stackoverflow { .stackoverflow {

View File

@@ -0,0 +1,33 @@
/**Mocked data */
export default function({faker, query, login = faker.internet.userName()}) {
console.debug("metrics/compute/mocks > mocking graphql api result > sponsorships/all")
return /after: "MOCKED_CURSOR"/m.test(query)
? ({
user: {
sponsorshipsAsSponsor: {
edges: [],
nodes: [],
},
},
})
: ({
user: {
sponsorshipsAsSponsor: {
edges: new Array(10).fill("MOCKED_CURSOR"),
nodes: new Array(10).fill(null).map(_ => ({
createdAt: `${faker.date.recent()}`,
isActive: faker.datatype.boolean(),
isOneTimePayment: faker.datatype.boolean(),
tier: {
name: "$X a month"
},
privacyLevel: "PUBLIC",
sponsorable: {
login: faker.internet.userName(),
avatarUrl: null,
},
})),
},
},
})
}

View File

@@ -0,0 +1,12 @@
/**Mocked data */
export default function({faker, query, login = faker.internet.userName()}) {
console.debug("metrics/compute/mocks > mocking graphql api result > sponsorships/default")
return ({
user: {
totalSponsorshipAmountAsSponsorInCents:faker.datatype.number(100000),
sponsorshipsAsSponsor:{
nodes:[{createdAt: `${faker.date.recent()}`}]
}
},
})
}