feat: working jobs page with sse

main
Antonio De Lucreziis 11 months ago
parent 28b16aee7a
commit e87a9bef18

@ -1 +1,2 @@
DATA_PATH=./data.local
CONFIG_PATH=config.yaml CONFIG_PATH=config.yaml

25
.gitignore vendored

@ -1,26 +1,21 @@
# build output # Build Output
dist/ dist/
# generated types
.astro/ .astro/
# dependencies # Node JS
node_modules/ node_modules/
*.log*
# logs # Environment Variables
npm-debug.log*
yarn-debug.log*
yarn-error.log*
pnpm-debug.log*
# environment variables
.env .env
.env.production .env.production
# macOS-specific files # Editors
.DS_Store
# editors
.vscode .vscode
# Locals
*.local*
config.yaml config.yaml
# macOS-specific files
.DS_Store

@ -1,35 +1,39 @@
{ {
"name": "phcd", "name": "phcd",
"type": "module", "type": "module",
"version": "0.0.1", "version": "0.0.1",
"scripts": { "scripts": {
"dev": "astro dev", "dev": "astro dev",
"start": "astro dev", "start": "astro dev",
"build": "astro check && astro build", "build": "astro sync && astro check && astro build",
"preview": "astro preview", "preview": "astro preview",
"astro": "astro" "astro": "astro"
}, },
"dependencies": { "dependencies": {
"@astrojs/check": "^0.5.2", "@astrojs/check": "^0.5.6",
"@astrojs/node": "^8.2.0", "@astrojs/node": "^8.2.3",
"@astrojs/preact": "^3.1.0", "@astrojs/preact": "^3.1.1",
"@fontsource-variable/material-symbols-outlined": "^5.0.22", "@fontsource-variable/material-symbols-outlined": "^5.0.24",
"@fontsource/jetbrains-mono": "^5.0.18", "@fontsource/jetbrains-mono": "^5.0.19",
"@fontsource/lato": "^5.0.18", "@fontsource/lato": "^5.0.19",
"astro": "^4.3.5", "astro": "^4.4.11",
"async-mutex": "^0.4.1", "async-mutex": "^0.4.1",
"dockerode": "^4.0.2", "dockerode": "^4.0.2",
"js-yaml": "^4.1.0", "js-yaml": "^4.1.0",
"lodash": "^4.17.21", "lodash": "^4.17.21",
"nodegit": "^0.27.0", "nodegit": "^0.27.0",
"preact": "^10.19.4", "preact": "^10.19.6",
"typescript": "^5.3.3", "typescript": "^5.3.3",
"zod": "^3.22.4" "zod": "^3.22.4"
}, },
"devDependencies": { "devDependencies": {
"@types/dockerode": "^3.3.23", "@types/babel-generator": "^6.25.8",
"@types/js-yaml": "^4.0.9", "@types/babel__core": "^7.20.5",
"@types/lodash": "^4.14.202", "@types/dockerode": "^3.3.24",
"sass": "^1.70.0" "@types/js-yaml": "^4.0.9",
} "@types/lodash": "^4.14.202",
"@types/node-fetch": "^2.6.11",
"node-fetch": "^3.3.2",
"sass": "^1.71.1"
}
} }

File diff suppressed because it is too large Load Diff

@ -0,0 +1,75 @@
import fetch from 'node-fetch'
const base = process.argv[2] ?? 'https://git.example.org/example/foo'
await fetch('http://localhost:4321/api/webhook', {
method: 'POST',
body: JSON.stringify({
secret: '3gEsCfjlV2ugRwgpU#w1*WaW*wa4NXgGmpCfkbG3',
ref: 'refs/heads/main',
before: '28e1879d029cb852e4844d9c718537df08844e03',
after: 'bffeb74224043ba2feb48d137756c8a9331c449a',
compare_url: `${base}/compare/28e1879d029cb852e4844d9c718537df08844e03...bffeb74224043ba2feb48d137756c8a9331c449a`,
commits: [
{
id: 'bffeb74224043ba2feb48d137756c8a9331c449a',
message: 'Webhooks Yay!',
url: `${base}/commit/bffeb74224043ba2feb48d137756c8a9331c449a`,
author: {
name: 'Gitea',
email: 'someone@gitea.io',
username: 'gitea',
},
committer: {
name: 'Gitea',
email: 'someone@gitea.io',
username: 'gitea',
},
timestamp: '2017-03-13T13:52:11-04:00',
},
],
repository: {
id: 140,
owner: {
id: 1,
login: 'gitea',
full_name: 'Gitea',
email: 'someone@gitea.io',
avatar_url: `${base}/avatars/1`,
username: 'gitea',
},
name: 'foo',
full_name: 'example/foo',
description: 'An example repo',
private: false,
fork: false,
html_url: `${base}`,
ssh_url: 'ssh://gitea@git.example.org.git',
clone_url: `${base}.git`,
website: '',
stars_count: 0,
forks_count: 1,
watchers_count: 1,
open_issues_count: 7,
default_branch: 'master',
created_at: '2017-02-26T04:29:06-05:00',
updated_at: '2017-03-13T13:51:58-04:00',
},
pusher: {
id: 1,
login: 'gitea',
full_name: 'Gitea',
email: 'someone@gitea.io',
avatar_url: `${base}/avatars/1`,
username: 'gitea',
},
sender: {
id: 1,
login: 'gitea',
full_name: 'Gitea',
email: 'someone@gitea.io',
avatar_url: `${base}/avatars/1`,
username: 'gitea',
},
}),
})

