Spaces:
Sleeping
Sleeping
| ; | |
| var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { | |
| if (k2 === undefined) k2 = k; | |
| Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } }); | |
| }) : (function(o, m, k, k2) { | |
| if (k2 === undefined) k2 = k; | |
| o[k2] = m[k]; | |
| })); | |
| var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { | |
| Object.defineProperty(o, "default", { enumerable: true, value: v }); | |
| }) : function(o, v) { | |
| o["default"] = v; | |
| }); | |
| var __importStar = (this && this.__importStar) || function (mod) { | |
| if (mod && mod.__esModule) return mod; | |
| var result = {}; | |
| if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); | |
| __setModuleDefault(result, mod); | |
| return result; | |
| }; | |
| var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { | |
| function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } | |
| return new (P || (P = Promise))(function (resolve, reject) { | |
| function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } | |
| function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } | |
| function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } | |
| step((generator = generator.apply(thisArg, _arguments || [])).next()); | |
| }); | |
| }; | |
| var __importDefault = (this && this.__importDefault) || function (mod) { | |
| return (mod && mod.__esModule) ? mod : { "default": mod }; | |
| }; | |
| Object.defineProperty(exports, "__esModule", { value: true }); | |
| const fs_1 = __importDefault(require("fs")); | |
| const events_1 = require("events"); | |
| const sha1_1 = __importDefault(require("sha1")); | |
| const diff = __importStar(require("diff")); | |
| const asyncCall_1 = __importDefault(require("./asyncCall")); | |
| class FileProxy extends events_1.EventEmitter { | |
| constructor(filePath) { | |
| super(); | |
| this.alive = true; | |
| this.filePath = filePath; | |
| //console.log("File proxy created:", filePath); | |
| if (!fs_1.default.existsSync(filePath)) | |
| throw new Error(`file not exist: ${filePath}`); | |
| asyncCall_1.default(fs_1.default.stat, filePath) | |
| .then(stats => { | |
| this.diskTimestamp = stats.mtime.getTime(); | |
| this.timestamp = this.diskTimestamp; | |
| return asyncCall_1.default(fs_1.default.readFile, filePath); | |
| }) | |
| .then(buffer => { | |
| this.content = buffer.toString(); | |
| this.fullSync(); | |
| }); | |
| this.fileListener = (current) => __awaiter(this, void 0, void 0, function* () { | |
| this.diskTimestamp = current.mtime.getTime(); | |
| this.timestamp = this.diskTimestamp; | |
| const buffer = yield asyncCall_1.default(fs_1.default.readFile, filePath); | |
| if (!buffer) { | |
| this.emit("error", { description: "file reading failed" }); | |
| return; | |
| } | |
| const newContent = buffer.toString(); | |
| const newHash = sha1_1.default(newContent); | |
| if (newHash !== this.hash) { | |
| const patch = diff.createPatch(filePath, this.content, newContent); | |
| this.emit("increase", { | |
| timestamp: this.timestamp, | |
| fromHash: this.hash, | |
| toHash: newHash, | |
| patch, | |
| }); | |
| this.content = newContent; | |
| } | |
| }); | |
| fs_1.default.watchFile(filePath, this.fileListener); | |
| this.keepWriteFile(); | |
| } | |
| dispose() { | |
| this.alive = false; | |
| if (this.fileListener) | |
| fs_1.default.unwatchFile(this.filePath, this.fileListener); | |
| } | |
| makeWritePromise() { | |
| return new Promise(resolve => this.writeFile = resolve); | |
| } | |
| keepWriteFile() { | |
| return __awaiter(this, void 0, void 0, function* () { | |
| let writeSignal = this.makeWritePromise(); | |
| while (this.alive) { | |
| yield writeSignal; | |
| writeSignal = this.makeWritePromise(); | |
| //console.debug("keepWriteFile:", this.timestamp, this.diskTimestamp); | |
| if (this.timestamp > this.diskTimestamp) { | |
| yield asyncCall_1.default(fs_1.default.writeFile, this.filePath, this.content); | |
| } | |
| } | |
| }); | |
| } | |
| get hash() { | |
| return sha1_1.default(this.content); | |
| } | |
| fullSync() { | |
| this.emit("fullSync", { | |
| timestamp: this.timestamp, | |
| content: this.content, | |
| hash: this.hash, | |
| }); | |
| } | |
| increase({ timestamp, fromHash, toHash, patch }) { | |
| if (this.hash !== fromHash) { | |
| if (this.timestamp > timestamp) | |
| // web content is out of date | |
| this.fullSync(); | |
| else | |
| console.warn("[FileProxy] disk file content is behind increase base:", this.timestamp, timestamp); | |
| } | |
| else { | |
| this.content = diff.applyPatch(this.content, patch); | |
| this.timestamp = timestamp; | |
| console.assert(this.hash === toHash, "[FileProxy] verify failed:", this.hash, toHash, this.content); | |
| // trigger file writing | |
| this.writeFile(); | |
| } | |
| } | |
| } | |
| exports.default = FileProxy; | |
| ; | |