/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ 'use strict'; import * as assert from 'assert'; import * as platform from 'vs/base/common/platform'; import crypto = require('crypto'); import os = require('os'); import fs = require('fs'); import path = require('path'); import extfs = require('vs/base/node/extfs'); import pfs = require('vs/base/node/pfs'); import Uri from 'vs/base/common/uri'; import { BackupFileService, BackupFilesModel } from 'vs/workbench/services/backup/node/backupFileService'; import { FileService } from 'vs/workbench/services/files/node/fileService'; import { TextModel } from 'vs/editor/common/model/textModel'; import { TestContextService, TestTextResourceConfigurationService, getRandomTestPath, TestLifecycleService } from 'vs/workbench/test/workbenchTestServices'; import { Workspace, toWorkspaceFolders } from 'vs/platform/workspace/common/workspace'; import { TestConfigurationService } from 'vs/platform/configuration/test/common/testConfigurationService'; import { DefaultEndOfLine } from 'vs/editor/common/model'; import { snapshotToString } from 'vs/platform/files/common/files'; const parentDir = getRandomTestPath(os.tmpdir(), 'vsctests', 'backupfileservice'); const backupHome = path.join(parentDir, 'Backups'); const workspacesJsonPath = path.join(backupHome, 'workspaces.json'); const workspaceResource = Uri.file(platform.isWindows ? 'c:\\workspace' : '/workspace'); const workspaceBackupPath = path.join(backupHome, crypto.createHash('md5').update(workspaceResource.fsPath).digest('hex')); const fooFile = Uri.file(platform.isWindows ? 'c:\\Foo' : '/Foo'); const barFile = Uri.file(platform.isWindows ? 'c:\\Bar' : '/Bar'); const untitledFile = Uri.from({ scheme: 'untitled', path: 'Untitled-1' }); const fooBackupPath = path.join(workspaceBackupPath, 'file', crypto.createHash('md5').update(fooFile.fsPath).digest('hex')); const barBackupPath = path.join(workspaceBackupPath, 'file', crypto.createHash('md5').update(barFile.fsPath).digest('hex')); const untitledBackupPath = path.join(workspaceBackupPath, 'untitled', crypto.createHash('md5').update(untitledFile.fsPath).digest('hex')); class TestBackupFileService extends BackupFileService { constructor(workspace: Uri, backupHome: string, workspacesJsonPath: string) { const fileService = new FileService(new TestContextService(new Workspace(workspace.fsPath, workspace.fsPath, toWorkspaceFolders([{ path: workspace.fsPath }]))), new TestTextResourceConfigurationService(), new TestConfigurationService(), new TestLifecycleService(), { disableWatcher: true }); super(workspaceBackupPath, fileService); } public toBackupResource(resource: Uri): Uri { return super.toBackupResource(resource); } } suite('BackupFileService', () => { let service: TestBackupFileService; setup(done => { service = new TestBackupFileService(workspaceResource, backupHome, workspacesJsonPath); // Delete any existing backups completely and then re-create it. extfs.del(backupHome, os.tmpdir(), () => { pfs.mkdirp(backupHome).then(() => { pfs.writeFile(workspacesJsonPath, '').then(() => { done(); }); }); }); }); teardown(done => { extfs.del(backupHome, os.tmpdir(), done); }); suite('getBackupResource', () => { test('should get the correct backup path for text files', () => { // Format should be: /// const backupResource = fooFile; const workspaceHash = crypto.createHash('md5').update(workspaceResource.fsPath).digest('hex'); const filePathHash = crypto.createHash('md5').update(backupResource.fsPath).digest('hex'); const expectedPath = Uri.file(path.join(backupHome, workspaceHash, 'file', filePathHash)).fsPath; assert.equal(service.toBackupResource(backupResource).fsPath, expectedPath); }); test('should get the correct backup path for untitled files', () => { // Format should be: /// const backupResource = Uri.from({ scheme: 'untitled', path: 'Untitled-1' }); const workspaceHash = crypto.createHash('md5').update(workspaceResource.fsPath).digest('hex'); const filePathHash = crypto.createHash('md5').update(backupResource.fsPath).digest('hex'); const expectedPath = Uri.file(path.join(backupHome, workspaceHash, 'untitled', filePathHash)).fsPath; assert.equal(service.toBackupResource(backupResource).fsPath, expectedPath); }); }); suite('loadBackupResource', () => { test('should return whether a backup resource exists', done => { pfs.mkdirp(path.dirname(fooBackupPath)).then(() => { fs.writeFileSync(fooBackupPath, 'foo'); service = new TestBackupFileService(workspaceResource, backupHome, workspacesJsonPath); service.loadBackupResource(fooFile).then(resource => { assert.ok(resource); assert.equal(path.basename(resource.fsPath), path.basename(fooBackupPath)); return service.hasBackups().then(hasBackups => { assert.ok(hasBackups); done(); }); }); }); }); }); suite('backupResource', () => { test('text file', function (done: () => void) { service.backupResource(fooFile, 'test').then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'file')).length, 1); assert.equal(fs.existsSync(fooBackupPath), true); assert.equal(fs.readFileSync(fooBackupPath), `${fooFile.toString()}\ntest`); done(); }); }); test('untitled file', function (done: () => void) { service.backupResource(untitledFile, 'test').then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'untitled')).length, 1); assert.equal(fs.existsSync(untitledBackupPath), true); assert.equal(fs.readFileSync(untitledBackupPath), `${untitledFile.toString()}\ntest`); done(); }); }); test('text file (ITextSnapshot)', function (done: () => void) { const model = TextModel.createFromString('test'); service.backupResource(fooFile, model.createSnapshot()).then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'file')).length, 1); assert.equal(fs.existsSync(fooBackupPath), true); assert.equal(fs.readFileSync(fooBackupPath), `${fooFile.toString()}\ntest`); model.dispose(); done(); }); }); test('untitled file (ITextSnapshot)', function (done: () => void) { const model = TextModel.createFromString('test'); service.backupResource(untitledFile, model.createSnapshot()).then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'untitled')).length, 1); assert.equal(fs.existsSync(untitledBackupPath), true); assert.equal(fs.readFileSync(untitledBackupPath), `${untitledFile.toString()}\ntest`); model.dispose(); done(); }); }); test('text file (large file, ITextSnapshot)', function (done: () => void) { const largeString = (new Array(10 * 1024)).join('Large String\n'); const model = TextModel.createFromString(largeString); service.backupResource(fooFile, model.createSnapshot()).then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'file')).length, 1); assert.equal(fs.existsSync(fooBackupPath), true); assert.equal(fs.readFileSync(fooBackupPath), `${fooFile.toString()}\n${largeString}`); model.dispose(); done(); }); }); test('untitled file (large file, ITextSnapshot)', function (done: () => void) { const largeString = (new Array(10 * 1024)).join('Large String\n'); const model = TextModel.createFromString(largeString); service.backupResource(untitledFile, model.createSnapshot()).then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'untitled')).length, 1); assert.equal(fs.existsSync(untitledBackupPath), true); assert.equal(fs.readFileSync(untitledBackupPath), `${untitledFile.toString()}\n${largeString}`); model.dispose(); done(); }); }); }); suite('discardResourceBackup', () => { test('text file', function (done: () => void) { service.backupResource(fooFile, 'test').then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'file')).length, 1); service.discardResourceBackup(fooFile).then(() => { assert.equal(fs.existsSync(fooBackupPath), false); assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'file')).length, 0); done(); }); }); }); test('untitled file', function (done: () => void) { service.backupResource(untitledFile, 'test').then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'untitled')).length, 1); service.discardResourceBackup(untitledFile).then(() => { assert.equal(fs.existsSync(untitledBackupPath), false); assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'untitled')).length, 0); done(); }); }); }); }); suite('discardAllWorkspaceBackups', () => { test('text file', function (done: () => void) { service.backupResource(fooFile, 'test').then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'file')).length, 1); service.backupResource(barFile, 'test').then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'file')).length, 2); service.discardAllWorkspaceBackups().then(() => { assert.equal(fs.existsSync(fooBackupPath), false); assert.equal(fs.existsSync(barBackupPath), false); assert.equal(fs.existsSync(path.join(workspaceBackupPath, 'file')), false); done(); }); }); }); }); test('untitled file', function (done: () => void) { service.backupResource(untitledFile, 'test').then(() => { assert.equal(fs.readdirSync(path.join(workspaceBackupPath, 'untitled')).length, 1); service.discardAllWorkspaceBackups().then(() => { assert.equal(fs.existsSync(untitledBackupPath), false); assert.equal(fs.existsSync(path.join(workspaceBackupPath, 'untitled')), false); done(); }); }); }); test('should disable further backups', function (done: () => void) { service.discardAllWorkspaceBackups().then(() => { service.backupResource(untitledFile, 'test').then(() => { assert.equal(fs.existsSync(workspaceBackupPath), false); done(); }); }); }); }); suite('getWorkspaceFileBackups', () => { test('("file") - text file', done => { service.backupResource(fooFile, `test`).then(() => { service.getWorkspaceFileBackups().then(textFiles => { assert.deepEqual(textFiles.map(f => f.fsPath), [fooFile.fsPath]); service.backupResource(barFile, `test`).then(() => { service.getWorkspaceFileBackups().then(textFiles => { assert.deepEqual(textFiles.map(f => f.fsPath), [fooFile.fsPath, barFile.fsPath]); done(); }); }); }); }); }); test('("file") - untitled file', done => { service.backupResource(untitledFile, `test`).then(() => { service.getWorkspaceFileBackups().then(textFiles => { assert.deepEqual(textFiles.map(f => f.fsPath), [untitledFile.fsPath]); done(); }); }); }); test('("untitled") - untitled file', done => { service.backupResource(untitledFile, `test`).then(() => { service.getWorkspaceFileBackups().then(textFiles => { assert.deepEqual(textFiles.map(f => f.fsPath), ['Untitled-1']); done(); }); }); }); }); test('resolveBackupContent', () => { test('should restore the original contents (untitled file)', () => { const contents = 'test\nand more stuff'; service.backupResource(untitledFile, contents).then(() => { service.resolveBackupContent(service.toBackupResource(untitledFile)).then(factory => { assert.equal(contents, snapshotToString(factory.create(platform.isWindows ? DefaultEndOfLine.CRLF : DefaultEndOfLine.LF).createSnapshot(true))); }); }); }); test('should restore the original contents (text file)', () => { const contents = [ 'Lorem ipsum ', 'dolor öäü sit amet ', 'consectetur ', 'adipiscing ßß elit', ].join(''); service.backupResource(fooFile, contents).then(() => { service.resolveBackupContent(service.toBackupResource(untitledFile)).then(factory => { assert.equal(contents, snapshotToString(factory.create(platform.isWindows ? DefaultEndOfLine.CRLF : DefaultEndOfLine.LF).createSnapshot(true))); }); }); }); }); }); suite('BackupFilesModel', () => { test('simple', () => { const model = new BackupFilesModel(); const resource1 = Uri.file('test.html'); assert.equal(model.has(resource1), false); model.add(resource1); assert.equal(model.has(resource1), true); assert.equal(model.has(resource1, 0), true); assert.equal(model.has(resource1, 1), false); model.remove(resource1); assert.equal(model.has(resource1), false); model.add(resource1); assert.equal(model.has(resource1), true); assert.equal(model.has(resource1, 0), true); assert.equal(model.has(resource1, 1), false); model.clear(); assert.equal(model.has(resource1), false); model.add(resource1, 1); assert.equal(model.has(resource1), true); assert.equal(model.has(resource1, 0), false); assert.equal(model.has(resource1, 1), true); const resource2 = Uri.file('test1.html'); const resource3 = Uri.file('test2.html'); const resource4 = Uri.file('test3.html'); model.add(resource2); model.add(resource3); model.add(resource4); assert.equal(model.has(resource1), true); assert.equal(model.has(resource2), true); assert.equal(model.has(resource3), true); assert.equal(model.has(resource4), true); }); test('resolve', (done) => { pfs.mkdirp(path.dirname(fooBackupPath)).then(() => { fs.writeFileSync(fooBackupPath, 'foo'); const model = new BackupFilesModel(); model.resolve(workspaceBackupPath).then(model => { assert.equal(model.has(Uri.file(fooBackupPath)), true); done(); }); }); }); test('get', () => { const model = new BackupFilesModel(); assert.deepEqual(model.get(), []); const file1 = Uri.file('/root/file/foo.html'); const file2 = Uri.file('/root/file/bar.html'); const untitled = Uri.file('/root/untitled/bar.html'); model.add(file1); model.add(file2); model.add(untitled); assert.deepEqual(model.get().map(f => f.fsPath), [file1.fsPath, file2.fsPath, untitled.fsPath]); }); });