Special regexp tokens were allowed unchanged previously, which incorrectly broke the include/exclude behaviour. Now, they're escaped first. Closes #1721 Closes #1752
		
			
				
	
	
		
			140 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			140 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /// <reference path="../typings/node/node.d.ts" />
 | |
| 
 | |
| import fs = require('fs');
 | |
| import path = require('path');
 | |
| 
 | |
| 
 | |
| export class TreeDiffer {
 | |
|   private fingerprints: {[key: string]: string} = Object.create(null);
 | |
|   private nextFingerprints: {[key: string]: string} = Object.create(null);
 | |
|   private rootDirName: string;
 | |
|   private include: RegExp = null;
 | |
|   private exclude: RegExp = null;
 | |
| 
 | |
|   constructor(private rootPath: string, includeExtensions?: string[],
 | |
|               excludeExtensions?: string[]) {
 | |
|     this.rootDirName = path.basename(rootPath);
 | |
| 
 | |
|     let buildRegexp = (arr) => new RegExp(`(${arr.reduce(combine, "")})$`, "i");
 | |
| 
 | |
|     this.include = (includeExtensions || []).length ? buildRegexp(includeExtensions) : null;
 | |
|     this.exclude = (excludeExtensions || []).length ? buildRegexp(excludeExtensions) : null;
 | |
| 
 | |
|     function combine(prev, curr) {
 | |
|       if (curr.charAt(0) !== ".") throw new TypeError("Extension must begin with '.'");
 | |
|       let kSpecialRegexpChars = /[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g;
 | |
|       curr = '(' + curr.replace(kSpecialRegexpChars, '\\$&') + ')';
 | |
|       return prev ? (prev + '|' + curr) : curr;
 | |
|     }
 | |
|   }
 | |
| 
 | |
| 
 | |
|   public diffTree(): DiffResult {
 | |
|     let result = new DirtyCheckingDiffResult(this.rootDirName);
 | |
|     this.dirtyCheckPath(this.rootPath, result);
 | |
|     this.detectDeletionsAndUpdateFingerprints(result);
 | |
|     result.endTime = Date.now();
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   private dirtyCheckPath(rootDir: string, result: DirtyCheckingDiffResult) {
 | |
|     fs.readdirSync(rootDir).forEach((segment) => {
 | |
|       let absolutePath = path.join(rootDir, segment);
 | |
|       let pathStat = fs.statSync(absolutePath);
 | |
| 
 | |
|       if (pathStat.isDirectory()) {
 | |
|         result.directoriesChecked++;
 | |
|         this.dirtyCheckPath(absolutePath, result);
 | |
|       } else {
 | |
|         if (!(this.include && !absolutePath.match(this.include)) &&
 | |
|             !(this.exclude && absolutePath.match(this.exclude))) {
 | |
|           result.filesChecked++;
 | |
|           if (this.isFileDirty(absolutePath, pathStat)) {
 | |
|             result.changedPaths.push(path.relative(this.rootPath, absolutePath));
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     });
 | |
| 
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   private isFileDirty(path: string, stat: fs.Stats): boolean {
 | |
|     let oldFingerprint = this.fingerprints[path];
 | |
|     let newFingerprint = `${stat.mtime.getTime()} # ${stat.size}`;
 | |
| 
 | |
|     this.nextFingerprints[path] = newFingerprint;
 | |
| 
 | |
|     if (oldFingerprint) {
 | |
|       this.fingerprints[path] = null;
 | |
| 
 | |
|       if (oldFingerprint === newFingerprint) {
 | |
|         // nothing changed
 | |
|         return false;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
| 
 | |
|   private detectDeletionsAndUpdateFingerprints(result: DiffResult) {
 | |
|     for (let absolutePath in this.fingerprints) {
 | |
|       if (!(this.include && !absolutePath.match(this.include)) &&
 | |
|           !(this.exclude && absolutePath.match(this.exclude))) {
 | |
|         if (this.fingerprints[absolutePath] !== null) {
 | |
|           let relativePath = path.relative(this.rootPath, absolutePath);
 | |
|           result.removedPaths.push(relativePath);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     this.fingerprints = this.nextFingerprints;
 | |
|     this.nextFingerprints = Object.create(null);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| export interface DiffResult {
 | |
|   changedPaths: string[];
 | |
|   removedPaths: string[];
 | |
|   log(verbose: boolean): void;
 | |
|   toString(): string;
 | |
| }
 | |
| 
 | |
| 
 | |
| class DirtyCheckingDiffResult {
 | |
|   public filesChecked: number = 0;
 | |
|   public directoriesChecked: number = 0;
 | |
|   public changedPaths: string[] = [];
 | |
|   public removedPaths: string[] = [];
 | |
|   public startTime: number = Date.now();
 | |
|   public endTime: number = null;
 | |
| 
 | |
|   constructor(public name: string) {}
 | |
| 
 | |
|   toString() {
 | |
|     return `${pad(this.name, 40)}, duration: ${pad(this.endTime - this.startTime, 5)}ms, ` +
 | |
|            `${pad(this.changedPaths.length + this.removedPaths.length, 5)} changes detected ` +
 | |
|            `(files: ${pad(this.filesChecked, 5)}, directories: ${pad(this.directoriesChecked, 4)})`;
 | |
|   }
 | |
| 
 | |
|   log(verbose) {
 | |
|     let prefixedPaths =
 | |
|         this.changedPaths.map((p) => `* ${p}`).concat(this.removedPaths.map((p) => `- ${p}`));
 | |
|     console.log(`Tree diff: ${this}` + ((verbose && prefixedPaths.length) ?
 | |
|                                              ` [\n  ${prefixedPaths.join('\n  ')}\n]` :
 | |
|                                              ''));
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| function pad(value, length) {
 | |
|   value = '' + value;
 | |
|   let whitespaceLength = (value.length < length) ? length - value.length : 0;
 | |
|   whitespaceLength = whitespaceLength + 1;
 | |
|   return new Array(whitespaceLength).join(' ') + value;
 | |
| }
 |