mirror of
https://github.com/cupcakearmy/autorestic.git
synced 2025-01-04 22:36:25 +00:00
initial 2 commit
This commit is contained in:
parent
732f728ff8
commit
7c11ba18fd
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
node_modules
|
||||||
|
/autorestic
|
14
migration.md
Normal file
14
migration.md
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Rename backend to repository
|
||||||
|
|
||||||
|
# Env variables
|
||||||
|
|
||||||
|
AUTORESTIC_BB_B2_ACCOUNT_ID=123 -> AUTORESTIC_BACKENDS_BB_ENV_B2**ACCOUNT**ID=123
|
||||||
|
|
||||||
|
- All fields can be configured by env now
|
||||||
|
- To escape `_` replace it with double underscore `__`
|
||||||
|
|
||||||
|
# Rest property on backend config
|
||||||
|
|
||||||
|
No rest property anymore, can be used in string extrapolation
|
||||||
|
|
||||||
|
# Every string is now replaceable with env variables
|
23
package.json
Normal file
23
package.json
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
{
|
||||||
|
"private": true,
|
||||||
|
"name": "autorestic",
|
||||||
|
"module": "src/index.ts",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"schema:gen": "bun run ./scripts/generateSchema.ts",
|
||||||
|
"bin": "bun build ./src/index.ts --compile --outfile autorestic"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"bun-types": "^0.6.0",
|
||||||
|
"typescript": "^5.0.0",
|
||||||
|
"zod-to-json-schema": "^3.21.2"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@commander-js/extra-typings": "^11.0.0",
|
||||||
|
"commander": "^11.0.0",
|
||||||
|
"pino": "^8.14.1",
|
||||||
|
"pino-pretty": "^10.0.0",
|
||||||
|
"yaml": "^2.3.1",
|
||||||
|
"zod": "^3.21.4"
|
||||||
|
}
|
||||||
|
}
|
252
schema/config.json
Normal file
252
schema/config.json
Normal file
@ -0,0 +1,252 @@
|
|||||||
|
{
|
||||||
|
"$ref": "#/definitions/mySchema",
|
||||||
|
"definitions": {
|
||||||
|
"mySchema": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"version": {
|
||||||
|
"type": "number",
|
||||||
|
"description": "version number"
|
||||||
|
},
|
||||||
|
"repos": {
|
||||||
|
"type": "object",
|
||||||
|
"additionalProperties": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"type": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": [
|
||||||
|
"local",
|
||||||
|
"sftp",
|
||||||
|
"rest",
|
||||||
|
"swift",
|
||||||
|
"s3",
|
||||||
|
"b2",
|
||||||
|
"azure",
|
||||||
|
"gs",
|
||||||
|
"rclone"
|
||||||
|
],
|
||||||
|
"description": "type of repository"
|
||||||
|
},
|
||||||
|
"path": {
|
||||||
|
"type": "string",
|
||||||
|
"minLength": 1,
|
||||||
|
"description": "restic path"
|
||||||
|
},
|
||||||
|
"key": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/path",
|
||||||
|
"description": "encryption key for the repository"
|
||||||
|
},
|
||||||
|
"env": {
|
||||||
|
"type": "object",
|
||||||
|
"additionalProperties": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/path",
|
||||||
|
"description": "value of the environment variable"
|
||||||
|
},
|
||||||
|
"description": "environment variables"
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"all": {
|
||||||
|
"type": "object",
|
||||||
|
"additionalProperties": {
|
||||||
|
"anyOf": [
|
||||||
|
{
|
||||||
|
"type": "boolean",
|
||||||
|
"const": true,
|
||||||
|
"description": "boolean flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"anyOf": [
|
||||||
|
{
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/path",
|
||||||
|
"description": "non-empty string that can extrapolate env variables inside it"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/options/properties/all/additionalProperties/anyOf/1/anyOf/0"
|
||||||
|
},
|
||||||
|
"minItems": 1
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "value of option"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"backup": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/options/properties/all"
|
||||||
|
},
|
||||||
|
"forget": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/options/properties/all"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": false,
|
||||||
|
"description": "options"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"type",
|
||||||
|
"path",
|
||||||
|
"key"
|
||||||
|
],
|
||||||
|
"additionalProperties": false
|
||||||
|
},
|
||||||
|
"description": "available repositories"
|
||||||
|
},
|
||||||
|
"locations": {
|
||||||
|
"type": "object",
|
||||||
|
"additionalProperties": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"from": {
|
||||||
|
"anyOf": [
|
||||||
|
{
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/path",
|
||||||
|
"description": "local path to backup"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/locations/additionalProperties/properties/from/anyOf/0"
|
||||||
|
},
|
||||||
|
"minItems": 1
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"to": {
|
||||||
|
"anyOf": [
|
||||||
|
{
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/path",
|
||||||
|
"description": "repository to backup to"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/locations/additionalProperties/properties/to/anyOf/0"
|
||||||
|
},
|
||||||
|
"minItems": 1
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"copy": {
|
||||||
|
"type": "object",
|
||||||
|
"additionalProperties": {
|
||||||
|
"anyOf": [
|
||||||
|
{
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/path",
|
||||||
|
"description": "destination repository"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/locations/additionalProperties/properties/copy/additionalProperties/anyOf/0"
|
||||||
|
},
|
||||||
|
"minItems": 1
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"cron": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/path",
|
||||||
|
"description": "execute backups for the given cron job"
|
||||||
|
},
|
||||||
|
"hooks": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"before": {
|
||||||
|
"anyOf": [
|
||||||
|
{
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/path",
|
||||||
|
"description": "command to be executed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/locations/additionalProperties/properties/hooks/properties/before/anyOf/0"
|
||||||
|
},
|
||||||
|
"minItems": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"description": "list of commands"
|
||||||
|
},
|
||||||
|
"after": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/locations/additionalProperties/properties/hooks/properties/before",
|
||||||
|
"description": "list of commands"
|
||||||
|
},
|
||||||
|
"failure": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/locations/additionalProperties/properties/hooks/properties/before",
|
||||||
|
"description": "list of commands"
|
||||||
|
},
|
||||||
|
"success": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/locations/additionalProperties/properties/hooks/properties/before",
|
||||||
|
"description": "list of commands"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": false,
|
||||||
|
"description": "hooks to be executed"
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"all": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/options/properties/all"
|
||||||
|
},
|
||||||
|
"backup": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/options/properties/backup"
|
||||||
|
},
|
||||||
|
"forget": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/repos/additionalProperties/properties/options/properties/forget"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": false,
|
||||||
|
"description": "native restic options"
|
||||||
|
},
|
||||||
|
"forget": {
|
||||||
|
"anyOf": [
|
||||||
|
{
|
||||||
|
"type": "boolean",
|
||||||
|
"description": "automatically run \"forget\" when backing up"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "string",
|
||||||
|
"const": "prune",
|
||||||
|
"description": "also prune when forgetting"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"from",
|
||||||
|
"to"
|
||||||
|
],
|
||||||
|
"additionalProperties": false,
|
||||||
|
"description": "Location"
|
||||||
|
},
|
||||||
|
"description": "available locations"
|
||||||
|
},
|
||||||
|
"global": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"options": {
|
||||||
|
"$ref": "#/definitions/mySchema/properties/locations/additionalProperties/properties/options",
|
||||||
|
"description": "native restic options"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": false,
|
||||||
|
"description": "global configuration"
|
||||||
|
},
|
||||||
|
"extras": {}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"version",
|
||||||
|
"repos",
|
||||||
|
"locations"
|
||||||
|
],
|
||||||
|
"additionalProperties": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"$schema": "http://json-schema.org/draft-07/schema#"
|
||||||
|
}
|
17
scripts/generateSchema.ts
Normal file
17
scripts/generateSchema.ts
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
import { mkdir, rm, writeFile } from 'node:fs/promises'
|
||||||
|
import { zodToJsonSchema } from 'zod-to-json-schema'
|
||||||
|
import { ConfigSchema } from '../src/config/schema/config'
|
||||||
|
|
||||||
|
const OUTPUT = './schema'
|
||||||
|
|
||||||
|
await rm(OUTPUT, { recursive: true, force: true })
|
||||||
|
await mkdir(OUTPUT, { recursive: true })
|
||||||
|
|
||||||
|
const Schemas = {
|
||||||
|
config: ConfigSchema,
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const [name, schema] of Object.entries(Schemas)) {
|
||||||
|
const jsonSchema = zodToJsonSchema(schema, 'mySchema')
|
||||||
|
await writeFile(`${OUTPUT}/${name}.json`, JSON.stringify(jsonSchema, null, 2), { encoding: 'utf-8' })
|
||||||
|
}
|
14
src/cmd/backup.ts
Normal file
14
src/cmd/backup.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { Log } from '../logger'
|
||||||
|
import { Context } from '../models/context'
|
||||||
|
|
||||||
|
export async function backup(ctx: Context) {
|
||||||
|
const log = Log.child({ cmd: 'check' })
|
||||||
|
log.trace('starting')
|
||||||
|
|
||||||
|
// Locations
|
||||||
|
for (const location of ctx.locations) {
|
||||||
|
await location.backup()
|
||||||
|
}
|
||||||
|
|
||||||
|
log.trace('done')
|
||||||
|
}
|
25
src/cmd/check.ts
Normal file
25
src/cmd/check.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { unlockRepo, waitForRepo } from '../lock'
|
||||||
|
import { Log } from '../logger'
|
||||||
|
import { Context } from '../models/context'
|
||||||
|
import { isResticAvailable } from '../restic'
|
||||||
|
|
||||||
|
export async function check(ctx: Context) {
|
||||||
|
const l = Log.child({ cmd: 'check' })
|
||||||
|
l.trace('starting')
|
||||||
|
|
||||||
|
// Restic
|
||||||
|
isResticAvailable()
|
||||||
|
|
||||||
|
// Repos
|
||||||
|
for (const repo of ctx.repos) {
|
||||||
|
await waitForRepo(ctx, repo.name)
|
||||||
|
try {
|
||||||
|
await repo.init()
|
||||||
|
await repo.check()
|
||||||
|
} finally {
|
||||||
|
unlockRepo(ctx, repo.name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
l.trace('done')
|
||||||
|
}
|
21
src/config/env/file.test.ts
vendored
Normal file
21
src/config/env/file.test.ts
vendored
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import { describe, expect, test } from 'bun:test'
|
||||||
|
import { InvalidEnvFileLine } from '../../errors'
|
||||||
|
import { parseFile } from './file'
|
||||||
|
|
||||||
|
describe('env file', () => {
|
||||||
|
test('simple', () => {
|
||||||
|
expect(parseFile(`test_foo=ok`)).toEqual({ test_foo: 'ok' })
|
||||||
|
})
|
||||||
|
|
||||||
|
test('multiple values', () => {
|
||||||
|
expect(parseFile(`test_foo=ok\n \n spacing = foo \n`)).toEqual({ test_foo: 'ok', spacing: 'foo' })
|
||||||
|
})
|
||||||
|
|
||||||
|
test('invalid: key', () => {
|
||||||
|
expect(() => parseFile(`a=123\na f=ok`)).toThrow(new InvalidEnvFileLine('a f=ok'))
|
||||||
|
})
|
||||||
|
|
||||||
|
test('invalid: missing =', () => {
|
||||||
|
expect(() => parseFile(`a=123\na ok`)).toThrow(new InvalidEnvFileLine('a ok'))
|
||||||
|
})
|
||||||
|
})
|
55
src/config/env/file.ts
vendored
Normal file
55
src/config/env/file.ts
vendored
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
import { exists, readFile } from 'node:fs/promises'
|
||||||
|
import { InvalidEnvFileLine } from '../../errors'
|
||||||
|
import { setByPath } from '../../utils/path'
|
||||||
|
import { relativePath } from '../resolution'
|
||||||
|
|
||||||
|
export function parseFile(contents: string) {
|
||||||
|
const variables: Record<string, string> = {}
|
||||||
|
const lines = contents
|
||||||
|
.trim()
|
||||||
|
.split('\n')
|
||||||
|
.map((l) => l.trim())
|
||||||
|
const matcher = /^\s*(?<variable>\w+)\s*=(?<value>.*)$/
|
||||||
|
for (const line of lines) {
|
||||||
|
if (!line) continue
|
||||||
|
const match = matcher.exec(line)
|
||||||
|
if (!match) throw new InvalidEnvFileLine(line)
|
||||||
|
variables[match.groups!.variable] = match.groups!.value.trim()
|
||||||
|
}
|
||||||
|
return variables
|
||||||
|
}
|
||||||
|
|
||||||
|
const PREFIX = 'AUTORESTIC_'
|
||||||
|
|
||||||
|
function envVariableToObjectPath(env: string): string {
|
||||||
|
if (env.startsWith(PREFIX)) env = env.replace(PREFIX, '')
|
||||||
|
return (
|
||||||
|
env
|
||||||
|
// Convert to object path
|
||||||
|
.replaceAll('_', '.')
|
||||||
|
// Escape the double unterscore. __ -> .. -> _
|
||||||
|
.replaceAll('..', '_')
|
||||||
|
.toLowerCase()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fill the config file with the env file variables.
|
||||||
|
* These take precedence before the config file itself.
|
||||||
|
*/
|
||||||
|
export async function enrichConfig(rawConfig: any, path: string) {
|
||||||
|
const envFilePath = relativePath(path, '.autorestic.env')
|
||||||
|
let variables: Record<string, string> = {}
|
||||||
|
|
||||||
|
if (await exists(envFilePath)) {
|
||||||
|
const envFile = parseFile(await readFile(envFilePath, 'utf-8'))
|
||||||
|
Object.assign(variables, envFile)
|
||||||
|
}
|
||||||
|
|
||||||
|
Object.assign(variables, process.env)
|
||||||
|
|
||||||
|
for (const [key, value] of Object.entries(variables)) {
|
||||||
|
if (!key.startsWith(PREFIX)) continue
|
||||||
|
setByPath(rawConfig, envVariableToObjectPath(key), value)
|
||||||
|
}
|
||||||
|
}
|
29
src/config/index.ts
Normal file
29
src/config/index.ts
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
import { exists, readFile } from 'node:fs/promises'
|
||||||
|
import yaml from 'yaml'
|
||||||
|
import { ConfigFileNotFound, CustomError, InvalidConfigFile } from '../errors'
|
||||||
|
import { enrichConfig } from './env/file'
|
||||||
|
import { autoLocateConfig } from './resolution'
|
||||||
|
import { Config, ConfigWithMetaSchema } from './schema/config'
|
||||||
|
import { basename } from 'node:path'
|
||||||
|
|
||||||
|
export async function loadConfig(customPath?: string): Promise<Config> {
|
||||||
|
let path: string
|
||||||
|
if (customPath) {
|
||||||
|
path = customPath
|
||||||
|
if (!(await exists(path))) throw new ConfigFileNotFound([path])
|
||||||
|
} else {
|
||||||
|
path = await autoLocateConfig()
|
||||||
|
}
|
||||||
|
|
||||||
|
const rawConfig = await readFile(path, 'utf-8')
|
||||||
|
const config = yaml.parse(rawConfig)
|
||||||
|
await enrichConfig(config, path)
|
||||||
|
config.meta = { path: basename(path) }
|
||||||
|
const parsed = ConfigWithMetaSchema.safeParse(config)
|
||||||
|
if (!parsed.success)
|
||||||
|
throw new InvalidConfigFile(parsed.error.errors.map((e) => `${e.path.join(' > ')}: ${e.message}`))
|
||||||
|
|
||||||
|
// Check for semantics
|
||||||
|
|
||||||
|
return parsed.data
|
||||||
|
}
|
23
src/config/resolution.ts
Normal file
23
src/config/resolution.ts
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
import { exists } from 'node:fs/promises'
|
||||||
|
import { dirname, isAbsolute, join, resolve } from 'node:path'
|
||||||
|
import { ConfigFileNotFound } from '../errors'
|
||||||
|
|
||||||
|
const DEFAULT_DIRS = ['./', '~/', '~/.config/autorestic']
|
||||||
|
const FILENAMES = ['.autorestic.yaml', '.autorestic.yml', '.autorestic.json']
|
||||||
|
|
||||||
|
export async function autoLocateConfig(): Promise<string> {
|
||||||
|
const paths = DEFAULT_DIRS
|
||||||
|
const xdgHome = process.env['XDG_CONFIG_HOME']
|
||||||
|
if (xdgHome) paths.push(xdgHome)
|
||||||
|
for (const path in paths) {
|
||||||
|
for (const filename in FILENAMES) {
|
||||||
|
const file = join(path, filename)
|
||||||
|
if (await exists(file)) return file
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new ConfigFileNotFound(paths)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function relativePath(base: string, path: string): string {
|
||||||
|
return isAbsolute(path) ? path : resolve(base, path)
|
||||||
|
}
|
20
src/config/schema/common.ts
Normal file
20
src/config/schema/common.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import { ZodTypeAny, z } from 'zod'
|
||||||
|
import { Log } from '../../logger'
|
||||||
|
|
||||||
|
export const NonEmptyString = z
|
||||||
|
.string()
|
||||||
|
.min(1)
|
||||||
|
// Extrapolate env variables from a string
|
||||||
|
.transform((s) => {
|
||||||
|
return s.replaceAll(/\$(\w+)|\${(\w+)}/g, (_, g0, g1) => {
|
||||||
|
const variable = g0 || g1
|
||||||
|
const value = process.env[variable] ?? ''
|
||||||
|
if (!value) Log.error(`cannot find environment variable "${variable}" to replace in ${s}`)
|
||||||
|
return value
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.describe('non-empty string that can extrapolate env variables inside it')
|
||||||
|
|
||||||
|
export function OptionallyArray<T extends ZodTypeAny>(type: T) {
|
||||||
|
return z.union([type, z.array(type).min(1)])
|
||||||
|
}
|
69
src/config/schema/config.ts
Normal file
69
src/config/schema/config.ts
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
import { z } from 'zod'
|
||||||
|
import { asArray } from '../../utils/array'
|
||||||
|
import { RepositorySchema } from './repository'
|
||||||
|
import { NonEmptyString } from './common'
|
||||||
|
import { LocationSchema } from './location'
|
||||||
|
import { OptionsSchema } from './options'
|
||||||
|
|
||||||
|
export const ConfigSchema = z.strictObject({
|
||||||
|
version: z.number().describe('version number'),
|
||||||
|
repos: z.record(NonEmptyString.describe('repository name'), RepositorySchema).describe('available repositories'),
|
||||||
|
locations: z.record(NonEmptyString.describe('location name'), LocationSchema).describe('available locations'),
|
||||||
|
global: z
|
||||||
|
.strictObject({
|
||||||
|
options: OptionsSchema.optional(),
|
||||||
|
})
|
||||||
|
.describe('global configuration')
|
||||||
|
.optional(),
|
||||||
|
extras: z.any().optional(),
|
||||||
|
})
|
||||||
|
|
||||||
|
const ConfigMeta = z
|
||||||
|
.strictObject({
|
||||||
|
path: NonEmptyString.describe('The path of the loaded config'),
|
||||||
|
})
|
||||||
|
.describe('Meta information about the config')
|
||||||
|
|
||||||
|
export const ConfigWithMetaSchema = ConfigSchema.extend({
|
||||||
|
meta: ConfigMeta,
|
||||||
|
}).superRefine((config, ctx) => {
|
||||||
|
const availableRepos = Object.keys(config.repos)
|
||||||
|
for (const [name, location] of Object.entries(config.locations)) {
|
||||||
|
const locationPath = [...ctx.path, 'locations', name]
|
||||||
|
const toRepos = asArray(location.to)
|
||||||
|
// Check if all target repos are valid
|
||||||
|
for (const to of toRepos) {
|
||||||
|
if (!availableRepos.includes(to)) {
|
||||||
|
const message = `location "${name}" has an invalid repository "${to}"`
|
||||||
|
ctx.addIssue({ message, code: 'custom', path: [...locationPath, 'to'] })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Check copy field
|
||||||
|
if (!location.copy) continue
|
||||||
|
for (const [source, destinations] of Object.entries(location.copy)) {
|
||||||
|
const path = [...locationPath, 'copy', source]
|
||||||
|
if (!toRepos.includes(source))
|
||||||
|
ctx.addIssue({
|
||||||
|
code: 'custom',
|
||||||
|
path,
|
||||||
|
message: `copy source "${source}" must be also a backup target`,
|
||||||
|
})
|
||||||
|
for (const destination of asArray(destinations)) {
|
||||||
|
if (destination === source)
|
||||||
|
ctx.addIssue({
|
||||||
|
code: 'custom',
|
||||||
|
path: [...path, destination],
|
||||||
|
message: `destination repository "${destination}" cannot be also the source in copy field`,
|
||||||
|
})
|
||||||
|
if (!availableRepos.includes(destination))
|
||||||
|
ctx.addIssue({
|
||||||
|
code: 'custom',
|
||||||
|
path: [...path, destination],
|
||||||
|
message: `destination repository "${destination}" does not exist`,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
export type Config = z.infer<typeof ConfigWithMetaSchema>
|
14
src/config/schema/hooks.ts
Normal file
14
src/config/schema/hooks.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { z } from 'zod'
|
||||||
|
import { NonEmptyString, OptionallyArray } from './common'
|
||||||
|
|
||||||
|
const Command = NonEmptyString.describe('command to be executed')
|
||||||
|
const Commands = OptionallyArray(Command).describe('list of commands')
|
||||||
|
|
||||||
|
export const HooksSchema = z
|
||||||
|
.strictObject({
|
||||||
|
before: Commands.optional(),
|
||||||
|
after: Commands.optional(),
|
||||||
|
failure: Commands.optional(),
|
||||||
|
success: Commands.optional(),
|
||||||
|
})
|
||||||
|
.describe('hooks to be executed')
|
28
src/config/schema/location.ts
Normal file
28
src/config/schema/location.ts
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import { z } from 'zod'
|
||||||
|
import { NonEmptyString, OptionallyArray } from './common'
|
||||||
|
import { HooksSchema } from './hooks'
|
||||||
|
import { OptionsSchema } from './options'
|
||||||
|
|
||||||
|
export const LocationSchema = z
|
||||||
|
.strictObject({
|
||||||
|
from: OptionallyArray(NonEmptyString.describe('local path to backup')),
|
||||||
|
to: OptionallyArray(NonEmptyString.describe('repository to backup to')),
|
||||||
|
copy: z
|
||||||
|
.record(
|
||||||
|
NonEmptyString.describe('source repository from which to copy from'),
|
||||||
|
OptionallyArray(NonEmptyString.describe('destination repository'))
|
||||||
|
)
|
||||||
|
.optional(),
|
||||||
|
|
||||||
|
// adapter:
|
||||||
|
cron: NonEmptyString.describe('execute backups for the given cron job').optional(),
|
||||||
|
hooks: HooksSchema.optional(),
|
||||||
|
options: OptionsSchema.optional(),
|
||||||
|
forget: z
|
||||||
|
.union([
|
||||||
|
z.boolean().describe('automatically run "forget" when backing up'),
|
||||||
|
z.literal('prune').describe('also prune when forgetting'),
|
||||||
|
])
|
||||||
|
.optional(),
|
||||||
|
})
|
||||||
|
.describe('Location')
|
15
src/config/schema/options.ts
Normal file
15
src/config/schema/options.ts
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
import { z } from 'zod'
|
||||||
|
import { NonEmptyString, OptionallyArray } from './common'
|
||||||
|
|
||||||
|
const OptionSchema = z.record(
|
||||||
|
NonEmptyString.describe('native restic option'),
|
||||||
|
z.union([z.literal(true).describe('boolean flag'), OptionallyArray(NonEmptyString)]).describe('value of option')
|
||||||
|
)
|
||||||
|
|
||||||
|
export const OptionsSchema = z
|
||||||
|
.strictObject({
|
||||||
|
all: OptionSchema.optional(),
|
||||||
|
backup: OptionSchema.optional(),
|
||||||
|
forget: OptionSchema.optional(),
|
||||||
|
})
|
||||||
|
.describe('native restic options')
|
18
src/config/schema/repository.ts
Normal file
18
src/config/schema/repository.ts
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
import { z } from 'zod'
|
||||||
|
import { NonEmptyString } from './common'
|
||||||
|
import { OptionsSchema } from './options'
|
||||||
|
|
||||||
|
export const RepositorySchema = z.strictObject({
|
||||||
|
type: z.enum(['local', 'sftp', 'rest', 'swift', 's3', 'b2', 'azure', 'gs', 'rclone']).describe('type of repository'),
|
||||||
|
path: NonEmptyString.describe('restic path'),
|
||||||
|
key: NonEmptyString.describe('encryption key for the repository'),
|
||||||
|
env: z
|
||||||
|
.record(
|
||||||
|
NonEmptyString.describe('environment variable'),
|
||||||
|
NonEmptyString.describe('value of the environment variable')
|
||||||
|
)
|
||||||
|
.transform((env) => Object.fromEntries(Object.entries(env).map(([key, value]) => [key.toUpperCase(), value])))
|
||||||
|
.describe('environment variables')
|
||||||
|
.optional(),
|
||||||
|
options: OptionsSchema.describe('options').optional(),
|
||||||
|
})
|
47
src/errors/index.ts
Normal file
47
src/errors/index.ts
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
function formatLines(lines: string[]) {
|
||||||
|
return lines.map((p) => ` ▶ ${p}`).join('\n')
|
||||||
|
}
|
||||||
|
|
||||||
|
export class CustomError extends Error {}
|
||||||
|
|
||||||
|
export class InvalidEnvFileLine extends CustomError {
|
||||||
|
constructor(line: string) {
|
||||||
|
super(`invalid env file line: "${line}"`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class NotImplemented extends CustomError {
|
||||||
|
constructor(functionality: string) {
|
||||||
|
super(`not implemented: ${functionality}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ConfigFileNotFound extends CustomError {
|
||||||
|
constructor(paths: string[]) {
|
||||||
|
super(`could not locate config file.\nthe following paths were tried:\n${formatLines(paths)}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class InvalidConfigFile extends CustomError {
|
||||||
|
constructor(errors: string[]) {
|
||||||
|
super(`could not parse the config file.\n${formatLines(errors)}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class BinaryNotAvailable extends CustomError {
|
||||||
|
constructor(binary: string) {
|
||||||
|
super(`binary "${binary}" is not available in $PATH`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ResticError extends CustomError {
|
||||||
|
constructor(errors: string[]) {
|
||||||
|
super(`internal restic error.\n${formatLines(errors)}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class LockfileAlreadyLocked extends CustomError {
|
||||||
|
constructor(repo: string) {
|
||||||
|
super(`cannot acquire lock for repository "${repo}", already in use`)
|
||||||
|
}
|
||||||
|
}
|
138
src/index.ts
Normal file
138
src/index.ts
Normal file
@ -0,0 +1,138 @@
|
|||||||
|
import { Command, Help, Option, program } from '@commander-js/extra-typings'
|
||||||
|
import { loadConfig } from './config'
|
||||||
|
import { CustomError, NotImplemented } from './errors'
|
||||||
|
import { Log, LogLevel, setLevelFromFlag } from './logger'
|
||||||
|
import { check } from './cmd/check'
|
||||||
|
import { Context } from './models/context'
|
||||||
|
import { backup } from './cmd/backup'
|
||||||
|
|
||||||
|
export const helpConfig: Partial<Help> = {
|
||||||
|
showGlobalOptions: true,
|
||||||
|
sortOptions: true,
|
||||||
|
sortSubcommands: true,
|
||||||
|
helpWidth: 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
program
|
||||||
|
.name('autorestic')
|
||||||
|
.description('configuration manager and runner for restic')
|
||||||
|
.version('2.0.0-alpha.0')
|
||||||
|
.configureHelp(helpConfig)
|
||||||
|
.allowExcessArguments(false)
|
||||||
|
.allowUnknownOption(false)
|
||||||
|
|
||||||
|
// Global options
|
||||||
|
program.option('-c, --config <file>', 'specify custom configuration file')
|
||||||
|
program.option('-v', 'verbosity', (_, previous) => previous + 1, 1)
|
||||||
|
program.addOption(new Option('--ci', 'CI mode').env('CI').default(false))
|
||||||
|
|
||||||
|
// Common Options
|
||||||
|
const specificLocation = new Option('-l, --location <locations...>', 'location name, multiple possible')
|
||||||
|
specificLocation.variadic = true
|
||||||
|
const allLocations = new Option('-a, --all', 'all locations')
|
||||||
|
const specificRepo = new Option('-r, --repository <names...>', 'repository name, multiple possible')
|
||||||
|
specificLocation.variadic = true
|
||||||
|
const allRepos = new Option('-a, --all', 'all repositories')
|
||||||
|
|
||||||
|
function mergeOptions<T extends {}>(local: T, p: Command) {
|
||||||
|
const globals = p.optsWithGlobals() as { config?: string; verbosity: number; ci: boolean }
|
||||||
|
return {
|
||||||
|
...globals,
|
||||||
|
...local,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
program.hook('preAction', (command) => {
|
||||||
|
// @ts-ignore
|
||||||
|
const v: number = command.opts().v
|
||||||
|
setLevelFromFlag(v)
|
||||||
|
})
|
||||||
|
|
||||||
|
program
|
||||||
|
.command('check')
|
||||||
|
.description('check if the config is valid and sets up the repositories')
|
||||||
|
.configureHelp(helpConfig)
|
||||||
|
.action(async (options, p) => {
|
||||||
|
const merged = mergeOptions(options, p)
|
||||||
|
const config = await loadConfig(merged.config)
|
||||||
|
const ctx = new Context(config)
|
||||||
|
await check(ctx)
|
||||||
|
})
|
||||||
|
|
||||||
|
program
|
||||||
|
.command('backup')
|
||||||
|
.description('create backups')
|
||||||
|
.configureHelp(helpConfig)
|
||||||
|
.addOption(specificLocation)
|
||||||
|
.addOption(allLocations)
|
||||||
|
.action(async (options, p) => {
|
||||||
|
// throw new NotImplemented('backup')
|
||||||
|
const merged = mergeOptions(options, p)
|
||||||
|
const config = await loadConfig(merged.config)
|
||||||
|
const ctx = new Context(config)
|
||||||
|
await backup(ctx)
|
||||||
|
})
|
||||||
|
|
||||||
|
program
|
||||||
|
.command('exec')
|
||||||
|
.description('execute arbitrary native restic commands for given repositories')
|
||||||
|
.configureHelp(helpConfig)
|
||||||
|
.addOption(specificRepo)
|
||||||
|
.addOption(allRepos)
|
||||||
|
.allowExcessArguments(true)
|
||||||
|
.action((options, p) => {
|
||||||
|
throw new NotImplemented('exec')
|
||||||
|
})
|
||||||
|
|
||||||
|
program
|
||||||
|
.command('forget')
|
||||||
|
.description('forget snapshots according to the specified policies')
|
||||||
|
.configureHelp(helpConfig)
|
||||||
|
.addOption(specificLocation)
|
||||||
|
.addOption(allLocations)
|
||||||
|
// Pass natively
|
||||||
|
// .option('--dry-run', 'do not write changes, show what would be affected')
|
||||||
|
// .option('--prune', 'also prune repository')
|
||||||
|
.action((options) => {
|
||||||
|
throw new NotImplemented('backup')
|
||||||
|
})
|
||||||
|
|
||||||
|
program
|
||||||
|
.command('restore')
|
||||||
|
.description('restore a snapshot to a given location')
|
||||||
|
.option('--force', 'overwrite target folder')
|
||||||
|
.option('--from <repository>', 'repository from which to restore')
|
||||||
|
.option('--to <path>', 'path where to restore the data')
|
||||||
|
.option('-l, --location <location>', 'location to be restored')
|
||||||
|
.argument('[snapshot-id]', 'snapshot to be restored. if empty latest will be taken')
|
||||||
|
.action(() => {
|
||||||
|
throw new NotImplemented('restore')
|
||||||
|
})
|
||||||
|
|
||||||
|
const self = new Command('self').description('utility commands for managing autorestic').configureHelp(helpConfig)
|
||||||
|
self.command('install').action(() => {
|
||||||
|
throw new NotImplemented('install')
|
||||||
|
})
|
||||||
|
self.command('uninstall').action(() => {
|
||||||
|
throw new NotImplemented('uninstall')
|
||||||
|
})
|
||||||
|
self.command('upgrade').action(() => {
|
||||||
|
throw new NotImplemented('upgrade')
|
||||||
|
})
|
||||||
|
self.command('completion').action(() => {
|
||||||
|
throw new NotImplemented('completion')
|
||||||
|
})
|
||||||
|
program.addCommand(self)
|
||||||
|
|
||||||
|
try {
|
||||||
|
await program.parseAsync()
|
||||||
|
} catch (e) {
|
||||||
|
if (e instanceof CustomError) {
|
||||||
|
Log.fatal(e.message)
|
||||||
|
} else if (e instanceof Error) {
|
||||||
|
Log.fatal(`unknown error: ${e.message}`)
|
||||||
|
}
|
||||||
|
process.exit(1)
|
||||||
|
} finally {
|
||||||
|
// TODO: Unlock
|
||||||
|
}
|
20
src/lock/index.test.ts
Normal file
20
src/lock/index.test.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import { describe, expect, mock, test, beforeEach } from 'bun:test'
|
||||||
|
import { lockRepo } from '.'
|
||||||
|
import { Context } from '../models/context'
|
||||||
|
import { mkdir, rm } from 'node:fs/promises'
|
||||||
|
|
||||||
|
const mockPath = './test/'
|
||||||
|
const mockContext: Context = { config: { meta: { path: mockPath } } } as any
|
||||||
|
|
||||||
|
describe('lock', () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
// Cleanup lock file
|
||||||
|
await rm(mockPath, { recursive: true, force: true })
|
||||||
|
await mkdir(mockPath, { recursive: true })
|
||||||
|
})
|
||||||
|
|
||||||
|
test('lock', () => {
|
||||||
|
lockRepo(mockContext, 'foo')
|
||||||
|
// lockRepo(mockContext, 'foo')
|
||||||
|
})
|
||||||
|
})
|
78
src/lock/index.ts
Normal file
78
src/lock/index.ts
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
import { readFileSync, writeFileSync } from 'node:fs'
|
||||||
|
import yaml from 'yaml'
|
||||||
|
import { relativePath } from '../config/resolution'
|
||||||
|
import { LockfileAlreadyLocked } from '../errors'
|
||||||
|
import { Log } from '../logger'
|
||||||
|
import { Context } from '../models/context'
|
||||||
|
import { Lockfile, LockfileSchema } from './schema'
|
||||||
|
import { wait } from '../utils/time'
|
||||||
|
|
||||||
|
const LOCKFILE = '.autorestic.lock'
|
||||||
|
const VERSION = 2
|
||||||
|
const l = Log.child({ command: 'lock' })
|
||||||
|
|
||||||
|
function load(ctx: Context): Lockfile {
|
||||||
|
const defaultLockfile = { version: VERSION, cron: {}, running: {} }
|
||||||
|
try {
|
||||||
|
const path = relativePath(ctx.config.meta.path, LOCKFILE)
|
||||||
|
l.trace('looking for lock file', { path })
|
||||||
|
// throw new Error(path)
|
||||||
|
const rawConfig = readFileSync(path, 'utf-8')
|
||||||
|
const config = yaml.parse(rawConfig)
|
||||||
|
const parsed = LockfileSchema.safeParse(config)
|
||||||
|
if (!parsed.success) return defaultLockfile
|
||||||
|
if (parsed.data.version < VERSION) {
|
||||||
|
l.debug('lockfile is old and will be overwritten')
|
||||||
|
return defaultLockfile
|
||||||
|
}
|
||||||
|
return parsed.data
|
||||||
|
} catch {
|
||||||
|
return defaultLockfile
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function write(ctx: Context, lockfile: Lockfile) {
|
||||||
|
const path = relativePath(ctx.config.meta.path, LOCKFILE)
|
||||||
|
writeFileSync(path, yaml.stringify(lockfile), 'utf-8')
|
||||||
|
}
|
||||||
|
|
||||||
|
export function lockRepo(ctx: Context, repo: string) {
|
||||||
|
const lock = load(ctx)
|
||||||
|
l.trace('trying to lock repository', { repo })
|
||||||
|
if (lock.running[repo]) throw new LockfileAlreadyLocked(repo)
|
||||||
|
lock.running[repo] = true
|
||||||
|
write(ctx, lock)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Waits for a repo to become unlocked, and errors if it does not succeed in the given timeout.
|
||||||
|
*
|
||||||
|
* @param [timeout=10] max seconds to wait for repo to become unlocked
|
||||||
|
*/
|
||||||
|
export async function waitForRepo(ctx: Context, repo: string, timeout = 10) {
|
||||||
|
const now = Date.now()
|
||||||
|
while (Date.now() - now < timeout * 1_000) {
|
||||||
|
try {
|
||||||
|
lockRepo(ctx, repo)
|
||||||
|
l.trace('repo is free again', { repo })
|
||||||
|
break
|
||||||
|
} catch {
|
||||||
|
l.trace('waiting for repo to be unlocked', { repo })
|
||||||
|
await wait(0.1) // Wait for 100ms
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new LockfileAlreadyLocked(repo)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function updateLastRun(ctx: Context, location: string) {
|
||||||
|
const lock = load(ctx)
|
||||||
|
lock.cron[location] = Date.now()
|
||||||
|
write(ctx, lock)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function unlockRepo(ctx: Context, repo: string) {
|
||||||
|
l.trace('unlocking repository', { repo })
|
||||||
|
const lock = load(ctx)
|
||||||
|
lock.running[repo] = false
|
||||||
|
write(ctx, lock)
|
||||||
|
}
|
13
src/lock/schema.ts
Normal file
13
src/lock/schema.ts
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import { z } from 'zod'
|
||||||
|
|
||||||
|
export const LockfileSchema = z.strictObject({
|
||||||
|
version: z.number().min(0).describe('lockfile version'),
|
||||||
|
running: z
|
||||||
|
.record(z.string().describe('repository'), z.boolean().describe('whether repository is running'))
|
||||||
|
.describe('running information for each repository'),
|
||||||
|
cron: z
|
||||||
|
.record(z.string().describe('location'), z.number().describe('timestamp of last backup'))
|
||||||
|
.describe('information about last run for a given location. in milliseconds'),
|
||||||
|
})
|
||||||
|
|
||||||
|
export type Lockfile = z.infer<typeof LockfileSchema>
|
37
src/logger.ts
Normal file
37
src/logger.ts
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
import Pino, { type LoggerOptions } from 'pino'
|
||||||
|
import Pretty from 'pino-pretty'
|
||||||
|
|
||||||
|
// https://getpino.io/#/docs/api?id=loggerlevel-string-gettersetter
|
||||||
|
export enum LogLevel {
|
||||||
|
Trace = 'trace',
|
||||||
|
Debug = 'debug',
|
||||||
|
Info = 'info',
|
||||||
|
Warn = 'warn',
|
||||||
|
Error = 'error',
|
||||||
|
Fatal = 'fatal',
|
||||||
|
Silent = 'silent',
|
||||||
|
}
|
||||||
|
|
||||||
|
const pretty = !process.env.CI
|
||||||
|
const options: LoggerOptions = {
|
||||||
|
base: undefined,
|
||||||
|
level: LogLevel.Info,
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Log = pretty ? Pino(options, Pretty({ colorize: true })) : Pino(options)
|
||||||
|
|
||||||
|
export function setLevelFromFlag(flag: number) {
|
||||||
|
switch (flag) {
|
||||||
|
case 1:
|
||||||
|
Log.level = LogLevel.Info
|
||||||
|
break
|
||||||
|
case 2:
|
||||||
|
Log.level = LogLevel.Debug
|
||||||
|
break
|
||||||
|
case 3:
|
||||||
|
Log.level = LogLevel.Trace
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
Log.error('invalid logging level')
|
||||||
|
}
|
||||||
|
}
|
25
src/models/context.ts
Normal file
25
src/models/context.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { Config } from '../config/schema/config'
|
||||||
|
import { CustomError } from '../errors'
|
||||||
|
import { Location } from './location'
|
||||||
|
import { Repository } from './repository'
|
||||||
|
|
||||||
|
export class Context {
|
||||||
|
repos: Repository[]
|
||||||
|
locations: Location[]
|
||||||
|
|
||||||
|
constructor(public config: Config) {
|
||||||
|
this.repos = Object.entries(config.repos).map(([name, r]) => new Repository(this, name, r))
|
||||||
|
this.locations = Object.entries(config.locations).map(([name, l]) => new Location(this, name, l))
|
||||||
|
}
|
||||||
|
|
||||||
|
getRepo(name: string) {
|
||||||
|
const repo = this.repos.find((r) => r.name === name)
|
||||||
|
if (!repo) throw new CustomError(`could not find backend "${name}"`)
|
||||||
|
return repo
|
||||||
|
}
|
||||||
|
getLocation(name: string) {
|
||||||
|
const location = this.locations.find((l) => l.name === name)
|
||||||
|
if (!location) throw new CustomError(`could not find location "${name}"`)
|
||||||
|
return location
|
||||||
|
}
|
||||||
|
}
|
28
src/models/location.ts
Normal file
28
src/models/location.ts
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import { Logger } from 'pino'
|
||||||
|
import { z } from 'zod'
|
||||||
|
import { LocationSchema } from '../config/schema/location'
|
||||||
|
import { Log } from '../logger'
|
||||||
|
import { asArray } from '../utils/array'
|
||||||
|
import { Context } from './context'
|
||||||
|
import { execute } from '../restic'
|
||||||
|
|
||||||
|
export class Location {
|
||||||
|
l: Logger
|
||||||
|
|
||||||
|
constructor(public ctx: Context, public name: string, public data: z.infer<typeof LocationSchema>) {
|
||||||
|
this.l = Log.child({ location: name })
|
||||||
|
}
|
||||||
|
|
||||||
|
async backup() {
|
||||||
|
this.l.trace('backing up location')
|
||||||
|
for (const name of asArray(this.data.to)) {
|
||||||
|
const repo = this.ctx.getRepo(name)
|
||||||
|
this.l.debug(repo.name)
|
||||||
|
await execute({
|
||||||
|
command: 'restic',
|
||||||
|
args: ['backup', '--dry-run'],
|
||||||
|
env: repo.env,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
66
src/models/repository.ts
Normal file
66
src/models/repository.ts
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
import { Logger } from 'pino'
|
||||||
|
import { z } from 'zod'
|
||||||
|
import { relativePath } from '../config/resolution'
|
||||||
|
import { Config } from '../config/schema/config'
|
||||||
|
import { RepositorySchema } from '../config/schema/repository'
|
||||||
|
import { ResticError } from '../errors'
|
||||||
|
import { Log } from '../logger'
|
||||||
|
import { execute } from '../restic'
|
||||||
|
import { Context } from './context'
|
||||||
|
|
||||||
|
export class Repository {
|
||||||
|
l: Logger
|
||||||
|
|
||||||
|
constructor(public ctx: Context, public name: string, public data: z.infer<typeof RepositorySchema>) {
|
||||||
|
this.l = Log.child({ repository: this.name })
|
||||||
|
}
|
||||||
|
|
||||||
|
get repository(): string {
|
||||||
|
switch (this.data.type) {
|
||||||
|
case 'local':
|
||||||
|
return relativePath(this.ctx.config.meta.path, this.data.path)
|
||||||
|
case 'b2':
|
||||||
|
case 'azure':
|
||||||
|
case 'gs':
|
||||||
|
case 's3':
|
||||||
|
case 'sftp':
|
||||||
|
case 'rclone':
|
||||||
|
case 'swift':
|
||||||
|
case 'rest':
|
||||||
|
return `${this.data.type}:${this.data.path}`
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get env() {
|
||||||
|
return {
|
||||||
|
...this.data.env,
|
||||||
|
RESTIC_PASSWORD: this.data.key,
|
||||||
|
RESTIC_REPOSITORY: this.repository,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* true if initialized
|
||||||
|
* false if already initialized
|
||||||
|
*/
|
||||||
|
async init(): Promise<boolean> {
|
||||||
|
this.l.trace('initializing')
|
||||||
|
const output = await execute({ command: 'restic', args: ['init'], env: this.env })
|
||||||
|
if (!output.ok) {
|
||||||
|
if (output.stderr.includes('config file already exists')) {
|
||||||
|
this.l.debug('already initialized')
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
throw new ResticError([output.stderr])
|
||||||
|
}
|
||||||
|
this.l.debug('initialized repository')
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
async check() {
|
||||||
|
this.l.trace('checking')
|
||||||
|
const output = await execute({ command: 'restic', args: ['check'], env: this.env })
|
||||||
|
if (!output.ok) throw new ResticError(['could not check repository', output.stderr])
|
||||||
|
}
|
||||||
|
}
|
44
src/restic/index.ts
Normal file
44
src/restic/index.ts
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import { execFile } from 'node:child_process'
|
||||||
|
import { Log } from '../logger'
|
||||||
|
import { BinaryNotAvailable } from '../errors'
|
||||||
|
|
||||||
|
export type ExecutionContext = {
|
||||||
|
command: string
|
||||||
|
args?: string[]
|
||||||
|
env?: Record<string, string>
|
||||||
|
}
|
||||||
|
export async function execute({
|
||||||
|
env,
|
||||||
|
args,
|
||||||
|
command,
|
||||||
|
}: ExecutionContext): Promise<{ code: number; stderr: string; stdout: string; ok: boolean }> {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
execFile(command, args ?? [], { env }, (err, stdout, stderr) => {
|
||||||
|
const code = err?.code ?? 0
|
||||||
|
resolve({
|
||||||
|
code,
|
||||||
|
ok: code === 0,
|
||||||
|
stderr,
|
||||||
|
stdout,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function isBinaryAvailable(command: string): Promise<boolean> {
|
||||||
|
const l = Log.child({ command })
|
||||||
|
try {
|
||||||
|
l.trace('checking if command is installed')
|
||||||
|
const result = await execute({ command })
|
||||||
|
return result.ok
|
||||||
|
} catch {
|
||||||
|
l.trace('not installed')
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function isResticAvailable() {
|
||||||
|
const bin = 'restic'
|
||||||
|
const installed = await isBinaryAvailable(bin)
|
||||||
|
if (!installed) throw new BinaryNotAvailable(bin)
|
||||||
|
}
|
12
src/utils/array.test.ts
Normal file
12
src/utils/array.test.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import { describe, expect, test } from 'bun:test'
|
||||||
|
import { isSubset } from './array'
|
||||||
|
|
||||||
|
describe('set theory', () => {
|
||||||
|
test('subset', () => {
|
||||||
|
expect(isSubset([1], [1, 2])).toBe(true)
|
||||||
|
expect(isSubset([1], [2])).toBe(false)
|
||||||
|
expect(isSubset([], [])).toBe(true)
|
||||||
|
expect(isSubset([1, 2, 3], [1, 2])).toBe(false)
|
||||||
|
expect(isSubset([1, 2], [1, 2])).toBe(true)
|
||||||
|
})
|
||||||
|
})
|
7
src/utils/array.ts
Normal file
7
src/utils/array.ts
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
export function asArray<T>(singleOrArray: T | T[]): T[] {
|
||||||
|
return Array.isArray(singleOrArray) ? singleOrArray : [singleOrArray]
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isSubset<T>(subset: T[], set: T[]): boolean {
|
||||||
|
return subset.every((v) => set.includes(v))
|
||||||
|
}
|
33
src/utils/path.test.ts
Normal file
33
src/utils/path.test.ts
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
import { expect, test, describe } from 'bun:test'
|
||||||
|
import { setByPath } from './path'
|
||||||
|
|
||||||
|
describe('set by path', () => {
|
||||||
|
test('simple', () => {
|
||||||
|
expect(setByPath({}, 'a', true)).toEqual({ a: true })
|
||||||
|
expect(setByPath({}, 'f', { ok: true })).toEqual({ f: { ok: true } })
|
||||||
|
expect(setByPath([], '0', true)).toEqual([true])
|
||||||
|
expect(setByPath([], '2', false)).toEqual([undefined, undefined, false])
|
||||||
|
})
|
||||||
|
|
||||||
|
test('object', () => {
|
||||||
|
expect(setByPath({}, 'a.b', true)).toEqual({ a: { b: true } })
|
||||||
|
expect(setByPath({}, 'a.b.c', true)).toEqual({ a: { b: { c: true } } })
|
||||||
|
expect(setByPath({ a: true }, 'b', false)).toEqual({ a: true, b: false })
|
||||||
|
expect(setByPath({ a: { b: true } }, 'a.c', false)).toEqual({ a: { b: true, c: false } })
|
||||||
|
|
||||||
|
expect(() => setByPath({ a: 'foo' }, 'a.b', true)).toThrow()
|
||||||
|
expect(setByPath({ a: 'foo' }, 'a', true)).toEqual({ a: true })
|
||||||
|
})
|
||||||
|
|
||||||
|
test('array', () => {
|
||||||
|
expect(() => setByPath([], 'a', true)).toThrow()
|
||||||
|
expect(setByPath([], '0', true)).toEqual([true])
|
||||||
|
expect(setByPath([], '0.0.0', true)).toEqual([[[true]]])
|
||||||
|
expect(setByPath([], '0.1.2', true)).toEqual([[undefined, [undefined, undefined, true]]])
|
||||||
|
})
|
||||||
|
|
||||||
|
test('mixed', () => {
|
||||||
|
expect(setByPath({ items: [] }, 'items.0.name', 'John')).toEqual({ items: [{ name: 'John' }] })
|
||||||
|
expect(setByPath([], '0.name', 'John')).toEqual([{ name: 'John' }])
|
||||||
|
})
|
||||||
|
})
|
24
src/utils/path.ts
Normal file
24
src/utils/path.ts
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
function parseKey(key: any) {
|
||||||
|
const asNumber = parseInt(key)
|
||||||
|
const isString = isNaN(asNumber)
|
||||||
|
return [isString ? key : asNumber, isString]
|
||||||
|
}
|
||||||
|
|
||||||
|
export function setByPath(source: object, path: string, value: unknown) {
|
||||||
|
const segments = path.split('.')
|
||||||
|
const last = segments.length - 1
|
||||||
|
let node: any = source
|
||||||
|
for (const [i, segment] of segments.entries()) {
|
||||||
|
const [key, isString] = parseKey(segment)
|
||||||
|
if (Array.isArray(node) && isString) throw new Error(`array require a numeric index`)
|
||||||
|
if (typeof node !== 'object') throw new Error(`could not set path "${segment}" on ${node}.`)
|
||||||
|
if (i === last) {
|
||||||
|
node[key] = value
|
||||||
|
} else {
|
||||||
|
const [_, isNextString] = parseKey(segments[i + 1])
|
||||||
|
if (node[key] === undefined) node[key] = isNextString ? {} : []
|
||||||
|
node = node[key]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return source
|
||||||
|
}
|
3
src/utils/time.ts
Normal file
3
src/utils/time.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
export function wait(seconds: number): Promise<never> {
|
||||||
|
return new Promise((resolve) => setTimeout(resolve, seconds * 1000))
|
||||||
|
}
|
8
tsconfig.json
Normal file
8
tsconfig.json
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"extends": "./node_modules/bun-types/tsconfig.json",
|
||||||
|
"compilerOptions": {
|
||||||
|
"strict": true,
|
||||||
|
"noFallthroughCasesInSwitch": true,
|
||||||
|
"noEmit": true
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user