/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the Source EULA. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ 'use strict'; import { Observable } from 'rxjs/Observable'; import { Subject } from 'rxjs/Subject'; import { Observer } from 'rxjs/Observer'; import { ResultSetSubset, EditUpdateCellResult, EditSubsetResult, EditCreateRowResult } from 'azdata'; import { IQueryModelService } from 'sql/platform/query/common/queryModel'; import { ResultSerializer } from 'sql/platform/node/resultSerializer'; import { ISaveRequest } from 'sql/parts/grid/common/interfaces'; import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation'; import { IQueryEditorService } from 'sql/workbench/services/queryEditor/common/queryEditorService'; /** * DataService handles the interactions between QueryModel and app.component. Thus, it handles * query running and grid interaction communication for a single URI. */ export class DataService { public queryEventObserver: Subject; public gridContentObserver: Subject; private editQueue: Promise; constructor( private _uri: string, @IInstantiationService private _instantiationService: IInstantiationService, @IQueryModelService private _queryModel: IQueryModelService, @IQueryEditorService private _queryEditorService: IQueryEditorService ) { this.queryEventObserver = new Subject(); this.gridContentObserver = new Subject(); this.editQueue = Promise.resolve(); } /** * Get a specified number of rows starting at a specified row for * the current results set. Used for query results only. * @param start The starting row or the requested rows * @param numberOfRows The amount of rows to return * @param batchId The batch id of the batch you are querying * @param resultId The id of the result you want to get the rows for */ getQueryRows(rowStart: number, numberOfRows: number, batchId: number, resultId: number): Observable { const self = this; return Observable.create(function (observer: Observer) { self._queryModel.getQueryRows(self._uri, rowStart, numberOfRows, batchId, resultId).then(results => { observer.next(results); }); }); } /** * Get a specified number of rows starting at a specified row. Should only * be used for edit sessions. * @param rowStart The row to start retrieving from (inclusive) * @param numberOfRows The maximum number of rows to return */ getEditRows(rowStart: number, numberOfRows: number): Observable { const self = this; return Observable.create(function (observer: Observer) { self._queryModel.getEditRows(self._uri, rowStart, numberOfRows).then(results => { observer.next(results); }); }); } updateCell(rowId: number, columnId: number, newValue: string): Thenable { const self = this; self.editQueue = self.editQueue.then(() => { return self._queryModel.updateCell(self._uri, rowId, columnId, newValue).then(result => { return result; }, error => { // Start our editQueue over due to the rejected promise self.editQueue = Promise.resolve(); return Promise.reject(error); }); }); return self.editQueue; } commitEdit(): Thenable { const self = this; self.editQueue = self.editQueue.then(() => { return self._queryModel.commitEdit(self._uri).then(result => { return result; }, error => { // Start our editQueue over due to the rejected promise self.editQueue = Promise.resolve(); return Promise.reject(error); }); }); return self.editQueue; } createRow(): Thenable { const self = this; self.editQueue = self.editQueue.then(() => { return self._queryModel.createRow(self._uri).then(result => { return result; }, error => { // Start our editQueue over due to the rejected promise self.editQueue = Promise.resolve(); return Promise.reject(error); }); }); return self.editQueue; } deleteRow(rowId: number): Thenable { const self = this; self.editQueue = self.editQueue.then(() => { return self._queryModel.deleteRow(self._uri, rowId).then(result => { return result; }, error => { // Start our editQueue over due to the rejected promise self.editQueue = Promise.resolve(); self._queryModel.showCommitError(error.message); return Promise.reject(error); }); }); return self.editQueue; } revertCell(rowId: number, columnId: number): Thenable { const self = this; self.editQueue = self.editQueue.then(() => { return self._queryModel.revertCell(self._uri, rowId, columnId).then(result => { return result; }, error => { // Start our editQueue over due to the rejected promise self.editQueue = Promise.resolve(); return Promise.reject(error); }); }); return self.editQueue; } revertRow(rowId: number): Thenable { const self = this; self.editQueue = self.editQueue.then(() => { return self._queryModel.revertRow(self._uri, rowId).then(result => { return result; }, error => { // Start our editQueue over due to the rejected promise self.editQueue = Promise.resolve(); return Promise.reject(error); }); }); return self.editQueue; } /** * send request to save the selected result set as csv * @param uri of the calling document * @param batchId The batch id of the batch with the result to save * @param resultId The id of the result to save as csv */ sendSaveRequest(saveRequest: ISaveRequest): void { let serializer = this._instantiationService.createInstance(ResultSerializer); serializer.saveResults(this._uri, saveRequest); } /** * send request to open content in new editor * @param content The content to be opened * @param columnName The column name of the content */ openLink(content: string, columnName: string, linkType: string): void { let serializer = this._instantiationService.createInstance(ResultSerializer); serializer.openLink(content, columnName, linkType); } /** * Sends a copy request * @param selection The selection range to copy * @param batchId The batch id of the result to copy from * @param resultId The result id of the result to copy from * @param includeHeaders [Optional]: Should column headers be included in the copy selection */ copyResults(selection: Slick.Range[], batchId: number, resultId: number, includeHeaders?: boolean): void { this._queryModel.copyResults(this._uri, selection, batchId, resultId, includeHeaders); } /** * Sends a request to set the selection in the QueryEditor. */ setEditorSelection(index: number) { this._queryModel.setEditorSelection(this._uri, index); } showWarning(message: string): void { } showError(message: string): void { } get config(): Promise<{ [key: string]: any }> { return undefined; } onAngularLoaded(): void { this._queryModel.onAngularLoaded(this._uri); } }