Hans c889dd8e94 refactor(@angular-cli): get rid of ast-tools and base-href-webpack (#4411)
Those 2 packages were still using the wrong scope (@angular-cli) and were not used by anyone outside the CLI. Just moving the code in the main package is enough.
2017-02-03 22:57:42 -08:00

173 lines
5.0 KiB
TypeScript

import fs = require('fs');
import denodeify = require('denodeify');
const readFile = (denodeify(fs.readFile) as (...args: any[]) => Promise<any>);
const writeFile = (denodeify(fs.writeFile) as (...args: any[]) => Promise<any>);
export interface Host {
write(path: string, content: string): Promise<void>;
read(path: string): Promise<string>;
}
export const NodeHost: Host = {
write: (path: string, content: string) => writeFile(path, content, 'utf8'),
read: (path: string) => readFile(path, 'utf8')
};
export interface Change {
apply(host: Host): Promise<void>;
// The file this change should be applied to. Some changes might not apply to
// a file (maybe the config).
readonly path: string | null;
// The order this change should be applied. Normally the position inside the file.
// Changes are applied from the bottom of a file to the top.
readonly order: number;
// The description of this change. This will be outputted in a dry or verbose run.
readonly description: string;
}
/**
* An operation that does nothing.
*/
export class NoopChange implements Change {
description = 'No operation.';
order = Infinity;
path: string = null;
apply() { return Promise.resolve(); }
}
/**
* An operation that mixes two or more changes, and merge them (in order).
* Can only apply to a single file. Use a ChangeManager to apply changes to multiple
* files.
*/
export class MultiChange implements Change {
private _path: string;
private _changes: Change[];
constructor(...changes: (Change[] | Change)[]) {
this._changes = [];
[].concat(...changes).forEach(change => this.appendChange(change));
}
appendChange(change: Change) {
// Do not append Noop changes.
if (change instanceof NoopChange) {
return;
}
// Validate that the path is the same for everyone of those.
if (this._path === undefined) {
this._path = change.path;
} else if (change.path !== this._path) {
throw new Error('Cannot apply a change to a different path.');
}
this._changes.push(change);
}
get description() {
return `Changes:\n ${this._changes.map(x => x.description).join('\n ')}`;
}
// Always apply as early as the highest change.
get order() { return Math.max(...this._changes.map(c => c.order)); }
get path() { return this._path; }
apply(host: Host) {
return this._changes
.sort((a: Change, b: Change) => b.order - a.order)
.reduce((promise, change) => {
return promise.then(() => change.apply(host));
}, Promise.resolve());
}
}
/**
* Will add text to the source code.
*/
export class InsertChange implements Change {
order: number;
description: string;
constructor(public path: string, private pos: number, private toAdd: string) {
if (pos < 0) {
throw new Error('Negative positions are invalid');
}
this.description = `Inserted ${toAdd} into position ${pos} of ${path}`;
this.order = pos;
}
/**
* This method does not insert spaces if there is none in the original string.
*/
apply(host: Host): Promise<any> {
return host.read(this.path).then(content => {
let prefix = content.substring(0, this.pos);
let suffix = content.substring(this.pos);
return host.write(this.path, `${prefix}${this.toAdd}${suffix}`);
});
}
}
/**
* Will remove text from the source code.
*/
export class RemoveChange implements Change {
order: number;
description: string;
constructor(public path: string, private pos: number, private toRemove: string) {
if (pos < 0) {
throw new Error('Negative positions are invalid');
}
this.description = `Removed ${toRemove} into position ${pos} of ${path}`;
this.order = pos;
}
apply(host: Host): Promise<any> {
return host.read(this.path).then(content => {
let prefix = content.substring(0, this.pos);
let suffix = content.substring(this.pos + this.toRemove.length);
// TODO: throw error if toRemove doesn't match removed string.
return host.write(this.path, `${prefix}${suffix}`);
});
}
}
/**
* Will replace text from the source code.
*/
export class ReplaceChange implements Change {
order: number;
description: string;
constructor(public path: string, private pos: number, private oldText: string,
private newText: string) {
if (pos < 0) {
throw new Error('Negative positions are invalid');
}
this.description = `Replaced ${oldText} into position ${pos} of ${path} with ${newText}`;
this.order = pos;
}
apply(host: Host): Promise<any> {
return host.read(this.path).then(content => {
const prefix = content.substring(0, this.pos);
const suffix = content.substring(this.pos + this.oldText.length);
const text = content.substring(this.pos, this.pos + this.oldText.length);
if (text !== this.oldText) {
return Promise.reject(new Error(`Invalid replace: "${text}" != "${this.oldText}".`));
}
// TODO: throw error if oldText doesn't match removed string.
return host.write(this.path, `${prefix}${this.newText}${suffix}`);
});
}
}