mirror of
https://github.com/Start9Labs/start-os.git
synced 2026-03-26 10:21:52 +00:00
* add documentation for ai agents * docs: consolidate CLAUDE.md and CONTRIBUTING.md, add style guidelines - Refactor CLAUDE.md to reference CONTRIBUTING.md for build/test/format info - Expand CONTRIBUTING.md with comprehensive build targets, env vars, and testing - Add code style guidelines section with conventional commits - Standardize SDK prettier config to use single quotes (matching web) - Add project-level Claude Code settings to disable co-author attribution * style(sdk): apply prettier with single quotes Run prettier across sdk/base and sdk/package to apply the standardized quote style (single quotes matching web). * docs: add USER.md for per-developer TODO filtering - Add agents/USER.md to .gitignore (contains user identifier) - Document session startup flow in CLAUDE.md: - Create USER.md if missing, prompting for identifier - Filter TODOs by @username tags - Offer relevant TODOs on session start * docs: add i18n documentation task to agent TODOs * docs: document i18n ID patterns in core/ Add agents/i18n-patterns.md covering rust-i18n setup, translation file format, t!() macro usage, key naming conventions, and locale selection. Remove completed TODO item and add reference in CLAUDE.md. * chore: clarify that all builds work on any OS with Docker
169 lines
4.6 KiB
TypeScript
169 lines
4.6 KiB
TypeScript
import * as T from '../../../base/lib/types'
|
|
import { IdMap, MountOptions } from '../util/SubContainer'
|
|
|
|
type MountArray = { mountpoint: string; options: MountOptions }[]
|
|
|
|
type SharedOptions = {
|
|
/** The path within the resource to mount. Use `null` to mount the entire resource */
|
|
subpath: string | null
|
|
/** Where to mount the resource. e.g. /data */
|
|
mountpoint: string
|
|
/**
|
|
* Whether to mount this as a file or directory
|
|
*
|
|
* defaults to "directory"
|
|
* */
|
|
type?: 'file' | 'directory' | 'infer'
|
|
// /**
|
|
// * Whether to map uids/gids for the mount
|
|
// *
|
|
// * https://www.kernel.org/doc/html/latest/filesystems/idmappings.html
|
|
// */
|
|
// idmap?: {
|
|
// /** The (starting) id of the data on the filesystem (u) */
|
|
// fromId: number
|
|
// /** The (starting) id of the data in the mount point (k) */
|
|
// toId: number
|
|
// /**
|
|
// * Optional: the number of incremental ids to map (r)
|
|
// *
|
|
// * defaults to 1
|
|
// * */
|
|
// range?: number
|
|
// }[]
|
|
}
|
|
|
|
type VolumeOpts<Manifest extends T.SDKManifest> = {
|
|
/** The ID of the volume to mount. Must be one of the volume IDs defined in the manifest */
|
|
volumeId: Manifest['volumes'][number]
|
|
/** Whether or not the resource should be readonly for this subcontainer */
|
|
readonly: boolean
|
|
} & SharedOptions
|
|
|
|
type DependencyOpts<Manifest extends T.SDKManifest> = {
|
|
/** The ID of the dependency */
|
|
dependencyId: Manifest['id']
|
|
/** The ID of the volume to mount. Must be one of the volume IDs defined in the manifest of the dependency */
|
|
volumeId: Manifest['volumes'][number]
|
|
/** Whether or not the resource should be readonly for this subcontainer */
|
|
readonly: boolean
|
|
} & SharedOptions
|
|
|
|
export class Mounts<
|
|
Manifest extends T.SDKManifest,
|
|
Backups extends SharedOptions = never,
|
|
> {
|
|
private constructor(
|
|
readonly volumes: VolumeOpts<Manifest>[],
|
|
readonly assets: SharedOptions[],
|
|
readonly dependencies: DependencyOpts<T.SDKManifest>[],
|
|
readonly backups: Backups[],
|
|
) {}
|
|
|
|
static of<Manifest extends T.SDKManifest>() {
|
|
return new Mounts<Manifest>([], [], [], [])
|
|
}
|
|
|
|
mountVolume(options: VolumeOpts<Manifest>) {
|
|
return new Mounts<Manifest, Backups>(
|
|
[...this.volumes, options],
|
|
[...this.assets],
|
|
[...this.dependencies],
|
|
[...this.backups],
|
|
)
|
|
}
|
|
|
|
mountAssets(options: SharedOptions) {
|
|
return new Mounts<Manifest, Backups>(
|
|
[...this.volumes],
|
|
[...this.assets, options],
|
|
[...this.dependencies],
|
|
[...this.backups],
|
|
)
|
|
}
|
|
|
|
mountDependency<DependencyManifest extends T.SDKManifest>(
|
|
options: DependencyOpts<DependencyManifest>,
|
|
) {
|
|
return new Mounts<Manifest, Backups>(
|
|
[...this.volumes],
|
|
[...this.assets],
|
|
[...this.dependencies, options],
|
|
[...this.backups],
|
|
)
|
|
}
|
|
|
|
mountBackups(options: SharedOptions) {
|
|
return new Mounts<
|
|
Manifest,
|
|
{
|
|
subpath: string | null
|
|
mountpoint: string
|
|
}
|
|
>(
|
|
[...this.volumes],
|
|
[...this.assets],
|
|
[...this.dependencies],
|
|
[...this.backups, options],
|
|
)
|
|
}
|
|
|
|
build(): MountArray {
|
|
const mountpoints = new Set()
|
|
for (let mountpoint of this.volumes
|
|
.map((v) => v.mountpoint)
|
|
.concat(this.assets.map((a) => a.mountpoint))
|
|
.concat(this.dependencies.map((d) => d.mountpoint))) {
|
|
if (mountpoints.has(mountpoint)) {
|
|
throw new Error(
|
|
`cannot mount more than once to mountpoint ${mountpoint}`,
|
|
)
|
|
}
|
|
mountpoints.add(mountpoint)
|
|
}
|
|
return ([] as MountArray)
|
|
.concat(
|
|
this.volumes.map((v) => ({
|
|
mountpoint: v.mountpoint,
|
|
options: {
|
|
type: 'volume',
|
|
volumeId: v.volumeId,
|
|
subpath: v.subpath,
|
|
readonly: v.readonly,
|
|
filetype: v.type ?? 'directory',
|
|
idmap: [],
|
|
},
|
|
})),
|
|
)
|
|
.concat(
|
|
this.assets.map((a) => ({
|
|
mountpoint: a.mountpoint,
|
|
options: {
|
|
type: 'assets',
|
|
subpath: a.subpath,
|
|
filetype: a.type ?? 'directory',
|
|
idmap: [],
|
|
},
|
|
})),
|
|
)
|
|
.concat(
|
|
this.dependencies.map((d) => ({
|
|
mountpoint: d.mountpoint,
|
|
options: {
|
|
type: 'pointer',
|
|
packageId: d.dependencyId,
|
|
volumeId: d.volumeId,
|
|
subpath: d.subpath,
|
|
readonly: d.readonly,
|
|
filetype: d.type ?? 'directory',
|
|
idmap: [],
|
|
},
|
|
})),
|
|
)
|
|
}
|
|
}
|
|
|
|
const a = Mounts.of().mountBackups({ subpath: null, mountpoint: '' })
|
|
// @ts-expect-error
|
|
const m: Mounts<T.SDKManifest, never> = a
|