Files
vscode/extensions/git/src/repositoryCache.ts
Alex Ross d42b85fdc6 Support .git in repo cache and improve language (#275075)
* Support .git in repo cache and improve language
Done while testing https://github.com/microsoft/vscode/issues/274930

* Tests
2025-11-04 16:48:24 +01:00

195 lines
6.0 KiB
TypeScript

/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
import { LogOutputChannel, Memento, workspace } from 'vscode';
import { LRUCache } from './cache';
import { Remote } from './api/git';
import { isDescendant } from './util';
export interface RepositoryCacheInfo {
workspacePath: string; // path of the workspace folder or workspace file
}
function isRepositoryCacheInfo(obj: unknown): obj is RepositoryCacheInfo {
if (!obj || typeof obj !== 'object') {
return false;
}
const rec = obj as Record<string, unknown>;
return typeof rec.workspacePath === 'string';
}
export class RepositoryCache {
private static readonly STORAGE_KEY = 'git.repositoryCache';
private static readonly MAX_REPO_ENTRIES = 30; // Max repositories tracked
private static readonly MAX_FOLDER_ENTRIES = 10; // Max folders per repository
private normalizeRepoUrl(url: string): string {
try {
const trimmed = url.trim();
return trimmed.replace(/(?:\.git)?\/*$/i, '');
} catch {
return url;
}
}
// Outer LRU: repoUrl -> inner LRU (folderPathOrWorkspaceFile -> RepositoryCacheInfo).
private readonly lru = new LRUCache<string, LRUCache<string, RepositoryCacheInfo>>(RepositoryCache.MAX_REPO_ENTRIES);
constructor(public readonly _globalState: Memento, private readonly _logger: LogOutputChannel) {
this.load();
}
// Exposed for testing
protected get _workspaceFile() {
return workspace.workspaceFile;
}
// Exposed for testing
protected get _workspaceFolders() {
return workspace.workspaceFolders;
}
/**
* Associate a repository remote URL with a local workspace folder or workspace file.
* Re-associating bumps recency and persists the updated LRU state.
* @param repoUrl Remote repository URL (e.g. https://github.com/owner/repo.git)
* @param rootPath Root path of the local repo clone.
*/
set(repoUrl: string, rootPath: string): void {
const key = this.normalizeRepoUrl(repoUrl);
let foldersLru = this.lru.get(key);
if (!foldersLru) {
foldersLru = new LRUCache<string, RepositoryCacheInfo>(RepositoryCache.MAX_FOLDER_ENTRIES);
}
const folderPathOrWorkspaceFile: string | undefined = this._findWorkspaceForRepo(rootPath);
if (!folderPathOrWorkspaceFile) {
return;
}
foldersLru.set(folderPathOrWorkspaceFile, {
workspacePath: folderPathOrWorkspaceFile
}); // touch entry
this.lru.set(key, foldersLru);
this.save();
}
private _findWorkspaceForRepo(rootPath: string): string | undefined {
// If the current workspace is a workspace file, use that. Otherwise, find the workspace folder that contains the rootUri
let folderPathOrWorkspaceFile: string | undefined;
try {
if (this._workspaceFile) {
folderPathOrWorkspaceFile = this._workspaceFile.fsPath;
} else if (this._workspaceFolders && this._workspaceFolders.length) {
const sorted = [...this._workspaceFolders].sort((a, b) => b.uri.fsPath.length - a.uri.fsPath.length);
for (const folder of sorted) {
const folderPath = folder.uri.fsPath;
if (isDescendant(folderPath, rootPath) || isDescendant(rootPath, folderPath)) {
folderPathOrWorkspaceFile = folderPath;
break;
}
}
}
return folderPathOrWorkspaceFile;
} catch {
return;
}
}
update(addedRemotes: Remote[], removedRemotes: Remote[], rootPath: string): void {
for (const remote of removedRemotes) {
const url = remote.fetchUrl;
if (!url) {
continue;
}
const relatedWorkspace = this._findWorkspaceForRepo(rootPath);
if (relatedWorkspace) {
this.delete(url, relatedWorkspace);
}
}
for (const remote of addedRemotes) {
const url = remote.fetchUrl;
if (!url) {
continue;
}
this.set(url, rootPath);
}
}
/**
* We should possibly support converting between ssh remotes and http remotes.
*/
get(repoUrl: string): RepositoryCacheInfo[] | undefined {
const key = this.normalizeRepoUrl(repoUrl);
const inner = this.lru.get(key);
return inner ? Array.from(inner.values()) : undefined;
}
delete(repoUrl: string, folderPathOrWorkspaceFile: string) {
const key = this.normalizeRepoUrl(repoUrl);
const inner = this.lru.get(key);
if (!inner) {
return;
}
if (!inner.remove(folderPathOrWorkspaceFile)) {
return;
}
if (inner.size === 0) {
this.lru.remove(key);
} else {
// Re-set to bump outer LRU recency after modification
this.lru.set(key, inner);
}
this.save();
}
private load(): void {
try {
const raw = this._globalState.get<[string, [string, RepositoryCacheInfo][]][]>(RepositoryCache.STORAGE_KEY);
if (!Array.isArray(raw)) {
return;
}
for (const [repo, storedFolders] of raw) {
if (typeof repo !== 'string' || !Array.isArray(storedFolders)) {
continue;
}
const inner = new LRUCache<string, RepositoryCacheInfo>(RepositoryCache.MAX_FOLDER_ENTRIES);
for (const entry of storedFolders) {
if (!Array.isArray(entry) || entry.length !== 2) {
continue;
}
const [folderPath, info] = entry;
if (typeof folderPath !== 'string' || !isRepositoryCacheInfo(info)) {
continue;
}
inner.set(folderPath, info);
}
if (inner.size) {
this.lru.set(repo, inner);
}
}
} catch {
this._logger.warn('[CachedRepositories][load] Failed to load cached repositories from global state.');
}
}
private save(): void {
// Serialize as [repoUrl, [folderPathOrWorkspaceFile, RepositoryCacheInfo][]] preserving outer LRU order.
const serialized: [string, [string, RepositoryCacheInfo][]][] = [];
for (const [repo, inner] of this.lru) {
const folders: [string, RepositoryCacheInfo][] = [];
for (const [folder, info] of inner) {
folders.push([folder, info]);
}
serialized.push([repo, folders]);
}
void this._globalState.update(RepositoryCache.STORAGE_KEY, serialized);
}
}