@ -1,5 +1,5 @@
import _ from 'lodash' import _ from 'lodash'
import { clsx } from '../utils.js' import { clsx } from '@/lib/utils'
export const Value = ({ value, borderless }) => { export const Value = ({ value, borderless }) => {
return Array.isArray(value) ? ( return Array.isArray(value) ? (

@ -0,0 +1,114 @@
import _ from 'lodash'
import { useEffect, useState } from 'preact/hooks'
import { durationToString } from './lib/utils'
/**
* @param {import('@/jobs.ts').QueuedJob} props
*/
export const QueuedJob = ({ uuid, name, submitter, submittedAt }) => (
<div class="job queued" title={uuid}>
<div class="name">{name}</div>
<div class="footer">
<div class="submitted-at">{new Date(submittedAt).toLocaleString()}</div>
</div>
</div>
)
/**
* @param {import('@/jobs.ts').CompletedJob} props
*/
export const CompletedJob = ({
uuid,
name,
submitter,
submittedAt,
startedAt,
completedAt,
successful,
error,
}) => {
return (
<div class="job completed" title={uuid}>
<div class="name">{name}</div>
<div class="footer">
<div class="submitted-at">{new Date(submittedAt).toLocaleString()}</div>
<div class="delta">{durationToString(startedAt, completedAt)}</div>
</div>
</div>
)
}
export const JobsPage = ({}) => {
const [jobStore, setJobStore] = useState({
queuedJobs: {},
completedJobs: {},
})
useEffect(async () => {
const res = await fetch('/api/jobs')
const { queuedJobs, completedJobs } = await res.json()
const result = {
queuedJobs: {},
completedJobs: {},
}
for (const item of queuedJobs) {
result.queuedJobs[item.uuid] = item
}
for (const item of completedJobs) {
result.completedJobs[item.uuid] = item
}
setJobStore(result)
// Setup SSE
const es = new EventSource('/api/sse')
es.addEventListener('message', ({ data }) => {
const event = JSON.parse(data)
if (event.type === 'added') {
setJobStore(s => ({
...s,
queuedJobs: {
...s.queuedJobs,
[event.job.uuid]: event.job,
},
}))
}
if (event.type === 'completed') {
setJobStore(s => ({
queuedJobs: {
..._.omit(s.queuedJobs, event.job.uuid),
},
completedJobs: {
...s.completedJobs,
[event.job.uuid]: event.job,
},
}))
}
})
}, [])
return (
<>
<h2>Queued Jobs</h2>
<div class="list">
{Object.values(jobStore.queuedJobs)
.toReversed()
.map(queuedJob => (
<QueuedJob {...queuedJob} />
))}
</div>
<h2>Completed Jobs</h2>
<div class="list">
{Object.values(jobStore.completedJobs)
.toReversed()
.map(completedJob => (
<CompletedJob {...completedJob} />
))}
</div>
</>
)
}

@ -0,0 +1,233 @@
function inspectZod(schema, path = []) {
if ('typeName' in schema._def) {
debug(' '.repeat(path.length), path.at(-1) ?? '<root>', '::', schema._def.typeName)
if (schema._def.typeName === 'ZodUnion') {
schema._def.options.forEach(option => inspectZod(option, [...path, '<union>']))
return
}
if (schema._def.typeName === 'ZodObject') {
Object.entries(schema._def.shape()).forEach(([k, v]) => {
inspectZod(v, [...path, k])
})
return
}
if (schema._def.typeName === 'ZodTuple') {
schema._def.items.forEach((item, i) => {
inspectZod(item, [...path, i])
})
return
}
if (schema._def.typeName === 'ZodArray') {
inspectZod(schema._def.type, [...path, '<index>'])
return
}
if (schema._def.typeName === 'ZodOptional') {
inspectZod(schema._def.innerType, [...path, '?'])
return
}
if (schema._def.typeName === 'ZodLiteral') {
debug(' '.repeat(path.length + 1), schema._def)
return
}
if (schema._def.typeName === 'ZodString') {
return
}
if (schema._def.typeName === 'ZodRecord') {
inspectZod(schema._def.keyType, [...path, '<key>'])
inspectZod(schema._def.valueType, [...path, '<value>'])
return
}
debug('_def:', schema._def)
}
}
const ZodField = ({ value, setValue, schema, path }) => {
if (schema._def.typeName === 'ZodString') {
return (
<div>
<label>{path.at(-1)}</label>
<input value={value} onInput={e => setValue(path, e.target.value)} />
</div>
)
}
if (schema._def.typeName === 'ZodNumber') {
return (
<div>
<label>{path.at(-1)}</label>
<input value={value} onInput={e => setValue(path, e.target.value)} />
</div>
)
}
if (schema._def.typeName === 'ZodBoolean') {
return (
<div>
<label>{path.at(-1)}</label>
<input
type="checkbox"
checked={value}
onInput={e => setValue(path, e.target.checked)}
/>
</div>
)
}
if (schema._def.typeName === 'ZodDate') {
return (
<div>
<label>{path.at(-1)}</label>
<input type="date" value={value} onInput={e => setValue(path, e.target.value)} />
</div>
)
}
if (schema._def.typeName === 'ZodObject') {
return (
<div>
{Object.entries(schema._def.shape()).map(([k, v]) => {
return (
<ZodField
value={value[k]}
setValue={setValue}
schema={v}
path={[...path, k]}
/>
)
})}
</div>
)
}
if (schema._def.typeName === 'ZodArray') {
return (
<div>
{value.map((v, i) => {
return (
<ZodField
value={v}
setValue={setValue}
schema={schema._def.type}
path={[...path, i]}
/>
)
})}
<button onClick={() => setValue(path, [...value, ''])}>Add</button>
</div>
)
}
if (schema._def.typeName === 'ZodOptional') {
return (
<div>
<ZodField
value={value}
setValue={setValue}
schema={schema._def.innerType}
path={path}
/>
</div>
)
}
if (schema._def.typeName === 'ZodLiteral') {
return (
<div>
<ZodField value={value} setValue={setValue} schema={schema} path={path} />
</div>
)
}
if (schema._def.typeName === 'ZodUnion') {
return (
<div>
<select value={value} onInput={e => setValue(path, e.target.value)}>
{schema._def.options.map(option => {
return <option value={option._def.value}>{option._def.value}</option>
})}
</select>
</div>
)
}
}
const ZodObject = ({ value, setValue, schema, path }) => {
return (
<div>
{Object.entries(schema._def.shape()).map(([k, v]) => {
return (
<ZodField value={value[k]} setValue={setValue} schema={v} path={[...path, k]} />
)
})}
</div>
)
}
const ZodArray = ({ value, setValue, schema, path }) => {
return (
<div>
{value.map((v, i) => {
return (
<ZodField
value={v}
setValue={setValue}
schema={schema._def.type}
path={[...path, i]}
/>
)
})}
<button onClick={() => setValue(path, [...value, ''])}>Add</button>
</div>
)
}
const ZodOptional = ({ value, setValue, schema, path }) => {
return (
<div>
<ZodField
value={value}
setValue={setValue}
schema={schema._def.innerType}
path={path}
/>
</div>
)
}
const ZodLiteral = ({ value, setValue, schema, path }) => {
return (
<div>
<ZodField value={value} setValue={setValue} schema={schema} path={path} />
</div>
)
}
const ZodUnion = ({ value, setValue, schema, path }) => {
return (
<div>
<select value={value} onInput={e => setValue(path, e.target.value)}>
{schema._def.options.map(option => {
return <option value={option._def.value}>{option._def.value}</option>
})}
</select>
</div>
)
}
export const ZodForm = ({ value, setValue, schema }) => {
return (
<div>
{schema._def.typeName === 'ZodObject' && (
<ZodObject value={value} setValue={setValue} schema={schema} path={[]} />
)}
{schema._def.typeName === 'ZodArray' && (
<ZodArray value={value} setValue={setValue} schema={schema} path={[]} />
)}
{schema._def.typeName === 'ZodOptional' && (
<ZodOptional value={value} setValue={setValue} schema={schema} path={[]} />
)}
{schema._def.typeName === 'ZodLiteral' && (
<ZodLiteral value={value} setValue={setValue} schema={schema} path={[]} />
)}
{schema._def.typeName === 'ZodUnion' && (
<ZodUnion value={value} setValue={setValue} schema={schema} path={[]} />
)}
</div>
)
}

@ -0,0 +1,21 @@
export function durationToString(from, to) {
from = new Date(from)
to = new Date(to)
let s = to.getTime() - from.getTime()
const millis = s % 1000
s = (s - millis) / 1000
if (s === 0) return `${millis}ms`
const seconds = s % 60
s = (s - seconds) / 60
if (s === 0) return `${seconds}s${millis}ms`
const minutes = s % 60
s = (s - minutes) / 60
if (s === 0) return `${minutes}s${seconds}s`
const hours = s
return `${hours}h${minutes}m`
}

@ -5,20 +5,34 @@
<a class="label" href="/"> <span class="material-symbols-outlined">home</span> Home</a> <a class="label" href="/"> <span class="material-symbols-outlined">home</span> Home</a>
</div> </div>
<div class="nav-item group"> <div class="nav-item group">
<a class="label" href="/containers"><span class="material-symbols-outlined">view_list</span> Containers</a> <a class="label" href="/containers"
<div class="children"> ><span class="material-symbols-outlined">view_list</span> Containers</a
>
<!-- <div class="children">
<div class="nav-item"><div class="label">Container 1</div></div> <div class="nav-item"><div class="label">Container 1</div></div>
<div class="nav-item"><div class="label">Container 2</div></div> <div class="nav-item"><div class="label">Container 2</div></div>
<div class="nav-item"><div class="label">Container 3</div></div> <div class="nav-item"><div class="label">Container 3</div></div>
</div> </div> -->
</div> </div>
<div class="nav-item group"> <div class="nav-item group">
<a class="label" href="/deploys"><span class="material-symbols-outlined">deployed_code</span> Deploys</a> <a class="label" href="/deploys"
<div class="children"> ><span class="material-symbols-outlined">deployed_code</span> Deploys</a
>
<!-- <div class="children">
<div class="nav-item"><div class="label">Deploy 1</div></div> <div class="nav-item"><div class="label">Deploy 1</div></div>
<div class="nav-item"><div class="label">Deploy 2</div></div> <div class="nav-item"><div class="label">Deploy 2</div></div>
<div class="nav-item"><div class="label">Deploy 3</div></div> <div class="nav-item"><div class="label">Deploy 3</div></div>
</div> </div> -->
</div>
<div class="nav-item group">
<a class="label" href="/jobs"
><span class="material-symbols-outlined">terminal</span> Jobs</a
>
<!-- <div class="children">
<div class="nav-item"><div class="label">Deploy 1</div></div>
<div class="nav-item"><div class="label">Deploy 2</div></div>
<div class="nav-item"><div class="label">Deploy 3</div></div>
</div> -->
</div> </div>
</nav> </nav>
</div> </div>

@ -1,60 +1,75 @@
import { Schema, ZodType, z } from 'zod'
import yaml from 'js-yaml' import yaml from 'js-yaml'
import { readFile, writeFile } from 'fs/promises' import { readFile, writeFile } from 'fs/promises'
import { Mutex } from 'async-mutex' import { Mutex } from 'async-mutex'
import { debug } from './logger'
export type GitRef = { type: 'default' } | { type: 'branch' | 'tag' | 'commit'; value: string }
export const GitRef = z.union([
export type BaseDeploy = { name: string } z.object({ type: z.literal('default') }),
z.object({
export type BaseGitDeploy = BaseDeploy & { type: z.union([z.literal('branch'), z.literal('branch'), z.literal('branch')]),
url: string value: z.string(),
ref: GitRef }),
} ])
export type DockerDeploy = BaseDeploy & { export const GitDeploy = z.object({
type: 'docker' name: z.string(),
options: { url: z.string(),
image: string ref: GitRef,
name?: string })
volumes?: string[]
ports?: string[] const DockerRunOptions = z.object({
env?: Record<string, string> name: z.string(),
} volumes: z.tuple([z.string(), z.string()]).array(),
} ports: z.tuple([z.string(), z.string()]).array(),
env: z.record(z.string()),
export type DockerfileDeploy = BaseGitDeploy & { })
type: 'dockerfile'
options: { export const DockerDeploy = z.object({
path?: string name: z.string(),
type: z.literal('docker'),
name?: string options: DockerRunOptions.extend({
volumes?: string[] image: z.string(),
ports?: string[] }),
env?: Record<string, string> })
}
} export const DockerfileDeploy = GitDeploy.extend({
type: z.literal('dockerfile'),
export type DockerComposeDeploy = BaseGitDeploy & { options: DockerRunOptions.extend({
type: 'docker-compose' path: z.string().optional(),
options: { }),
path?: string })
}
} export const DockerComposeDeploy = GitDeploy.extend({
type: z.literal('docker-compose'),
export type ShellDeploy = BaseGitDeploy & { options: z.object({
type: 'shell' path: z.string().optional(),
options: { }),
path?: string })
env?: Record<string, string>
} export const ShellDeploy = GitDeploy.extend({
} type: z.literal('shell'),
options: z.object({
export type Deploy = DockerDeploy | DockerfileDeploy | DockerComposeDeploy | ShellDeploy path: z.string().optional(),
env: z.record(z.string()),
export type Config = { }),
deploys: Deploy[] })
}
export const Deploy = z.union([DockerDeploy, DockerfileDeploy, DockerComposeDeploy, ShellDeploy])
export type GitRef = z.infer<typeof GitRef>
export type GitDeploy = z.infer<typeof GitDeploy>
export type DockerDeploy = z.infer<typeof DockerDeploy>
export type DockerfileDeploy = z.infer<typeof DockerfileDeploy>
export type DockerComposeDeploy = z.infer<typeof DockerComposeDeploy>
export type ShellDeploy = z.infer<typeof ShellDeploy>
export type Deploy = z.infer<typeof Deploy>
export type Config = { deploys: Deploy[] }
export const refToString = (ref: GitRef, defaultStr: string = '<default>') => { export const refToString = (ref: GitRef, defaultStr: string = '<default>') => {
return ref.type === 'default' ? defaultStr : ref.value return ref.type === 'default' ? defaultStr : ref.value

@ -0,0 +1,87 @@
import type { Deploy, GitDeploy, GitRef, ShellDeploy } from '@/config'
import type { Job, Worker } from '@/jobs'
import path from 'path'
import { exists, normalizeURL, sleep } from '@/lib/utils'
import { runCommand } from '@/runners'
import { debug } from '@/logger'
const toSafePath = (target: string) => {
return '.' + path.posix.normalize('/' + target)
}
function getDeployDirectory(deploy: GitDeploy): string {
const { url, ref } = deploy
const repoSlug = url
.replace(/(^\w+:|^)\/\//, '') // strip protocol
.replace(/[^a-zA-Z]+/g, '-') // only keep letters, other symbols become dashes
.replace(/^\-|\-$/g, '') // remove leading or trailing dashes
const slug =
ref.type === 'default'
? `${deploy.name}_${repoSlug}`
: `${deploy.name}_${repoSlug}@${ref.value}`
return `${import.meta.env.CLONE_PATH ?? `${import.meta.env.DATA_PATH}/clone`}/${slug}`
}
async function cloneOrUpdateRepo(deploy: Deploy & { url: string; ref: GitRef }) {
const repoDir = getDeployDirectory(deploy)
if (await exists(repoDir)) {
await runCommand(`git -C "${repoDir}" pull`)
} else {
await runCommand(`mkdir -p "${repoDir}"`)
await runCommand(`git clone "${normalizeURL(deploy.url)}" "${repoDir}"`)
}
if (deploy.ref.type !== 'default') {
await runCommand(`git -C "${repoDir}" checkout "${deploy.ref.value}"`)
}
}
export async function shellRunner(deploy: ShellDeploy) {
const { path, env } = deploy.options
const repoDir = getDeployDirectory(deploy)
await cloneOrUpdateRepo(deploy)
const script = [
// mode to correct directory
`cd ${repoDir}`,
// append env variables
Object.entries(env ?? {})
.map(([key, value]) => `export ${key}="${value.replace(/"/g, '\\"')}"`)
.join('\n'),
// launch program
toSafePath(path ?? './deploy.sh'),
].join('\n\n')
await runCommand(script)
}
export function createDeployJob(deploy: Deploy, submitter: any): Job & Worker {
return {
name: deploy.name,
submitter,
submittedAt: new Date(),
async work() {
debug('[Runner]', `Deploying "${deploy.name}"`)
await sleep(1000)
// TODO: Add other deploy types
if (deploy.type === 'shell') await shellRunner(deploy)
else {
throw new Error(`deploy type "${deploy.type}" not yet implemented`)
}
debug('[Runner]', 'Finished deploy')
},
}
}

@ -1,5 +1,15 @@
import type { Deploy, GitRef } from './config' import type { Deploy, GitRef } from './config'
function splitBinding(pair: string): [string, string] {
const parts = pair.trim().split(':')
if (parts.length !== 2) {
throw new Error(`invalid binding format`)
}
const [external, internal] = parts
return [external, internal]
}
function cutString(s: string, sep: string): [string, string] { function cutString(s: string, sep: string): [string, string] {
return [s.slice(0, s.indexOf(sep)), s.slice(s.indexOf(sep) + sep.length)] return [s.slice(0, s.indexOf(sep)), s.slice(s.indexOf(sep) + sep.length)]
} }
@ -38,8 +48,8 @@ export function parseDeploy(form: Record<string, any>): Deploy {
image, image,
name: containerName, name: containerName,
env: Object.fromEntries(env.split(/\n/g).map(line => cutString(line.trim(), '='))), env: Object.fromEntries(env.split(/\n/g).map(line => cutString(line.trim(), '='))),
ports: ports.split(/\n/g).map(line => line.trim()), ports: ports.split(/\n/g).map(splitBinding),
volumes: volumes.split(/\n/g).map(line => line.trim()), volumes: volumes.split(/\n/g).map(splitBinding),
}, },
} }
} }
@ -67,6 +77,7 @@ export function parseDeploy(form: Record<string, any>): Deploy {
const url = form['deploy-url'] as string const url = form['deploy-url'] as string
const ref = parseRef(form) const ref = parseRef(form)
const containerName = form['deploy-options-name'] as string
const path = form['deploy-options-path'] as string const path = form['deploy-options-path'] as string
const ports = form['deploy-options-ports'] as string const ports = form['deploy-options-ports'] as string
const env = form['deploy-options-env'] as string const env = form['deploy-options-env'] as string
@ -78,10 +89,11 @@ export function parseDeploy(form: Record<string, any>): Deploy {
ref, ref,
type: 'dockerfile', type: 'dockerfile',
options: { options: {
name: containerName,
path, path,
env: Object.fromEntries(env.split(/\n/g).map(line => cutString(line.trim(), '='))), env: Object.fromEntries(env.split(/\n/g).map(line => cutString(line.trim(), '='))),
ports: ports.split(/\n/g).map(line => line.trim()), ports: ports.split(/\n/g).map(splitBinding),
volumes: volumes.split(/\n/g).map(line => line.trim()), volumes: volumes.split(/\n/g).map(splitBinding),
}, },
} }
} }

