250 lines
6.1 KiB
TypeScript
250 lines
6.1 KiB
TypeScript
import { readFile, readdir } from 'node:fs/promises';
|
|
import { platform } from 'node:os';
|
|
import { join, resolve } from 'node:path';
|
|
import { kill } from 'node:process';
|
|
import { setTimeout } from 'node:timers/promises';
|
|
import { Config, CoreV1Api, KubeConfig } from '@kubernetes/client-node';
|
|
import { HeadscaleError, healthcheck } from '~/utils/headscale';
|
|
import log from '~/utils/log';
|
|
import { HeadplaneConfig } from '~server/context/parser';
|
|
import { Integration } from './abstract';
|
|
|
|
// TODO: Upgrade to the new CoreV1Api from @kubernetes/client-node
|
|
type T = NonNullable<HeadplaneConfig['integration']>['kubernetes'];
|
|
export default class KubernetesIntegration extends Integration<T> {
|
|
private pid: number | undefined;
|
|
private maxAttempts = 10;
|
|
|
|
get name() {
|
|
return 'Kubernetes (k8s)';
|
|
}
|
|
|
|
async isAvailable() {
|
|
if (platform() !== 'linux') {
|
|
log.error('INTG', 'Kubernetes is only available on Linux');
|
|
return false;
|
|
}
|
|
|
|
const svcRoot = Config.SERVICEACCOUNT_ROOT;
|
|
try {
|
|
log.debug('INTG', 'Checking Kubernetes service account at %s', svcRoot);
|
|
const files = await readdir(svcRoot);
|
|
if (files.length === 0) {
|
|
log.error('INTG', 'Kubernetes service account not found');
|
|
return false;
|
|
}
|
|
|
|
const mappedFiles = new Set(files.map((file) => join(svcRoot, file)));
|
|
const expectedFiles = [
|
|
Config.SERVICEACCOUNT_CA_PATH,
|
|
Config.SERVICEACCOUNT_TOKEN_PATH,
|
|
Config.SERVICEACCOUNT_NAMESPACE_PATH,
|
|
];
|
|
|
|
log.debug('INTG', 'Looking for %s', expectedFiles.join(', '));
|
|
if (!expectedFiles.every((file) => mappedFiles.has(file))) {
|
|
log.error('INTG', 'Malformed Kubernetes service account');
|
|
return false;
|
|
}
|
|
} catch (error) {
|
|
log.error('INTG', 'Failed to access %s: %s', svcRoot, error);
|
|
return false;
|
|
}
|
|
|
|
log.debug('INTG', 'Reading Kubernetes service account at %s', svcRoot);
|
|
const namespace = await readFile(
|
|
Config.SERVICEACCOUNT_NAMESPACE_PATH,
|
|
'utf8',
|
|
);
|
|
|
|
// Some very ugly nesting but it's necessary
|
|
if (process.env.HEADSCALE_INTEGRATION_UNSTRICT === 'true') {
|
|
log.warn('INTG', 'Skipping strict Pod status check');
|
|
} else {
|
|
const pod = process.env.POD_NAME;
|
|
if (!pod) {
|
|
log.error('INTG', 'Missing POD_NAME variable');
|
|
return false;
|
|
}
|
|
|
|
if (pod.trim().length === 0) {
|
|
log.error('INTG', 'Pod name is empty');
|
|
return false;
|
|
}
|
|
|
|
log.debug(
|
|
'INTG',
|
|
'Checking Kubernetes pod %s in namespace %s',
|
|
pod,
|
|
namespace,
|
|
);
|
|
|
|
try {
|
|
log.debug('INTG', 'Attempgin to get cluster KubeConfig');
|
|
const kc = new KubeConfig();
|
|
kc.loadFromCluster();
|
|
|
|
const cluster = kc.getCurrentCluster();
|
|
if (!cluster) {
|
|
log.error('INTG', 'Malformed kubeconfig');
|
|
return false;
|
|
}
|
|
|
|
log.info(
|
|
'INTG',
|
|
'Service account connected to %s (%s)',
|
|
cluster.name,
|
|
cluster.server,
|
|
);
|
|
|
|
const kCoreV1Api = kc.makeApiClient(CoreV1Api);
|
|
|
|
log.info(
|
|
'INTG',
|
|
'Checking pod %s in namespace %s (%s)',
|
|
pod,
|
|
namespace,
|
|
kCoreV1Api.basePath,
|
|
);
|
|
|
|
log.debug('INTG', 'Reading pod info for %s', pod);
|
|
const { response, body } = await kCoreV1Api.readNamespacedPod(
|
|
pod,
|
|
namespace,
|
|
);
|
|
|
|
if (response.statusCode !== 200) {
|
|
log.error(
|
|
'INTG',
|
|
'Failed to read pod info: http %d',
|
|
response.statusCode,
|
|
);
|
|
return false;
|
|
}
|
|
|
|
log.debug('INTG', 'Got pod info: %o', body.spec);
|
|
const shared = body.spec?.shareProcessNamespace;
|
|
if (shared === undefined) {
|
|
log.error('INTG', 'Pod does not have spec.shareProcessNamespace set');
|
|
return false;
|
|
}
|
|
|
|
if (!shared) {
|
|
log.error(
|
|
'INTG',
|
|
'Pod has set but disabled spec.shareProcessNamespace',
|
|
);
|
|
return false;
|
|
}
|
|
|
|
log.info('INTG', 'Pod %s enabled shared processes', pod);
|
|
} catch (error) {
|
|
log.error('INTG', 'Failed to read pod info: %s', error);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
log.debug('INTG', 'Looking for namespaced process in /proc');
|
|
const dir = resolve('/proc');
|
|
try {
|
|
const subdirs = await readdir(dir);
|
|
const promises = subdirs.map(async (dir) => {
|
|
const pid = Number.parseInt(dir, 10);
|
|
|
|
if (Number.isNaN(pid)) {
|
|
return;
|
|
}
|
|
|
|
const path = join('/proc', dir, 'cmdline');
|
|
try {
|
|
log.debug('INTG', 'Reading %s', path);
|
|
const data = await readFile(path, 'utf8');
|
|
if (data.includes('headscale')) {
|
|
return pid;
|
|
}
|
|
} catch (error) {
|
|
log.debug('INTG', 'Failed to read %s: %s', path, error);
|
|
}
|
|
});
|
|
|
|
const results = await Promise.allSettled(promises);
|
|
const pids = [];
|
|
|
|
for (const result of results) {
|
|
if (result.status === 'fulfilled' && result.value) {
|
|
pids.push(result.value);
|
|
}
|
|
}
|
|
|
|
log.debug('INTG', 'Found Headscale processes: %o', pids);
|
|
if (pids.length > 1) {
|
|
log.error(
|
|
'INTG',
|
|
'Found %d Headscale processes: %s',
|
|
pids.length,
|
|
pids.join(', '),
|
|
);
|
|
return false;
|
|
}
|
|
|
|
if (pids.length === 0) {
|
|
log.error('INTG', 'Could not find Headscale process');
|
|
return false;
|
|
}
|
|
|
|
this.pid = pids[0];
|
|
log.info('INTG', 'Found Headscale process with PID: %d', this.pid);
|
|
return true;
|
|
} catch {
|
|
log.error('INTG', 'Failed to read /proc');
|
|
return false;
|
|
}
|
|
}
|
|
|
|
async onConfigChange() {
|
|
if (!this.pid) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
log.info('INTG', 'Sending SIGTERM to Headscale');
|
|
kill(this.pid, 'SIGTERM');
|
|
} catch (error) {
|
|
log.error('INTG', 'Failed to send SIGTERM to Headscale: %s', error);
|
|
log.debug('INTG', 'kill(1) error: %o', error);
|
|
}
|
|
|
|
await setTimeout(1000);
|
|
let attempts = 0;
|
|
while (attempts <= this.maxAttempts) {
|
|
try {
|
|
log.debug('INTG', 'Checking Headscale status (attempt %d)', attempts);
|
|
await healthcheck();
|
|
log.info('INTG', 'Headscale is up and running');
|
|
return;
|
|
} catch (error) {
|
|
if (error instanceof HeadscaleError && error.status === 401) {
|
|
break;
|
|
}
|
|
|
|
if (error instanceof HeadscaleError && error.status === 404) {
|
|
break;
|
|
}
|
|
|
|
if (attempts < this.maxAttempts) {
|
|
attempts++;
|
|
await setTimeout(1000);
|
|
continue;
|
|
}
|
|
|
|
log.error(
|
|
'INTG',
|
|
'Missed restart deadline for Headscale (pid %d)',
|
|
this.pid,
|
|
);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|