2019-05-26 17:50:19 +02:00
|
|
|
/* eslint-disable no-console */
|
|
|
|
|
2018-11-25 01:52:25 +01:00
|
|
|
const remark = require('remark')
|
2019-11-08 23:00:47 +01:00
|
|
|
const parse = require('remark-parse')
|
2018-11-25 01:52:25 +01:00
|
|
|
const html = require('remark-html')
|
2019-05-26 17:50:19 +02:00
|
|
|
const axios = require('axios')
|
|
|
|
const fs = require('fs')
|
|
|
|
const yaml = require('js-yaml')
|
|
|
|
const reposYaml = yaml.load(fs.readFileSync('./content/repos.yml', 'utf8'))
|
2019-05-26 22:20:16 +02:00
|
|
|
const { performance } = require('perf_hooks')
|
|
|
|
const chalk = require('chalk')
|
2018-11-25 01:52:25 +01:00
|
|
|
|
2018-12-07 11:20:02 +01:00
|
|
|
function truncate(n, useWordBoundary) {
|
|
|
|
if (this.length <= n) {
|
|
|
|
return this
|
|
|
|
}
|
|
|
|
const subString = this.substr(0, n - 1)
|
|
|
|
return (
|
|
|
|
(useWordBoundary
|
|
|
|
? subString.substr(0, subString.lastIndexOf(' '))
|
|
|
|
: subString) + '...'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2019-05-26 22:55:28 +02:00
|
|
|
//
|
|
|
|
// Get GitHub repos
|
|
|
|
//
|
2019-11-10 15:29:23 +01:00
|
|
|
const gitHubConfig = {
|
|
|
|
headers: {
|
|
|
|
'User-Agent': 'kremalicious/portfolio',
|
2019-11-13 18:53:31 +01:00
|
|
|
Authorization: `token ${process.env.GATSBY_GITHUB_TOKEN}`
|
2019-11-10 15:29:23 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-26 17:50:19 +02:00
|
|
|
async function getGithubRepos(data) {
|
2019-10-10 00:40:57 +02:00
|
|
|
let repos = []
|
2019-07-04 09:48:48 +02:00
|
|
|
let holder = {}
|
|
|
|
|
2019-10-10 00:40:57 +02:00
|
|
|
for (let item of data) {
|
|
|
|
const user = item.split('/')[0]
|
|
|
|
const repoName = item.split('/')[1]
|
|
|
|
const repo = await axios.get(
|
|
|
|
`https://api.github.com/repos/${user}/${repoName}`,
|
2019-11-10 15:29:23 +01:00
|
|
|
gitHubConfig
|
2019-10-10 00:40:57 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
holder.name = repo.data.name
|
|
|
|
holder.full_name = repo.data.full_name
|
|
|
|
holder.description = repo.data.description
|
|
|
|
holder.html_url = repo.data.html_url
|
|
|
|
holder.homepage = repo.data.homepage
|
|
|
|
holder.stargazers_count = repo.data.stargazers_count
|
|
|
|
holder.pushed_at = repo.data.pushed_at
|
|
|
|
repos.push(holder)
|
2019-07-04 09:48:48 +02:00
|
|
|
holder = {}
|
|
|
|
}
|
|
|
|
|
2019-10-10 00:40:57 +02:00
|
|
|
// sort by pushed to, newest first
|
|
|
|
repos = repos.sort((a, b) => b.pushed_at.localeCompare(a.pushed_at))
|
|
|
|
|
|
|
|
return repos
|
2019-05-26 17:50:19 +02:00
|
|
|
}
|
|
|
|
|
2019-05-26 22:55:28 +02:00
|
|
|
//
|
|
|
|
// Get GitHub repos once and store for later build stages
|
|
|
|
//
|
2019-05-26 17:50:19 +02:00
|
|
|
let repos
|
|
|
|
|
|
|
|
exports.onPreBootstrap = async () => {
|
2019-05-26 22:20:16 +02:00
|
|
|
const t0 = performance.now()
|
|
|
|
|
2019-05-26 17:50:19 +02:00
|
|
|
try {
|
2019-10-10 00:40:57 +02:00
|
|
|
repos = await getGithubRepos(reposYaml)
|
2019-05-26 22:20:16 +02:00
|
|
|
const t1 = performance.now()
|
|
|
|
const ms = t1 - t0
|
|
|
|
const s = ((ms / 1000) % 60).toFixed(3)
|
|
|
|
console.log(
|
|
|
|
chalk.green('success ') + `getGithubRepos: ${repos.length} repos - ${s} s`
|
|
|
|
)
|
2019-05-26 17:50:19 +02:00
|
|
|
} catch (error) {
|
2019-05-26 22:20:16 +02:00
|
|
|
throw Error(error.message)
|
2019-05-26 17:50:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-26 22:55:28 +02:00
|
|
|
//
|
2020-11-21 14:44:48 +01:00
|
|
|
// Add pageContext
|
2019-05-26 22:55:28 +02:00
|
|
|
//
|
|
|
|
exports.onCreatePage = async ({ page, actions }) => {
|
2020-11-21 14:44:48 +01:00
|
|
|
const { createPage, deletePage } = actions
|
|
|
|
|
|
|
|
// Regex for auto-attaching project images to pages based on slug.
|
|
|
|
// Image file names follow the pattern slug-01.png.
|
|
|
|
// Regex inspiration from https://stackoverflow.com/a/7124976
|
2020-11-22 00:25:43 +01:00
|
|
|
const imageRegex = `/${page.path.replace(/\//g, '')}+?(?=-\\d)/`
|
2020-11-21 14:44:48 +01:00
|
|
|
|
|
|
|
deletePage(page)
|
|
|
|
createPage({
|
|
|
|
...page,
|
|
|
|
context: {
|
|
|
|
...page.context,
|
|
|
|
imageRegex,
|
|
|
|
// Add repos only to front page's context
|
|
|
|
...(page.path === '/' && { repos })
|
|
|
|
}
|
|
|
|
})
|
2019-05-26 22:55:28 +02:00
|
|
|
}
|
|
|
|
|
2018-11-25 01:52:25 +01:00
|
|
|
exports.onCreateNode = ({ node, actions }) => {
|
|
|
|
const { createNodeField } = actions
|
|
|
|
|
|
|
|
// Projects YAML nodes
|
|
|
|
if (node.internal.type === 'ProjectsYaml') {
|
|
|
|
// Add transformed Markdown descriptions
|
|
|
|
const description = node.description
|
|
|
|
const descriptionWithLineBreaks = description.split('\n').join('\n\n')
|
|
|
|
|
|
|
|
let descriptionHtml
|
|
|
|
|
|
|
|
remark()
|
2019-11-08 23:00:47 +01:00
|
|
|
.use(parse, { gfm: true, commonmark: true, pedantic: true })
|
2018-11-25 01:52:25 +01:00
|
|
|
.use(html)
|
|
|
|
.process(descriptionWithLineBreaks, (err, file) => {
|
|
|
|
if (err) throw Error('Could not transform project description')
|
|
|
|
|
|
|
|
descriptionHtml = file.contents
|
|
|
|
return descriptionHtml
|
|
|
|
})
|
|
|
|
|
|
|
|
createNodeField({
|
|
|
|
node,
|
|
|
|
name: 'descriptionHtml',
|
|
|
|
value: descriptionHtml
|
|
|
|
})
|
2018-12-07 11:20:02 +01:00
|
|
|
|
|
|
|
// Create excerpt from description
|
|
|
|
const excerpt = truncate.apply(description, [320, true])
|
|
|
|
|
|
|
|
createNodeField({
|
|
|
|
node,
|
|
|
|
name: 'excerpt',
|
|
|
|
value: excerpt
|
|
|
|
})
|
2018-11-25 01:52:25 +01:00
|
|
|
}
|
|
|
|
}
|