@ -1,28 +1,148 @@
const queue: (() => Promise<void>)[] = [] import { EventEmitter } from 'events'
// to ensure that the while loop inside triggerProcessQueue is getting executed from only one call at a time import { debug } from './logger'
let working = false
import { createJsonDatabase } from './lib/file-db'
import { randomUUID } from 'crypto'
export type QueuedJob = {
uuid: string
name: string
submitter: any
submittedAt: Date
}
export type CompletedJob = {
uuid: string
name: string
successful: boolean
error?: any
submitter: any
submittedAt: string
startedAt: string
completedAt: string
}
export type Job = {
name: string
export function runPendingJobs() { submitter: any
triggerProcessQueue() submittedAt: Date
} }
async function triggerProcessQueue() { export type Worker = {
work: () => Promise<void>
}
// Event emitter & Queue (runtime only)
const emitter = new EventEmitter<{
'job:add': [QueuedJob]
'job:completed': [CompletedJob]
}>()
const queue: (QueuedJob & Worker)[] = []
// Job db for logging purposes
const jobsDB = createJsonDatabase<CompletedJob[]>(`${import.meta.env.DATA_PATH}/jobs.json`, [])
// To ensure that the while loop inside "processQueue" is getting executed from
// only one call at a time.
let working = false
async function processQueue() {
if (working) return if (working) return
working = true working = true
{ {
while (queue.length > 0) { while (queue.length > 0) {
const job = queue.shift()! const job = queue.shift()!
await job()
const startedAt = new Date().toISOString()
let error: string | undefined
debug(`[Jobs] Starting job "${job.name}"`)
try {
await job.work()
} catch (e) {
error = e!.toString()
}
debug(`[Jobs] Finished job`)
const completedAt = new Date().toISOString()
const completedJob = {
uuid: job.uuid,
name: job.name,
successful: error === undefined,
error,
submitter: job.submitter,
submittedAt: job.submittedAt.toISOString(),
startedAt,
completedAt,
}
await jobsDB.update(async jobs => {
jobs.push(completedJob)
})
emitter.emit('job:completed', completedJob)
} }
} }
working = false working = false
} }
export function addJob(job: () => Promise<void>) { /**
queue.push(job) * Use this function to add new jobs to the work queue
*/
export function enqueueJob(job: Job & Worker) {
const queueJob = { ...job, uuid: randomUUID() }
queue.push(queueJob)
emitter.emit('job:add', {
uuid: queueJob.uuid,
name: queueJob.name,
submitter: queueJob.submitter,
submittedAt: queueJob.submittedAt,
})
// starts concurrently a function to process jobs // starts concurrently a function to process jobs
triggerProcessQueue() processQueue()
}
export async function getQueuedJobs(): Promise<QueuedJob[]> {
return queue.map(({ uuid, name, submitter, submittedAt }) => ({
uuid,
name,
submitter,
submittedAt,
}))
}
export function getCompletedJobs(): Promise<CompletedJob[]> {
return jobsDB.load()
}
export const OnJobAdded = {
addListener(cb: (job: QueuedJob) => void) {
emitter.on('job:add', cb)
},
removeListener(cb: (job: QueuedJob) => void) {
emitter.off('job:add', cb)
},
}
export const OnJobCompleted = {
addListener(cb: (job: CompletedJob) => void) {
emitter.on('job:completed', cb)
},
removeListener(cb: (job: CompletedJob) => void) {
emitter.off('job:completed', cb)
},
} }

