Improve FileSearchProvider (exclude grob support + no command)

feature/search
Kelvin Schoofs 6 years ago
parent 119f2bdc1e
commit 543c95c99a

14
package-lock.json generated

@ -25,6 +25,12 @@
"@types/node": "*"
}
},
"@types/minimatch": {
"version": "3.0.3",
"resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz",
"integrity": "sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA==",
"dev": true
},
"@types/node": {
"version": "8.10.43",
"resolved": "https://registry.npmjs.org/@types/node/-/node-8.10.43.tgz",
@ -483,8 +489,7 @@
"balanced-match": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
"integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=",
"dev": true
"integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c="
},
"base": {
"version": "0.11.2",
@ -589,7 +594,6 @@
"version": "1.1.11",
"resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
"integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
"dev": true,
"requires": {
"balanced-match": "^1.0.0",
"concat-map": "0.0.1"
@ -987,8 +991,7 @@
"concat-map": {
"version": "0.0.1",
"resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
"integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
"dev": true
"integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s="
},
"concat-stream": {
"version": "1.6.2",
@ -3083,7 +3086,6 @@
"version": "3.0.4",
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
"integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
"dev": true,
"requires": {
"brace-expansion": "^1.1.7"
}

@ -437,6 +437,7 @@
"postinstall": "node ./node_modules/vscode/bin/install"
},
"devDependencies": {
"@types/minimatch": "^3.0.3",
"@types/node": "^8.10.43",
"@types/request": "^2.48.1",
"@types/ssh2": "^0.5.35",
@ -452,6 +453,7 @@
"dependencies": {
"event-stream": "^3.3.4",
"jsonc-parser": "^2.0.0",
"minimatch": "^3.0.4",
"socks": "^2.2.0",
"ssh2": "^0.8.2",
"winreg": "^1.2.4"

@ -1,57 +1,56 @@
import * as minimatch from 'minimatch';
import { posix as path } from 'path';
import { createInterface } from 'readline';
import { ClientChannel } from 'ssh2';
import * as vscode from 'vscode';
import { Manager } from './manager';
import { toPromise } from './toPromise';
export class SearchProvider implements vscode.FileSearchProvider {
protected cache: [vscode.CancellationToken, Promise<string[]>][] = [];
protected cache: [vscode.CancellationToken, Promise<vscode.Uri[]>][] = [];
constructor(protected manager: Manager) { }
public async provideFileSearchResults(query: vscode.FileSearchQuery, options: vscode.FileSearchOptions, token: vscode.CancellationToken): Promise<vscode.Uri[]> {
const { folder, session = token } = options;
const { folder, session } = options;
let cached = this.cache.find(([t]) => session === t);
if (!cached) {
if (!cached && session) {
cached = [session, this.buildTree(options, session)] as SearchProvider['cache'][0];
this.cache.push(cached);
session.onCancellationRequested(() => {
this.cache.splice(this.cache.indexOf(cached!));
});
} else if (!cached) {
cached = [token, this.buildTree(options, token)] as SearchProvider['cache'][0];
}
const paths = await cached[1];
console.log('Found', paths.length);
if (token.isCancellationRequested) return [];
const pattern = query.pattern.toLowerCase();
return paths.map<vscode.Uri | null>((relative) => {
if (!relative.toLowerCase().includes(pattern)) return null;
return folder.with({ path: path.join(folder.path, relative) });
if (!relative.path.toLowerCase().includes(pattern)) return null;
return folder.with({ path: path.join(folder.path, relative.path) });
}).filter(s => !!s) as vscode.Uri[];
}
protected async buildTree(options: vscode.FileSearchOptions, token: vscode.CancellationToken): Promise<string[]> {
protected async buildTree(options: vscode.FileSearchOptions, token: vscode.CancellationToken): Promise<vscode.Uri[]> {
const { folder } = options;
const fs = await this.manager.getFs(folder);
if (!fs || token.isCancellationRequested) return [];
const cmd = `ls -AQ1R "${fs.absoluteFromRelative(folder.path)}"`;
console.log('Creating tree with command:', cmd);
const exec = await toPromise<ClientChannel>(cb => fs.client.exec(cmd, cb)).catch(() => null);
if (!exec || token.isCancellationRequested) return [];
const res: string[] = [];
const rl = createInterface(exec);
let root = '';
rl.on('line', (line: string) => {
if (!line) return;
if (line.endsWith(':')) {
root = JSON.parse(line.substr(0, line.length - 1));
} else {
let relative = JSON.parse(line);
relative = path.join(root, relative);
relative = fs.relativeFromAbsolute(relative);
res.push(relative);
}
});
token.onCancellationRequested(rl.close, rl);
await toPromise(cb => rl.on('close', cb));
const excludes = options.excludes.map(e => minimatch.makeRe(e, { nocase: true }));
const exclude = (p: string) => excludes.some(reg => reg.test(p));
const res: vscode.Uri[] = [];
async function readDirectory(uri: vscode.Uri) {
if (token.isCancellationRequested) return;
const entries = await fs!.readDirectory(uri).catch(() => [] as never);
if (token.isCancellationRequested) return;
return Promise.all(entries.map(([name, type]) => {
if (token.isCancellationRequested) return;
const joined = path.join(uri.path, name);
if (exclude(joined)) return;
// tslint:disable-next-line:no-bitwise
if (type & vscode.FileType.Directory) {
return readDirectory(uri.with({ path: joined }));
} else {
res.push(uri.with({ path: joined }));
}
}));
}
await readDirectory(folder);
return res;
}
}

Loading…
Cancel
Save