repo: reorder .github
This commit is contained in:
82
.github/examples.mjs
vendored
82
.github/examples.mjs
vendored
@@ -1,82 +0,0 @@
|
|||||||
//Imports
|
|
||||||
import fs from "fs/promises"
|
|
||||||
import fss from "fs"
|
|
||||||
import paths from "path"
|
|
||||||
import url from "url"
|
|
||||||
import metadata from "../source/app/metrics/metadata.mjs"
|
|
||||||
import yaml from "js-yaml"
|
|
||||||
|
|
||||||
//Paths
|
|
||||||
const __metrics = paths.join(paths.dirname(url.fileURLToPath(import.meta.url)), "..")
|
|
||||||
const __templates = paths.join(paths.join(__metrics, "source/templates/"))
|
|
||||||
const __plugins = paths.join(paths.join(__metrics, "source/plugins/"))
|
|
||||||
const __test_plugins = paths.join(paths.join(__metrics, "tests/plugins"))
|
|
||||||
|
|
||||||
//Load plugins metadata
|
|
||||||
const {plugins, templates} = await metadata({log:false, diff:true})
|
|
||||||
|
|
||||||
async function plugin(id) {
|
|
||||||
const path = paths.join(__plugins, id)
|
|
||||||
const readme = paths.join(path, "README.md")
|
|
||||||
const examples = paths.join(path, "examples.yml")
|
|
||||||
const tests = paths.join(__test_plugins, `${id}.yml`)
|
|
||||||
return {
|
|
||||||
readme:{
|
|
||||||
path:readme,
|
|
||||||
content:`${await fs.readFile(readme)}`
|
|
||||||
},
|
|
||||||
tests:{
|
|
||||||
path:tests
|
|
||||||
},
|
|
||||||
examples:fss.existsSync(examples) ? yaml.load(await fs.readFile(examples), "utf8") ?? [] : [],
|
|
||||||
options:plugins[id].readme.table
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const secrets = {
|
|
||||||
$regex:/\$\{\{\s*secrets\.(?<secret>\w+)\s*\}\}/,
|
|
||||||
METRICS_TOKEN:"MOCKED_TOKEN",
|
|
||||||
METRICS_BOT_TOKEN:"MOCKED_TOKEN",
|
|
||||||
PAGESPEED_TOKEN:"MOCKED_TOKEN",
|
|
||||||
SPOTIFY_TOKENS:"MOCKED_CLIENT_ID, MOCKED_CLIENT_SECRET, MOCKED_REFRESH_TOKEN",
|
|
||||||
YOUTUBE_MUSIC_TOKENS:"SAPISID=MOCKED_COOKIE; OTHER_PARAM=OTHER_VALUE;",
|
|
||||||
LASTFM_TOKEN:"MOCKED_TOKEN",
|
|
||||||
NIGHTSCOUT_URL:"https://testapp.herokuapp.com/",
|
|
||||||
WAKATIME_TOKEN:"MOCKED_TOKEN",
|
|
||||||
WAKATIME_TOKEN_NO_PROJECTS:"MOCKED_TOKEN_NO_PROJECTS",
|
|
||||||
TWITTER_TOKEN:"MOCKED_TOKEN",
|
|
||||||
RAPIDAPI_TOKEN:"MOCKED_TOKEN",
|
|
||||||
}
|
|
||||||
|
|
||||||
//Plugins
|
|
||||||
for (const id of Object.keys(plugins)) {
|
|
||||||
const {examples, options, readme, tests} = await plugin(id)
|
|
||||||
|
|
||||||
//Plugin readme
|
|
||||||
await fs.writeFile(readme.path, readme.content
|
|
||||||
.replace(/(<!--examples-->)[\s\S]*(<!--\/examples-->)/g, `$1\n${examples.map(({test, prod, ...step}) => ["```yaml", yaml.dump(step), "```"].join("\n")).join("\n")}\n$2`)
|
|
||||||
.replace(/(<!--options-->)[\s\S]*(<!--\/options-->)/g, `$1\n${options}\n$2`)
|
|
||||||
)
|
|
||||||
//Plugin tests
|
|
||||||
await fs.writeFile(tests.path, yaml.dump(examples.map(({prod, test = {}, name = "", ...step}) => {
|
|
||||||
if (test.skip)
|
|
||||||
return null
|
|
||||||
const result = {name:`${plugins[id].name} - ${name}`, ...step, ...test}
|
|
||||||
test.with ??= {}
|
|
||||||
for (const [k, v] of Object.entries(result.with)) {
|
|
||||||
if (k in test.with)
|
|
||||||
result.with[k] = test.with[k]
|
|
||||||
if (secrets.$regex.test(v))
|
|
||||||
result.with[k] = v.replace(secrets.$regex, secrets[v.match(secrets.$regex)?.groups?.secret])
|
|
||||||
}
|
|
||||||
if (!result.with.base)
|
|
||||||
delete result.with.base
|
|
||||||
delete result.with.filename
|
|
||||||
return result
|
|
||||||
}).filter(t => t)))
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
//Templates
|
|
||||||
|
|
||||||
//Workflow
|
|
||||||
65
.github/index.mjs
vendored
65
.github/index.mjs
vendored
@@ -1,65 +0,0 @@
|
|||||||
//Imports
|
|
||||||
import ejs from "ejs"
|
|
||||||
import fs from "fs/promises"
|
|
||||||
import paths from "path"
|
|
||||||
import url from "url"
|
|
||||||
import sgit from "simple-git"
|
|
||||||
import metadata from "../source/app/metrics/metadata.mjs"
|
|
||||||
|
|
||||||
//Mode
|
|
||||||
const [mode = "dryrun"] = process.argv.slice(2)
|
|
||||||
console.log(`Mode: ${mode}`)
|
|
||||||
|
|
||||||
//Paths
|
|
||||||
const __metrics = paths.join(paths.dirname(url.fileURLToPath(import.meta.url)), "..")
|
|
||||||
const __action = paths.join(__metrics, "source/app/action")
|
|
||||||
const __web = paths.join(__metrics, "source/app/web")
|
|
||||||
const __readme = paths.join(__metrics, ".github/readme")
|
|
||||||
|
|
||||||
//Git setup
|
|
||||||
const git = sgit(__metrics)
|
|
||||||
const staged = new Set()
|
|
||||||
|
|
||||||
for (const step of ["config", "documentation"]) {
|
|
||||||
|
|
||||||
//Load plugins metadata
|
|
||||||
const {plugins, templates, packaged, descriptor} = await metadata({log:false})
|
|
||||||
|
|
||||||
//Update generated files
|
|
||||||
async function update({source, output, options = {}}) {
|
|
||||||
//Regenerate file
|
|
||||||
console.log(`Generating ${output}`)
|
|
||||||
const content = await ejs.renderFile(source, {plugins, templates, packaged, descriptor}, {async:true, ...options})
|
|
||||||
//Save result
|
|
||||||
const file = paths.join(__metrics, output)
|
|
||||||
await fs.writeFile(file, content)
|
|
||||||
//Add to git
|
|
||||||
staged.add(file)
|
|
||||||
}
|
|
||||||
|
|
||||||
//Templating
|
|
||||||
switch (step) {
|
|
||||||
case "config":
|
|
||||||
await update({source:paths.join(__action, "action.yml"), output:"action.yml"})
|
|
||||||
await update({source:paths.join(__web, "settings.example.json"), output:"settings.example.json"})
|
|
||||||
break
|
|
||||||
case "documentation":
|
|
||||||
await update({source:paths.join(__readme, "README.md"), output:"README.md", options:{root:__readme}})
|
|
||||||
await update({source:paths.join(__readme, "partials/documentation/plugins.md"), output:"source/plugins/README.md"})
|
|
||||||
await update({source:paths.join(__readme, "partials/documentation/templates.md"), output:"source/templates/README.md"})
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//Commit and push
|
|
||||||
if (mode === "publish") {
|
|
||||||
console.log(`Pushing staged changes: \n${[...staged].map(file => ` - ${file}`).join("\n")}`)
|
|
||||||
const gitted = await git
|
|
||||||
.addConfig("user.name", "github-actions[bot]")
|
|
||||||
.addConfig("user.email", "41898282+github-actions[bot]@users.noreply.github.com")
|
|
||||||
.add([...staged])
|
|
||||||
.commit("ci: auto-regenerate files")
|
|
||||||
.push("origin", "master")
|
|
||||||
console.log(gitted)
|
|
||||||
}
|
|
||||||
console.log("Success!")
|
|
||||||
130
.github/scripts/build.mjs
vendored
Normal file
130
.github/scripts/build.mjs
vendored
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
//Imports
|
||||||
|
import ejs from "ejs"
|
||||||
|
import fs from "fs/promises"
|
||||||
|
import fss from "fs"
|
||||||
|
import paths from "path"
|
||||||
|
import url from "url"
|
||||||
|
import sgit from "simple-git"
|
||||||
|
import metadata from "../source/app/metrics/metadata.mjs"
|
||||||
|
import yaml from "js-yaml"
|
||||||
|
|
||||||
|
//Mode
|
||||||
|
const [mode = "dryrun"] = process.argv.slice(2)
|
||||||
|
console.log(`Mode: ${mode}`)
|
||||||
|
|
||||||
|
//Paths
|
||||||
|
const __metrics = paths.join(paths.dirname(url.fileURLToPath(import.meta.url)), "..")
|
||||||
|
const __action = paths.join(__metrics, "source/app/action")
|
||||||
|
const __web = paths.join(__metrics, "source/app/web")
|
||||||
|
const __readme = paths.join(__metrics, ".github/readme")
|
||||||
|
const __templates = paths.join(paths.join(__metrics, "source/templates/"))
|
||||||
|
const __plugins = paths.join(paths.join(__metrics, "source/plugins/"))
|
||||||
|
const __test_plugins = paths.join(paths.join(__metrics, "tests/plugins"))
|
||||||
|
const __test_secrets = paths.join(paths.join(__metrics, "tests/secrets.json"))
|
||||||
|
|
||||||
|
//Git setup
|
||||||
|
const git = sgit(__metrics)
|
||||||
|
const staged = new Set()
|
||||||
|
|
||||||
|
//Config and general documentation auto-generation
|
||||||
|
for (const step of ["config", "documentation"]) {
|
||||||
|
|
||||||
|
//Load plugins metadata
|
||||||
|
const {plugins, templates, packaged, descriptor} = await metadata({log:false})
|
||||||
|
|
||||||
|
//Update generated files
|
||||||
|
async function update({source, output, options = {}}) {
|
||||||
|
//Regenerate file
|
||||||
|
console.log(`Generating ${output}`)
|
||||||
|
const content = await ejs.renderFile(source, {plugins, templates, packaged, descriptor}, {async:true, ...options})
|
||||||
|
//Save result
|
||||||
|
const file = paths.join(__metrics, output)
|
||||||
|
await fs.writeFile(file, content)
|
||||||
|
//Add to git
|
||||||
|
staged.add(file)
|
||||||
|
}
|
||||||
|
|
||||||
|
//Templating
|
||||||
|
switch (step) {
|
||||||
|
case "config":
|
||||||
|
await update({source:paths.join(__action, "action.yml"), output:"action.yml"})
|
||||||
|
await update({source:paths.join(__web, "settings.example.json"), output:"settings.example.json"})
|
||||||
|
break
|
||||||
|
case "documentation":
|
||||||
|
await update({source:paths.join(__readme, "README.md"), output:"README.md", options:{root:__readme}})
|
||||||
|
await update({source:paths.join(__readme, "partials/documentation/plugins.md"), output:"source/plugins/README.md"})
|
||||||
|
await update({source:paths.join(__readme, "partials/documentation/templates.md"), output:"source/templates/README.md"})
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
//Load plugins metadata and secrets
|
||||||
|
const {plugins, templates} = await metadata({log:false, diff:true})
|
||||||
|
const secrets = Object.assign(JSON.parse(`${await fs.readFile(__test_secrets)}`), {$regex:/\$\{\{\s*secrets\.(?<secret>\w+)\s*\}\}/})
|
||||||
|
|
||||||
|
//Get plugin infos
|
||||||
|
async function plugin(id) {
|
||||||
|
const path = paths.join(__plugins, id)
|
||||||
|
const readme = paths.join(path, "README.md")
|
||||||
|
const examples = paths.join(path, "examples.yml")
|
||||||
|
const tests = paths.join(__test_plugins, `${id}.yml`)
|
||||||
|
return {
|
||||||
|
readme:{
|
||||||
|
path:readme,
|
||||||
|
content:`${await fs.readFile(readme)}`
|
||||||
|
},
|
||||||
|
tests:{
|
||||||
|
path:tests
|
||||||
|
},
|
||||||
|
examples:fss.existsSync(examples) ? yaml.load(await fs.readFile(examples), "utf8") ?? [] : [],
|
||||||
|
options:plugins[id].readme.table
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Plugins
|
||||||
|
for (const id of Object.keys(plugins)) {
|
||||||
|
const {examples, options, readme, tests} = await plugin(id)
|
||||||
|
|
||||||
|
//Plugin readme
|
||||||
|
await fs.writeFile(readme.path, readme.content
|
||||||
|
.replace(/(<!--examples-->)[\s\S]*(<!--\/examples-->)/g, `$1\n${examples.map(({test, prod, ...step}) => ["```yaml", yaml.dump(step), "```"].join("\n")).join("\n")}\n$2`)
|
||||||
|
.replace(/(<!--options-->)[\s\S]*(<!--\/options-->)/g, `$1\n${options}\n$2`)
|
||||||
|
)
|
||||||
|
console.log(`Generating ${readme.path}`)
|
||||||
|
|
||||||
|
//Plugin tests
|
||||||
|
await fs.writeFile(tests.path, yaml.dump(examples.map(({prod, test = {}, name = "", ...step}) => {
|
||||||
|
if (test.skip)
|
||||||
|
return null
|
||||||
|
const result = {name:`${plugins[id].name} - ${name}`, ...step, ...test}
|
||||||
|
test.with ??= {}
|
||||||
|
for (const [k, v] of Object.entries(result.with)) {
|
||||||
|
if (k in test.with)
|
||||||
|
result.with[k] = test.with[k]
|
||||||
|
if (secrets.$regex.test(v))
|
||||||
|
result.with[k] = v.replace(secrets.$regex, secrets[v.match(secrets.$regex)?.groups?.secret])
|
||||||
|
}
|
||||||
|
if (!result.with.base)
|
||||||
|
delete result.with.base
|
||||||
|
delete result.with.filename
|
||||||
|
return result
|
||||||
|
}).filter(t => t)))
|
||||||
|
console.log(`Generating ${tests.path}`)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
//Commit and push
|
||||||
|
if (mode === "publish") {
|
||||||
|
console.log(`Pushing staged changes: \n${[...staged].map(file => ` - ${file}`).join("\n")}`)
|
||||||
|
const gitted = await git
|
||||||
|
.addConfig("user.name", "github-actions[bot]")
|
||||||
|
.addConfig("user.email", "41898282+github-actions[bot]@users.noreply.github.com")
|
||||||
|
.add([...staged])
|
||||||
|
.commit("ci: auto-regenerate files")
|
||||||
|
.push("origin", "master")
|
||||||
|
console.log(gitted)
|
||||||
|
}
|
||||||
|
console.log("Success!")
|
||||||
@@ -8,8 +8,8 @@ import ejs from "ejs"
|
|||||||
const [mode, name] = process.argv.slice(2)
|
const [mode, name] = process.argv.slice(2)
|
||||||
|
|
||||||
//Paths
|
//Paths
|
||||||
const __metrics = paths.join(paths.dirname(url.fileURLToPath(import.meta.url)), "../..")
|
const __metrics = paths.join(paths.dirname(url.fileURLToPath(import.meta.url)), "../../..")
|
||||||
const __quickstart = paths.join(__metrics, ".github/quickstart")
|
const __quickstart = paths.join(__metrics, ".github/scripts/quickstart")
|
||||||
|
|
||||||
//Check arguments
|
//Check arguments
|
||||||
if ((!mode)||(!name))
|
if ((!mode)||(!name))
|
||||||
@@ -1,5 +1,7 @@
|
|||||||
- name: <%= `${name.charAt(0).toLocaleUpperCase()}${name.substring(1)}` %> plugin (default)
|
- name: <%= `${name.charAt(0).toLocaleUpperCase()}${name.substring(1)}` %> plugin (default)
|
||||||
uses: lowlighter/metrics@latest
|
uses: lowlighter/metrics@latest
|
||||||
with:
|
with:
|
||||||
token: MOCKED_TOKEN
|
filename: metrics.plugin.<%= name %>.svg
|
||||||
plugin_<%= name %>: yes
|
token: ${{ secrets.METRICS_TOKEN }}
|
||||||
|
base: ""
|
||||||
|
plugin_<%= name %>: yes
|
||||||
|
Before Width: | Height: | Size: 562 B After Width: | Height: | Size: 562 B |
@@ -5,7 +5,7 @@ import url from "url"
|
|||||||
import sgit from "simple-git"
|
import sgit from "simple-git"
|
||||||
|
|
||||||
//Git setup
|
//Git setup
|
||||||
const __metrics = paths.join(paths.dirname(url.fileURLToPath(import.meta.url)), "..")
|
const __metrics = paths.join(paths.dirname(url.fileURLToPath(import.meta.url)), "../..")
|
||||||
const git = sgit(__metrics)
|
const git = sgit(__metrics)
|
||||||
|
|
||||||
//Setup octokit
|
//Setup octokit
|
||||||
4
.github/workflows/workflow.yml
vendored
4
.github/workflows/workflow.yml
vendored
@@ -115,7 +115,7 @@ jobs:
|
|||||||
- name: Setup metrics
|
- name: Setup metrics
|
||||||
run: npm ci
|
run: npm ci
|
||||||
- name: Publish rebuild metrics indexes
|
- name: Publish rebuild metrics indexes
|
||||||
run: npm run index -- publish
|
run: npm run build -- publish
|
||||||
|
|
||||||
# Rebase main branch on master
|
# Rebase main branch on master
|
||||||
update-main:
|
update-main:
|
||||||
@@ -269,7 +269,7 @@ jobs:
|
|||||||
- name: Setup metrics
|
- name: Setup metrics
|
||||||
run: npm ci
|
run: npm ci
|
||||||
- name: Publish release
|
- name: Publish release
|
||||||
run: node .github/release.mjs
|
run: node .github/scripts/release.mjs
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ github.token }}
|
GITHUB_TOKEN: ${{ github.token }}
|
||||||
GITHUB_REPOSITORY: ${{ github.repository }}
|
GITHUB_REPOSITORY: ${{ github.repository }}
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ RUN chmod +x /metrics/source/app/action/index.mjs \
|
|||||||
&& rm -rf /var/lib/apt/lists/* \
|
&& rm -rf /var/lib/apt/lists/* \
|
||||||
# Install node modules and rebuild indexes
|
# Install node modules and rebuild indexes
|
||||||
&& npm ci \
|
&& npm ci \
|
||||||
&& npm run index
|
&& npm run build
|
||||||
|
|
||||||
# Environment variables
|
# Environment variables
|
||||||
ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD true
|
ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD true
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "node source/app/web/index.mjs",
|
"start": "node source/app/web/index.mjs",
|
||||||
"test": "jest --runInBand",
|
"test": "jest --runInBand",
|
||||||
"index": "node .github/index.mjs",
|
"build": "node .github/scripts/build.mjs",
|
||||||
"quickstart": "node .github/quickstart/index.mjs",
|
"quickstart": "node .github/scripts/quickstart/index.mjs",
|
||||||
"preview": "node .github/preview.mjs",
|
"preview": "node .github/scripts/preview.mjs",
|
||||||
"linter": "eslint source/**/*.mjs --quiet",
|
"linter": "eslint source/**/*.mjs --quiet",
|
||||||
"format": "eslint source/**/*.mjs --fix",
|
"format": "eslint source/**/*.mjs --fix",
|
||||||
"dev": "nodemon source/app/web/index.mjs -e mjs,css,ejs,json",
|
"dev": "nodemon source/app/web/index.mjs -e mjs,css,ejs,json",
|
||||||
|
|||||||
@@ -27,11 +27,8 @@ describe("Check files editions (checkout your files if needed)", () => {
|
|||||||
".github/readme/partials/documentation/compatibility.md",
|
".github/readme/partials/documentation/compatibility.md",
|
||||||
".github/readme/partials/introduction.md",
|
".github/readme/partials/introduction.md",
|
||||||
".github/workflows/*",
|
".github/workflows/*",
|
||||||
|
".github/scripts/*",
|
||||||
".github/FUNDING.yml",
|
".github/FUNDING.yml",
|
||||||
".github/examples.mjs",
|
|
||||||
".github/index.mjs",
|
|
||||||
".github/release.mjs",
|
|
||||||
".github/markdown_example.mjs",
|
|
||||||
".github/architecture.svg",
|
".github/architecture.svg",
|
||||||
"LICENSE",
|
"LICENSE",
|
||||||
"ARCHITECTURE.md",
|
"ARCHITECTURE.md",
|
||||||
|
|||||||
13
tests/secrets.json
Normal file
13
tests/secrets.json
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
{
|
||||||
|
"METRICS_TOKEN":"MOCKED_TOKEN",
|
||||||
|
"METRICS_BOT_TOKEN":"MOCKED_TOKEN",
|
||||||
|
"PAGESPEED_TOKEN":"MOCKED_TOKEN",
|
||||||
|
"SPOTIFY_TOKENS":"MOCKED_CLIENT_ID, MOCKED_CLIENT_SECRET, MOCKED_REFRESH_TOKEN",
|
||||||
|
"YOUTUBE_MUSIC_TOKENS":"SAPISID=MOCKED_COOKIE; OTHER_PARAM=OTHER_VALUE;",
|
||||||
|
"LASTFM_TOKEN":"MOCKED_TOKEN",
|
||||||
|
"NIGHTSCOUT_URL":"https://testapp.herokuapp.com/",
|
||||||
|
"WAKATIME_TOKEN":"MOCKED_TOKEN",
|
||||||
|
"WAKATIME_TOKEN_NO_PROJECTS":"MOCKED_TOKEN_NO_PROJECTS",
|
||||||
|
"TWITTER_TOKEN":"MOCKED_TOKEN",
|
||||||
|
"RAPIDAPI_TOKEN":"MOCKED_TOKEN"
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user