@ -0,0 +1,81 @@
import { readFile, writeFile } from 'fs/promises'
import { Mutex } from 'async-mutex'
import { exists } from './utils'
import yaml from 'js-yaml'
export function createJsonDatabase<T>(filename: string, initialValue: T) {
const mutex = new Mutex()
function ensureExists() {
return mutex.runExclusive(async () => {
if (!(await exists(filename))) {
await writeFile(filename, JSON.stringify(initialValue, null, 2))
}
})
}
return {
mutex,
async load(): Promise<T> {
await ensureExists()
return await mutex.runExclusive(async () => {
const data = await readFile(filename, 'utf8')
return JSON.parse(data) as T
})
},
async update<R>(fn: (value: T) => Promise<R>) {
await ensureExists()
return await mutex.runExclusive(async () => {
const data = await readFile(filename, 'utf8')
const value = JSON.parse(data) as T
const result = await fn(value)
await writeFile(filename, JSON.stringify(value, null, 2))
return result
})
},
}
}
export function createYamlDatabase<T>(filename: string, initialValue: T) {
const mutex = new Mutex()
function ensureExists() {
return mutex.runExclusive(async () => {
if (!(await exists(filename))) {
await writeFile(filename, yaml.dump(initialValue))
}
})
}
return {
async load(): Promise<T> {
await ensureExists()
return await mutex.runExclusive(async () => {
const data = await readFile(filename, 'utf8')
return yaml.load(data) as T
})
},
async update<R>(fn: (value: T) => Promise<R>) {
await ensureExists()
return await mutex.runExclusive(async () => {
const data = await readFile(filename, 'utf8')
const value = yaml.load(data) as T
const result = await fn(value)
await writeFile(filename, yaml.dump(value))
return result
})
},
}
}

