π File detail
utils/pasteStore.ts
π― Use case
This file lives under βutils/β, which covers cross-cutting helpers (shell, tempfiles, settings, messages, process input, β¦). On the API surface it exposes hashPastedText, storePastedText, retrievePastedText, and cleanupOldPastes β mainly functions, hooks, or classes. Dependencies touch crypto, Node filesystem, and Node path helpers. It composes internal code from debug, envUtils, and errors (relative imports).
Generated from folder role, exports, dependency roots, and inline comments β not hand-reviewed for every path.
π§ Inline summary
import { createHash } from 'crypto' import { mkdir, readdir, readFile, stat, unlink, writeFile } from 'fs/promises' import { join } from 'path' import { logForDebugging } from './debug.js' import { getClaudeConfigHomeDir } from './envUtils.js'
π€ Exports (heuristic)
hashPastedTextstorePastedTextretrievePastedTextcleanupOldPastes
π External import roots
Package roots from from "β¦" (relative paths omitted).
cryptofspath
π₯οΈ Source preview
import { createHash } from 'crypto'
import { mkdir, readdir, readFile, stat, unlink, writeFile } from 'fs/promises'
import { join } from 'path'
import { logForDebugging } from './debug.js'
import { getClaudeConfigHomeDir } from './envUtils.js'
import { isENOENT } from './errors.js'
const PASTE_STORE_DIR = 'paste-cache'
/**
* Get the paste store directory (persistent across sessions).
*/
function getPasteStoreDir(): string {
return join(getClaudeConfigHomeDir(), PASTE_STORE_DIR)
}
/**
* Generate a hash for paste content to use as filename.
* Exported so callers can get the hash synchronously before async storage.
*/
export function hashPastedText(content: string): string {
return createHash('sha256').update(content).digest('hex').slice(0, 16)
}
/**
* Get the file path for a paste by its content hash.
*/
function getPastePath(hash: string): string {
return join(getPasteStoreDir(), `${hash}.txt`)
}
/**
* Store pasted text content to disk.
* The hash should be pre-computed with hashPastedText() so the caller
* can use it immediately without waiting for the async disk write.
*/
export async function storePastedText(
hash: string,
content: string,
): Promise<void> {
try {
const dir = getPasteStoreDir()
await mkdir(dir, { recursive: true })
const pastePath = getPastePath(hash)
// Content-addressable: same hash = same content, so overwriting is safe
await writeFile(pastePath, content, { encoding: 'utf8', mode: 0o600 })
logForDebugging(`Stored paste ${hash} to ${pastePath}`)
} catch (error) {
logForDebugging(`Failed to store paste: ${error}`)
}
}
/**
* Retrieve pasted text content by its hash.
* Returns null if not found or on error.
*/
export async function retrievePastedText(hash: string): Promise<string | null> {
try {
const pastePath = getPastePath(hash)
return await readFile(pastePath, { encoding: 'utf8' })
} catch (error) {
// ENOENT is expected when paste doesn't exist
if (!isENOENT(error)) {
logForDebugging(`Failed to retrieve paste ${hash}: ${error}`)
}
return null
}
}
/**
* Clean up old paste files that are no longer referenced.
* This is a simple time-based cleanup - removes files older than cutoffDate.
*/
export async function cleanupOldPastes(cutoffDate: Date): Promise<void> {
const pasteDir = getPasteStoreDir()
let files
try {
files = await readdir(pasteDir)
} catch {
// Directory doesn't exist or can't be read - nothing to clean up
return
}
const cutoffTime = cutoffDate.getTime()
for (const file of files) {
if (!file.endsWith('.txt')) {
continue
}
const filePath = join(pasteDir, file)
try {
const stats = await stat(filePath)
if (stats.mtimeMs < cutoffTime) {
await unlink(filePath)
logForDebugging(`Cleaned up old paste: ${filePath}`)
}
} catch {
// Ignore errors for individual files
}
}
}