/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the Source EULA. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import { Model } from '../model'; import { Repository as BaseRepository, Resource } from '../repository'; import { InputBox, Git, API, Repository, Remote, RepositoryState, Branch, ForcePushMode, Ref, Submodule, Commit, Change, RepositoryUIState, Status, LogOptions, APIState, CommitOptions, RefType, CredentialsProvider, BranchQuery, PushErrorHandler, PublishEvent, FetchOptions, RemoteSourceProvider, RemoteSourcePublisher, ICloneOptions, PostCommitCommandsProvider } from './git'; // {{SQL CARBON EDIT}} add ICloneOptions import { Event, SourceControlInputBox, Uri, SourceControl, Disposable, commands, CancellationToken } from 'vscode'; // {{SQL CARBON EDIT}} Add cancellationToken import { combinedDisposable, mapEvent } from '../util'; import { toGitUri } from '../uri'; import { GitExtensionImpl } from './extension'; import { GitBaseApi } from '../git-base'; import { PickRemoteSourceOptions } from './git-base'; class ApiInputBox implements InputBox { set value(value: string) { this._inputBox.value = value; } get value(): string { return this._inputBox.value; } constructor(private _inputBox: SourceControlInputBox) { } } export class ApiChange implements Change { get uri(): Uri { return this.resource.resourceUri; } get originalUri(): Uri { return this.resource.original; } get renameUri(): Uri | undefined { return this.resource.renameResourceUri; } get status(): Status { return this.resource.type; } constructor(private readonly resource: Resource) { } } export class ApiRepositoryState implements RepositoryState { get HEAD(): Branch | undefined { return this._repository.HEAD; } get refs(): Ref[] { return [...this._repository.refs]; } get remotes(): Remote[] { return [...this._repository.remotes]; } get submodules(): Submodule[] { return [...this._repository.submodules]; } get rebaseCommit(): Commit | undefined { return this._repository.rebaseCommit; } get mergeChanges(): Change[] { return this._repository.mergeGroup.resourceStates.map(r => new ApiChange(r)); } get indexChanges(): Change[] { return this._repository.indexGroup.resourceStates.map(r => new ApiChange(r)); } get workingTreeChanges(): Change[] { return this._repository.workingTreeGroup.resourceStates.map(r => new ApiChange(r)); } readonly onDidChange: Event = this._repository.onDidRunGitStatus; constructor(private _repository: BaseRepository) { } } export class ApiRepositoryUIState implements RepositoryUIState { get selected(): boolean { return this._sourceControl.selected; } readonly onDidChange: Event = mapEvent(this._sourceControl.onDidChangeSelection, () => null); constructor(private _sourceControl: SourceControl) { } } export class ApiRepository implements Repository { readonly rootUri: Uri = Uri.file(this.repository.root); readonly inputBox: InputBox = new ApiInputBox(this.repository.inputBox); readonly state: RepositoryState = new ApiRepositoryState(this.repository); readonly ui: RepositoryUIState = new ApiRepositoryUIState(this.repository.sourceControl); constructor(readonly repository: BaseRepository) { } apply(patch: string, reverse?: boolean): Promise { return this.repository.apply(patch, reverse); } getConfigs(): Promise<{ key: string; value: string }[]> { return this.repository.getConfigs(); } getConfig(key: string): Promise { return this.repository.getConfig(key); } setConfig(key: string, value: string): Promise { return this.repository.setConfig(key, value); } getGlobalConfig(key: string): Promise { return this.repository.getGlobalConfig(key); } getObjectDetails(treeish: string, path: string): Promise<{ mode: string; object: string; size: number }> { return this.repository.getObjectDetails(treeish, path); } detectObjectType(object: string): Promise<{ mimetype: string; encoding?: string }> { return this.repository.detectObjectType(object); } buffer(ref: string, filePath: string): Promise { return this.repository.buffer(ref, filePath); } show(ref: string, path: string): Promise { return this.repository.show(ref, path); } getCommit(ref: string): Promise { return this.repository.getCommit(ref); } add(paths: string[]) { return this.repository.add(paths.map(p => Uri.file(p))); } revert(paths: string[]) { return this.repository.revert(paths.map(p => Uri.file(p))); } clean(paths: string[]) { return this.repository.clean(paths.map(p => Uri.file(p))); } diff(cached?: boolean) { return this.repository.diff(cached); } diffWithHEAD(): Promise; diffWithHEAD(path: string): Promise; diffWithHEAD(path?: string): Promise { return this.repository.diffWithHEAD(path); } diffWith(ref: string): Promise; diffWith(ref: string, path: string): Promise; diffWith(ref: string, path?: string): Promise { return this.repository.diffWith(ref, path); } diffIndexWithHEAD(): Promise; diffIndexWithHEAD(path: string): Promise; diffIndexWithHEAD(path?: string): Promise { return this.repository.diffIndexWithHEAD(path); } diffIndexWith(ref: string): Promise; diffIndexWith(ref: string, path: string): Promise; diffIndexWith(ref: string, path?: string): Promise { return this.repository.diffIndexWith(ref, path); } diffBlobs(object1: string, object2: string): Promise { return this.repository.diffBlobs(object1, object2); } diffBetween(ref1: string, ref2: string): Promise; diffBetween(ref1: string, ref2: string, path: string): Promise; diffBetween(ref1: string, ref2: string, path?: string): Promise { return this.repository.diffBetween(ref1, ref2, path); } hashObject(data: string): Promise { return this.repository.hashObject(data); } createBranch(name: string, checkout: boolean, ref?: string | undefined): Promise { return this.repository.branch(name, checkout, ref); } deleteBranch(name: string, force?: boolean): Promise { return this.repository.deleteBranch(name, force); } getBranch(name: string): Promise { return this.repository.getBranch(name); } getBranches(query: BranchQuery): Promise { return this.repository.getBranches(query); } setBranchUpstream(name: string, upstream: string): Promise { return this.repository.setBranchUpstream(name, upstream); } getMergeBase(ref1: string, ref2: string): Promise { return this.repository.getMergeBase(ref1, ref2); } tag(name: string, upstream: string): Promise { return this.repository.tag(name, upstream); } deleteTag(name: string): Promise { return this.repository.deleteTag(name); } status(): Promise { return this.repository.status(); } checkout(treeish: string): Promise { return this.repository.checkout(treeish); } addRemote(name: string, url: string): Promise { return this.repository.addRemote(name, url); } removeRemote(name: string): Promise { return this.repository.removeRemote(name); } renameRemote(name: string, newName: string): Promise { return this.repository.renameRemote(name, newName); } fetch(arg0?: FetchOptions | string | undefined, ref?: string | undefined, depth?: number | undefined, prune?: boolean | undefined ): Promise { if (arg0 !== undefined && typeof arg0 !== 'string') { return this.repository.fetch(arg0); } return this.repository.fetch({ remote: arg0, ref, depth, prune }); } pull(unshallow?: boolean): Promise { return this.repository.pull(undefined, unshallow); } push(remoteName?: string, branchName?: string, setUpstream: boolean = false, force?: ForcePushMode): Promise { return this.repository.pushTo(remoteName, branchName, setUpstream, force); } blame(path: string): Promise { return this.repository.blame(path); } log(options?: LogOptions): Promise { return this.repository.log(options); } commit(message: string, opts?: CommitOptions): Promise { return this.repository.commit(message, opts); } } export class ApiGit implements Git { get path(): string { return this._model.git.path; } constructor(private _model: Model) { } } export class ApiImpl implements API { readonly git = new ApiGit(this._model); get state(): APIState { return this._model.state; } get onDidChangeState(): Event { return this._model.onDidChangeState; } get onDidPublish(): Event { return this._model.onDidPublish; } get onDidOpenRepository(): Event { return mapEvent(this._model.onDidOpenRepository, r => new ApiRepository(r)); } get onDidCloseRepository(): Event { return mapEvent(this._model.onDidCloseRepository, r => new ApiRepository(r)); } get repositories(): Repository[] { return this._model.repositories.map(r => new ApiRepository(r)); } // {{SQL CARBON EDIT}} async clone(url: string, options: ICloneOptions, cancellationToken?: CancellationToken): Promise { return this._model.git.clone(url, options, cancellationToken); } toGitUri(uri: Uri, ref: string): Uri { return toGitUri(uri, ref); } getRepository(uri: Uri): Repository | null { const result = this._model.getRepository(uri); return result ? new ApiRepository(result) : null; } async init(root: Uri): Promise { const path = root.fsPath; await this._model.git.init(path); await this._model.openRepository(path); return this.getRepository(root) || null; } async openRepository(root: Uri): Promise { await this._model.openRepository(root.fsPath); return this.getRepository(root) || null; } registerRemoteSourceProvider(provider: RemoteSourceProvider): Disposable { const disposables: Disposable[] = []; if (provider.publishRepository) { disposables.push(this._model.registerRemoteSourcePublisher(provider as RemoteSourcePublisher)); } disposables.push(GitBaseApi.getAPI().registerRemoteSourceProvider(provider)); return combinedDisposable(disposables); } registerRemoteSourcePublisher(publisher: RemoteSourcePublisher): Disposable { return this._model.registerRemoteSourcePublisher(publisher); } registerCredentialsProvider(provider: CredentialsProvider): Disposable { return this._model.registerCredentialsProvider(provider); } registerPostCommitCommandsProvider(provider: PostCommitCommandsProvider): Disposable { return this._model.registerPostCommitCommandsProvider(provider); } registerPushErrorHandler(handler: PushErrorHandler): Disposable { return this._model.registerPushErrorHandler(handler); } constructor(private _model: Model) { } } function getRefType(type: RefType): string { switch (type) { case RefType.Head: return 'Head'; case RefType.RemoteHead: return 'RemoteHead'; case RefType.Tag: return 'Tag'; } return 'unknown'; } function getStatus(status: Status): string { switch (status) { case Status.INDEX_MODIFIED: return 'INDEX_MODIFIED'; case Status.INDEX_ADDED: return 'INDEX_ADDED'; case Status.INDEX_DELETED: return 'INDEX_DELETED'; case Status.INDEX_RENAMED: return 'INDEX_RENAMED'; case Status.INDEX_COPIED: return 'INDEX_COPIED'; case Status.MODIFIED: return 'MODIFIED'; case Status.DELETED: return 'DELETED'; case Status.UNTRACKED: return 'UNTRACKED'; case Status.IGNORED: return 'IGNORED'; case Status.INTENT_TO_ADD: return 'INTENT_TO_ADD'; case Status.ADDED_BY_US: return 'ADDED_BY_US'; case Status.ADDED_BY_THEM: return 'ADDED_BY_THEM'; case Status.DELETED_BY_US: return 'DELETED_BY_US'; case Status.DELETED_BY_THEM: return 'DELETED_BY_THEM'; case Status.BOTH_ADDED: return 'BOTH_ADDED'; case Status.BOTH_DELETED: return 'BOTH_DELETED'; case Status.BOTH_MODIFIED: return 'BOTH_MODIFIED'; } return 'UNKNOWN'; } export function registerAPICommands(extension: GitExtensionImpl): Disposable { const disposables: Disposable[] = []; disposables.push(commands.registerCommand('git.api.getRepositories', () => { const api = extension.getAPI(1); return api.repositories.map(r => r.rootUri.toString()); })); disposables.push(commands.registerCommand('git.api.getRepositoryState', (uri: string) => { const api = extension.getAPI(1); const repository = api.getRepository(Uri.parse(uri)); if (!repository) { return null; } const state = repository.state; const ref = (ref: Ref | undefined) => (ref && { ...ref, type: getRefType(ref.type) }); const change = (change: Change) => ({ uri: change.uri.toString(), originalUri: change.originalUri.toString(), renameUri: change.renameUri?.toString(), status: getStatus(change.status) }); return { HEAD: ref(state.HEAD), refs: state.refs.map(ref), remotes: state.remotes, submodules: state.submodules, rebaseCommit: state.rebaseCommit, mergeChanges: state.mergeChanges.map(change), indexChanges: state.indexChanges.map(change), workingTreeChanges: state.workingTreeChanges.map(change) }; })); disposables.push(commands.registerCommand('git.api.getRemoteSources', (opts?: PickRemoteSourceOptions) => { return commands.executeCommand('git-base.api.getRemoteSources', opts); })); return Disposable.from(...disposables); }