@ -0,0 +1,38 @@
import { access } from 'fs/promises'
export const normalizeURL = (url: string) => {
if (!url.startsWith('/') && !url.startsWith('https://') && !url.startsWith('http://')) {
url = `https://${url}`
}
return url
}
export const clsx = (...args: any[]) =>
args
.filter(Boolean)
.flatMap(s => (typeof s === 'string' ? s.split(' ') : [s]))
.join(' ')
/**
* Modern alternative to fs.existsSync
*/
export async function exists(path: string) {
try {
await access(path)
return true
} catch (err) {
// @ts-ignore
if (err.code === 'ENOENT') {
return false
} else {
throw err
}
}
}
export function sleep(timeout: number) {
return new Promise(resolve => {
setTimeout(resolve, timeout)
})
}

@ -0,0 +1,8 @@
import { inspect } from 'util'
export const debugToString = (value: any) =>
typeof value === 'string' ? value : inspect(value, false, 5, true)
export const debug = (...args: any[]) => {
process.stderr.write(args.map(arg => debugToString(arg)).join(' ') + '\n')
}

@ -0,0 +1,23 @@
import { getCompletedJobs, getQueuedJobs } from '@/jobs'
import { debug } from '@/logger'
import type { APIRoute } from 'astro'
export const GET: APIRoute = async ({ request, params }) => {
debug('[API] Jobs:', params)
const queuedJobs = await getQueuedJobs()
const completedJobs = await getCompletedJobs()
return new Response(
JSON.stringify({
queuedJobs,
completedJobs,
}),
{
status: 200,
headers: {
'Content-Type': 'application/json',
},
}
)
}

