Path: blob/main/src/vs/workbench/services/configuration/common/jsonEditingService.ts
3296 views
/*---------------------------------------------------------------------------------------------1* Copyright (c) Microsoft Corporation. All rights reserved.2* Licensed under the MIT License. See License.txt in the project root for license information.3*--------------------------------------------------------------------------------------------*/45import * as nls from '../../../../nls.js';6import { URI } from '../../../../base/common/uri.js';7import * as json from '../../../../base/common/json.js';8import { setProperty } from '../../../../base/common/jsonEdit.js';9import { Queue } from '../../../../base/common/async.js';10import { Edit } from '../../../../base/common/jsonFormatter.js';11import { IDisposable, IReference } from '../../../../base/common/lifecycle.js';12import { EditOperation } from '../../../../editor/common/core/editOperation.js';13import { Range } from '../../../../editor/common/core/range.js';14import { Selection } from '../../../../editor/common/core/selection.js';15import { ITextFileService } from '../../textfile/common/textfiles.js';16import { IFileService } from '../../../../platform/files/common/files.js';17import { ITextModelService, IResolvedTextEditorModel } from '../../../../editor/common/services/resolverService.js';18import { IJSONEditingService, IJSONValue, JSONEditingError, JSONEditingErrorCode } from './jsonEditing.js';19import { ITextModel } from '../../../../editor/common/model.js';20import { InstantiationType, registerSingleton } from '../../../../platform/instantiation/common/extensions.js';21import { IFilesConfigurationService } from '../../filesConfiguration/common/filesConfigurationService.js';2223export class JSONEditingService implements IJSONEditingService {2425public _serviceBrand: undefined;2627private queue: Queue<void>;2829constructor(30@IFileService private readonly fileService: IFileService,31@ITextModelService private readonly textModelResolverService: ITextModelService,32@ITextFileService private readonly textFileService: ITextFileService,33@IFilesConfigurationService private readonly filesConfigurationService: IFilesConfigurationService34) {35this.queue = new Queue<void>();36}3738write(resource: URI, values: IJSONValue[]): Promise<void> {39return Promise.resolve(this.queue.queue(() => this.doWriteConfiguration(resource, values))); // queue up writes to prevent race conditions40}4142private async doWriteConfiguration(resource: URI, values: IJSONValue[]): Promise<void> {43const reference = await this.resolveAndValidate(resource, true);44try {45await this.writeToBuffer(reference.object.textEditorModel, values);46} finally {47reference.dispose();48}49}5051private async writeToBuffer(model: ITextModel, values: IJSONValue[]): Promise<any> {52let disposable: IDisposable | undefined;53try {54// Optimization: we apply edits to a text model and save it55// right after. Use the files config service to signal this56// to the workbench to optimise the UI during this operation.57// For example, avoids to briefly show dirty indicators.58disposable = this.filesConfigurationService.enableAutoSaveAfterShortDelay(model.uri);5960let hasEdits: boolean = false;61for (const value of values) {62const edit = this.getEdits(model, value)[0];63hasEdits = (!!edit && this.applyEditsToBuffer(edit, model)) || hasEdits;64}65if (hasEdits) {66return this.textFileService.save(model.uri);67}68} finally {69disposable?.dispose();70}71}7273private applyEditsToBuffer(edit: Edit, model: ITextModel): boolean {74const startPosition = model.getPositionAt(edit.offset);75const endPosition = model.getPositionAt(edit.offset + edit.length);76const range = new Range(startPosition.lineNumber, startPosition.column, endPosition.lineNumber, endPosition.column);77const currentText = model.getValueInRange(range);78if (edit.content !== currentText) {79const editOperation = currentText ? EditOperation.replace(range, edit.content) : EditOperation.insert(startPosition, edit.content);80model.pushEditOperations([new Selection(startPosition.lineNumber, startPosition.column, startPosition.lineNumber, startPosition.column)], [editOperation], () => []);81return true;82}83return false;84}8586private getEdits(model: ITextModel, configurationValue: IJSONValue): Edit[] {87const { tabSize, insertSpaces } = model.getOptions();88const eol = model.getEOL();89const { path, value } = configurationValue;9091// With empty path the entire file is being replaced, so we just use JSON.stringify92if (!path.length) {93const content = JSON.stringify(value, null, insertSpaces ? ' '.repeat(tabSize) : '\t');94return [{95content,96length: content.length,97offset: 098}];99}100101return setProperty(model.getValue(), path, value, { tabSize, insertSpaces, eol });102}103104private async resolveModelReference(resource: URI): Promise<IReference<IResolvedTextEditorModel>> {105const exists = await this.fileService.exists(resource);106if (!exists) {107await this.textFileService.write(resource, '{}', { encoding: 'utf8' });108}109return this.textModelResolverService.createModelReference(resource);110}111112private hasParseErrors(model: ITextModel): boolean {113const parseErrors: json.ParseError[] = [];114json.parse(model.getValue(), parseErrors, { allowTrailingComma: true, allowEmptyContent: true });115return parseErrors.length > 0;116}117118private async resolveAndValidate(resource: URI, checkDirty: boolean): Promise<IReference<IResolvedTextEditorModel>> {119const reference = await this.resolveModelReference(resource);120121const model = reference.object.textEditorModel;122123if (this.hasParseErrors(model)) {124reference.dispose();125return this.reject<IReference<IResolvedTextEditorModel>>(JSONEditingErrorCode.ERROR_INVALID_FILE);126}127128return reference;129}130131private reject<T>(code: JSONEditingErrorCode): Promise<T> {132const message = this.toErrorMessage(code);133return Promise.reject(new JSONEditingError(message, code));134}135136private toErrorMessage(error: JSONEditingErrorCode): string {137switch (error) {138// User issues139case JSONEditingErrorCode.ERROR_INVALID_FILE: {140return nls.localize('errorInvalidFile', "Unable to write into the file. Please open the file to correct errors/warnings in the file and try again.");141}142}143}144}145146registerSingleton(IJSONEditingService, JSONEditingService, InstantiationType.Delayed);147148149