@ -0,0 +1,39 @@
import { OnJobAdded, OnJobCompleted, type CompletedJob, type QueuedJob } from '@/jobs'
import { debug } from '@/logger'
import type { APIRoute } from 'astro'
export const GET: APIRoute = async ({ request }) => {
let jobAddedEvent: any
let jobCompletedEvent: any
const stream = new ReadableStream({
start(controller) {
const sendEvent = (data: any) => {
controller.enqueue(`data: ${JSON.stringify(data)}\r\n\r\n`)
}
jobAddedEvent = (job: QueuedJob) => sendEvent({ type: 'added', job })
jobCompletedEvent = (job: CompletedJob) => sendEvent({ type: 'completed', job })
debug('[SSE] Registering client')
OnJobAdded.addListener(jobAddedEvent)
OnJobCompleted.addListener(jobCompletedEvent)
},
cancel() {
OnJobAdded.removeListener(jobAddedEvent)
OnJobCompleted.removeListener(jobCompletedEvent)
debug('[SSE] Un-registered client')
},
})
return new Response(stream, {
status: 200,
headers: {
'Content-Type': 'text/event-stream',
'Connection': 'keep-alive',
'Cache-Control': 'no-cache',
},
})
}

@ -1,5 +1,12 @@
import type { APIRoute } from 'astro' import type { APIRoute } from 'astro'
import { loadConfig } from '../../config'
import { loadConfig } from '@/config'
import { debug } from '@/logger'
import { normalizeURL } from '@/lib/utils'
import { enqueueJob } from '@/jobs'
import { createDeployJob } from '@/deploys'
export const POST: APIRoute = async ({ request }) => { export const POST: APIRoute = async ({ request }) => {
const body = await request.json() const body = await request.json()
@ -11,9 +18,9 @@ export const POST: APIRoute = async ({ request }) => {
for (const deploy of deploys) { for (const deploy of deploys) {
if (deploy.type !== 'docker') { if (deploy.type !== 'docker') {
if (URLS.includes(deploy.url)) { if (URLS.includes(normalizeURL(deploy.url))) {
// TODO: trigger deploy debug(`[Webhook] Triggering deploy for "${deploy.url}"`)
break enqueueJob(createDeployJob(deploy, { event: 'webhook', url: deploy.url }))
} }
} }
} }

@ -3,5 +3,9 @@ import Layout from '../layouts/Layout.astro'
--- ---
<Layout title="Homepage"> <Layout title="Homepage">
<h1>Astro</h1> <h1>phCD</h1>
<h2>Recent Deploys</h2>
<p>Bla bla recent jobs</p>
<h2>Containers</h2>
<p>Bla bla container status and stats</p>
</Layout> </Layout>

@ -0,0 +1,10 @@
---
import Layout from '@layouts/Layout.astro'
import { JobsPage } from '@client/JobsPage.jsx'
---
<Layout title="Deploys | phCD">
<h1>Jobs</h1>
<JobsPage client:load />
</Layout>

@ -1,46 +1,50 @@
import { refToString, type GitRef, type ShellDeploy, type Deploy } from './config'
import { existsSync } from 'fs'
import { promisify } from 'util'
import child_process from 'child_process' import child_process from 'child_process'
const exec = promisify(child_process.exec) import { debug } from './logger'
function getDeployDirectory(deploy: Deploy & { url: string; ref: GitRef }): string { function onLine(stream: any, cb: (line: string) => void) {
const { url, ref } = deploy let buffer = ''
const repoSlug = url.replace(/(^\w+:|^)\/\//, '').replace(/[^a-zA-Z]+/g, '-') stream.on('data', (data: any) => {
const slug = buffer += data
ref.type === 'default' let lines = buffer.split('\n')
? `${deploy.name}_${repoSlug}` buffer = lines.pop()!
: `${deploy.name}_${repoSlug}@${ref.value}` lines.forEach(cb)
})
return `${process.env.CLONE_PATH ?? './data.local/repos'}/${slug}` stream.on('end', () => {
if (buffer) cb(buffer)
})
} }
async function cloneOrUpdateRepo(deploy: Deploy & { url: string; ref: GitRef }) { export async function runCommand(source: string): Promise<void> {
const repoDir = getDeployDirectory(deploy) if (!source.includes('\n')) {
debug('[Runner] Command:')
if (existsSync(repoDir)) { debug('[Runner]', ' |', source)
await exec(`git -C ${repoDir} pull`)
} else { } else {
await exec(`git clone ${deploy.url} ${repoDir}`) debug('[Runner] Script:')
source.split('\n').forEach(line => {
debug('[Runner]', ' |', line)
})
} }
}
export async function shellRunner(deploy: ShellDeploy) {
const { url, ref, options } = deploy
const { path, env } = options
const projectName = url.split('/').slice(-1)[0].replace('.git', '')
await cloneOrUpdateRepo(deploy) const child = child_process.exec(source)
const envStr = Object.entries(env as any) onLine(child.stdout!, line => {
.map(([key, value]) => `${key}=${value}`) debug('[Runner]', ' >', line)
.join(' ') })
return exec(`cd ${projectName} && ${envStr} ${path}`) onLine(child.stderr!, line => {
debug('[Runner]', '!>', line)
})
return new Promise((resolve, reject) => {
child.on('close', code => {
if (code === 0) {
resolve()
} else {
reject(new Error(`Command failed with code ${code}`))
}
})
})
} }

@ -1,3 +1,6 @@
$accent-400: #f8f5ff;
$accent-500: #d8ccfa;
*, *,
*::before, *::before,
*::after { *::after {
@ -301,9 +304,10 @@ body {
.sidebar { .sidebar {
grid-area: sidebar; grid-area: sidebar;
min-width: 20vw; min-width: 300px;
width: 20vw;
background-color: #ededf0; background: $accent-400;
color: #333; color: #333;
display: flex; display: flex;
@ -352,7 +356,7 @@ body {
border-radius: 100px; border-radius: 100px;
&:hover { &:hover {
background-color: #00000018; background-color: $accent-500;
} }
.material-symbols-outlined { .material-symbols-outlined {
@ -372,22 +376,6 @@ body {
padding: 5px; padding: 5px;
position: relative; position: relative;
// &::before {
// content: '';
// background: #666;
// position: absolute;
// left: -0.5rem;
// width: 5px;
// height: 5px;
// border-radius: 100%;
// top: 50%;
// transform: translate(0, calc(-50% + 1px));
// }
} }
} }
@ -413,7 +401,52 @@ body {
justify-self: center; justify-self: center;
width: 80ch; width: 100%;
max-width: 100%; max-width: 80ch;
//
// Components
//
.list {
display: flex;
flex-direction: column;
width: 100%;
gap: 1rem;
}
.job {
padding: 1rem;
border-radius: 1rem;
width: 100%;
background: $accent-400;
display: grid;
grid-template-columns: 1fr;
grid-row: auto auto;
& > .name {
grid-column: span 3;
font-size: 18px;
font-weight: 600;
}
& > .footer {
display: flex;
justify-content: space-between;
align-items: center;
}
& .submitted-at {
font-size: 15px;
}
& .delta {
font-size: 15px;
}
}
} }
} }

@ -1,5 +0,0 @@
export const clsx = (...args) =>
args
.filter(Boolean)
.flatMap(s => (typeof s === 'string' ? s.split(' ') : [s]))
.join(' ')

@ -4,6 +4,14 @@
"strict": true, "strict": true,
"jsx": "react-jsx", "jsx": "react-jsx",
"jsxImportSource": "preact", "jsxImportSource": "preact",
"exactOptionalPropertyTypes": true "exactOptionalPropertyTypes": true,
"baseUrl": ".",
"paths": {
"@/*": ["src/*"],
"@lib/*": ["src/lib/*"],
"@components/*": ["src/components/*"],
"@layouts/*": ["src/layouts/*"],
"@client/*": ["src/client/*"]
}
} }
} }

Loading…
Cancel
Save