UNPKG

39.1 kBJavaScriptView Raw
1"use strict";
2/**
3 * @license
4 * Copyright Google LLC All Rights Reserved.
5 *
6 * Use of this source code is governed by an MIT-style license that can be
7 * found in the LICENSE file at https://angular.io/license
8 */
9Object.defineProperty(exports, "__esModule", { value: true });
10exports.SimpleMemoryHost = void 0;
11const rxjs_1 = require("rxjs");
12const exception_1 = require("../../exception");
13const path_1 = require("../path");
14class SimpleMemoryHost {
15 _newDirStats() {
16 return {
17 inspect() {
18 return '<Directory>';
19 },
20 isFile() {
21 return false;
22 },
23 isDirectory() {
24 return true;
25 },
26 size: 0,
27 atime: new Date(),
28 ctime: new Date(),
29 mtime: new Date(),
30 birthtime: new Date(),
31 content: null,
32 };
33 }
34 _newFileStats(content, oldStats) {
35 return {
36 inspect() {
37 return `<File size(${content.byteLength})>`;
38 },
39 isFile() {
40 return true;
41 },
42 isDirectory() {
43 return false;
44 },
45 size: content.byteLength,
46 atime: oldStats ? oldStats.atime : new Date(),
47 ctime: new Date(),
48 mtime: new Date(),
49 birthtime: oldStats ? oldStats.birthtime : new Date(),
50 content,
51 };
52 }
53 constructor() {
54 this._cache = new Map();
55 this._watchers = new Map();
56 this._cache.set((0, path_1.normalize)('/'), this._newDirStats());
57 }
58 _toAbsolute(path) {
59 return (0, path_1.isAbsolute)(path) ? path : (0, path_1.normalize)('/' + path);
60 }
61 _updateWatchers(path, type) {
62 const time = new Date();
63 let currentPath = path;
64 let parent = null;
65 if (this._watchers.size == 0) {
66 // Nothing to do if there's no watchers.
67 return;
68 }
69 const maybeWatcher = this._watchers.get(currentPath);
70 if (maybeWatcher) {
71 maybeWatcher.forEach((watcher) => {
72 const [options, subject] = watcher;
73 subject.next({ path, time, type });
74 if (!options.persistent && type == 2 /* HostWatchEventType.Deleted */) {
75 subject.complete();
76 this._watchers.delete(currentPath);
77 }
78 });
79 }
80 do {
81 currentPath = parent !== null ? parent : currentPath;
82 parent = (0, path_1.dirname)(currentPath);
83 const maybeWatcher = this._watchers.get(currentPath);
84 if (maybeWatcher) {
85 maybeWatcher.forEach((watcher) => {
86 const [options, subject] = watcher;
87 if (!options.recursive) {
88 return;
89 }
90 subject.next({ path, time, type });
91 if (!options.persistent && type == 2 /* HostWatchEventType.Deleted */) {
92 subject.complete();
93 this._watchers.delete(currentPath);
94 }
95 });
96 }
97 } while (parent != currentPath);
98 }
99 get capabilities() {
100 return { synchronous: true };
101 }
102 /**
103 * List of protected methods that give direct access outside the observables to the cache
104 * and internal states.
105 */
106 _write(path, content) {
107 path = this._toAbsolute(path);
108 const old = this._cache.get(path);
109 if (old && old.isDirectory()) {
110 throw new exception_1.PathIsDirectoryException(path);
111 }
112 // Update all directories. If we find a file we know it's an invalid write.
113 const fragments = (0, path_1.split)(path);
114 let curr = (0, path_1.normalize)('/');
115 for (const fr of fragments) {
116 curr = (0, path_1.join)(curr, fr);
117 const maybeStats = this._cache.get(fr);
118 if (maybeStats) {
119 if (maybeStats.isFile()) {
120 throw new exception_1.PathIsFileException(curr);
121 }
122 }
123 else {
124 this._cache.set(curr, this._newDirStats());
125 }
126 }
127 // Create the stats.
128 const stats = this._newFileStats(content, old);
129 this._cache.set(path, stats);
130 this._updateWatchers(path, old ? 0 /* HostWatchEventType.Changed */ : 1 /* HostWatchEventType.Created */);
131 }
132 _read(path) {
133 path = this._toAbsolute(path);
134 const maybeStats = this._cache.get(path);
135 if (!maybeStats) {
136 throw new exception_1.FileDoesNotExistException(path);
137 }
138 else if (maybeStats.isDirectory()) {
139 throw new exception_1.PathIsDirectoryException(path);
140 }
141 else if (!maybeStats.content) {
142 throw new exception_1.PathIsDirectoryException(path);
143 }
144 else {
145 return maybeStats.content;
146 }
147 }
148 _delete(path) {
149 path = this._toAbsolute(path);
150 if (this._isDirectory(path)) {
151 for (const [cachePath] of this._cache.entries()) {
152 if (cachePath.startsWith(path + path_1.NormalizedSep) || cachePath === path) {
153 this._cache.delete(cachePath);
154 }
155 }
156 }
157 else {
158 this._cache.delete(path);
159 }
160 this._updateWatchers(path, 2 /* HostWatchEventType.Deleted */);
161 }
162 _rename(from, to) {
163 from = this._toAbsolute(from);
164 to = this._toAbsolute(to);
165 if (!this._cache.has(from)) {
166 throw new exception_1.FileDoesNotExistException(from);
167 }
168 else if (this._cache.has(to)) {
169 throw new exception_1.FileAlreadyExistException(to);
170 }
171 if (this._isDirectory(from)) {
172 for (const path of this._cache.keys()) {
173 if (path.startsWith(from + path_1.NormalizedSep)) {
174 const content = this._cache.get(path);
175 if (content) {
176 // We don't need to clone or extract the content, since we're moving files.
177 this._cache.set((0, path_1.join)(to, path_1.NormalizedSep, path.slice(from.length)), content);
178 }
179 }
180 }
181 }
182 else {
183 const content = this._cache.get(from);
184 if (content) {
185 const fragments = (0, path_1.split)(to);
186 const newDirectories = [];
187 let curr = (0, path_1.normalize)('/');
188 for (const fr of fragments) {
189 curr = (0, path_1.join)(curr, fr);
190 const maybeStats = this._cache.get(fr);
191 if (maybeStats) {
192 if (maybeStats.isFile()) {
193 throw new exception_1.PathIsFileException(curr);
194 }
195 }
196 else {
197 newDirectories.push(curr);
198 }
199 }
200 for (const newDirectory of newDirectories) {
201 this._cache.set(newDirectory, this._newDirStats());
202 }
203 this._cache.delete(from);
204 this._cache.set(to, content);
205 }
206 }
207 this._updateWatchers(from, 3 /* HostWatchEventType.Renamed */);
208 }
209 _list(path) {
210 path = this._toAbsolute(path);
211 if (this._isFile(path)) {
212 throw new exception_1.PathIsFileException(path);
213 }
214 const fragments = (0, path_1.split)(path);
215 const result = new Set();
216 if (path !== path_1.NormalizedRoot) {
217 for (const p of this._cache.keys()) {
218 if (p.startsWith(path + path_1.NormalizedSep)) {
219 result.add((0, path_1.split)(p)[fragments.length]);
220 }
221 }
222 }
223 else {
224 for (const p of this._cache.keys()) {
225 if (p.startsWith(path_1.NormalizedSep) && p !== path_1.NormalizedRoot) {
226 result.add((0, path_1.split)(p)[1]);
227 }
228 }
229 }
230 return [...result];
231 }
232 _exists(path) {
233 return !!this._cache.get(this._toAbsolute(path));
234 }
235 _isDirectory(path) {
236 const maybeStats = this._cache.get(this._toAbsolute(path));
237 return maybeStats ? maybeStats.isDirectory() : false;
238 }
239 _isFile(path) {
240 const maybeStats = this._cache.get(this._toAbsolute(path));
241 return maybeStats ? maybeStats.isFile() : false;
242 }
243 _stat(path) {
244 const maybeStats = this._cache.get(this._toAbsolute(path));
245 if (!maybeStats) {
246 return null;
247 }
248 else {
249 return maybeStats;
250 }
251 }
252 _watch(path, options) {
253 path = this._toAbsolute(path);
254 const subject = new rxjs_1.Subject();
255 let maybeWatcherArray = this._watchers.get(path);
256 if (!maybeWatcherArray) {
257 maybeWatcherArray = [];
258 this._watchers.set(path, maybeWatcherArray);
259 }
260 maybeWatcherArray.push([options || {}, subject]);
261 return subject.asObservable();
262 }
263 write(path, content) {
264 return new rxjs_1.Observable((obs) => {
265 this._write(path, content);
266 obs.next();
267 obs.complete();
268 });
269 }
270 read(path) {
271 return new rxjs_1.Observable((obs) => {
272 const content = this._read(path);
273 obs.next(content);
274 obs.complete();
275 });
276 }
277 delete(path) {
278 return new rxjs_1.Observable((obs) => {
279 this._delete(path);
280 obs.next();
281 obs.complete();
282 });
283 }
284 rename(from, to) {
285 return new rxjs_1.Observable((obs) => {
286 this._rename(from, to);
287 obs.next();
288 obs.complete();
289 });
290 }
291 list(path) {
292 return new rxjs_1.Observable((obs) => {
293 obs.next(this._list(path));
294 obs.complete();
295 });
296 }
297 exists(path) {
298 return new rxjs_1.Observable((obs) => {
299 obs.next(this._exists(path));
300 obs.complete();
301 });
302 }
303 isDirectory(path) {
304 return new rxjs_1.Observable((obs) => {
305 obs.next(this._isDirectory(path));
306 obs.complete();
307 });
308 }
309 isFile(path) {
310 return new rxjs_1.Observable((obs) => {
311 obs.next(this._isFile(path));
312 obs.complete();
313 });
314 }
315 // Some hosts may not support stat.
316 stat(path) {
317 return new rxjs_1.Observable((obs) => {
318 obs.next(this._stat(path));
319 obs.complete();
320 });
321 }
322 watch(path, options) {
323 return this._watch(path, options);
324 }
325 reset() {
326 this._cache.clear();
327 this._watchers.clear();
328 }
329}
330exports.SimpleMemoryHost = SimpleMemoryHost;
331//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"memory.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/core/src/virtual-fs/host/memory.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport { Observable, Subject } from 'rxjs';\nimport {\n  FileAlreadyExistException,\n  FileDoesNotExistException,\n  PathIsDirectoryException,\n  PathIsFileException,\n} from '../../exception';\nimport {\n  NormalizedRoot,\n  NormalizedSep,\n  Path,\n  PathFragment,\n  dirname,\n  isAbsolute,\n  join,\n  normalize,\n  split,\n} from '../path';\nimport {\n  FileBuffer,\n  Host,\n  HostCapabilities,\n  HostWatchEvent,\n  HostWatchEventType,\n  HostWatchOptions,\n  Stats,\n} from './interface';\n\nexport interface SimpleMemoryHostStats {\n  readonly content: FileBuffer | null;\n}\n\nexport class SimpleMemoryHost implements Host<{}> {\n  protected _cache = new Map<Path, Stats<SimpleMemoryHostStats>>();\n  private _watchers = new Map<Path, [HostWatchOptions, Subject<HostWatchEvent>][]>();\n\n  protected _newDirStats() {\n    return {\n      inspect() {\n        return '<Directory>';\n      },\n\n      isFile() {\n        return false;\n      },\n      isDirectory() {\n        return true;\n      },\n      size: 0,\n\n      atime: new Date(),\n      ctime: new Date(),\n      mtime: new Date(),\n      birthtime: new Date(),\n\n      content: null,\n    };\n  }\n  protected _newFileStats(content: FileBuffer, oldStats?: Stats<SimpleMemoryHostStats>) {\n    return {\n      inspect() {\n        return `<File size(${content.byteLength})>`;\n      },\n\n      isFile() {\n        return true;\n      },\n      isDirectory() {\n        return false;\n      },\n      size: content.byteLength,\n\n      atime: oldStats ? oldStats.atime : new Date(),\n      ctime: new Date(),\n      mtime: new Date(),\n      birthtime: oldStats ? oldStats.birthtime : new Date(),\n\n      content,\n    };\n  }\n\n  constructor() {\n    this._cache.set(normalize('/'), this._newDirStats());\n  }\n\n  protected _toAbsolute(path: Path) {\n    return isAbsolute(path) ? path : normalize('/' + path);\n  }\n\n  protected _updateWatchers(path: Path, type: HostWatchEventType) {\n    const time = new Date();\n    let currentPath = path;\n    let parent: Path | null = null;\n\n    if (this._watchers.size == 0) {\n      // Nothing to do if there's no watchers.\n      return;\n    }\n\n    const maybeWatcher = this._watchers.get(currentPath);\n    if (maybeWatcher) {\n      maybeWatcher.forEach((watcher) => {\n        const [options, subject] = watcher;\n        subject.next({ path, time, type });\n\n        if (!options.persistent && type == HostWatchEventType.Deleted) {\n          subject.complete();\n          this._watchers.delete(currentPath);\n        }\n      });\n    }\n\n    do {\n      currentPath = parent !== null ? parent : currentPath;\n      parent = dirname(currentPath);\n\n      const maybeWatcher = this._watchers.get(currentPath);\n      if (maybeWatcher) {\n        maybeWatcher.forEach((watcher) => {\n          const [options, subject] = watcher;\n          if (!options.recursive) {\n            return;\n          }\n          subject.next({ path, time, type });\n\n          if (!options.persistent && type == HostWatchEventType.Deleted) {\n            subject.complete();\n            this._watchers.delete(currentPath);\n          }\n        });\n      }\n    } while (parent != currentPath);\n  }\n\n  get capabilities(): HostCapabilities {\n    return { synchronous: true };\n  }\n\n  /**\n   * List of protected methods that give direct access outside the observables to the cache\n   * and internal states.\n   */\n  protected _write(path: Path, content: FileBuffer): void {\n    path = this._toAbsolute(path);\n    const old = this._cache.get(path);\n    if (old && old.isDirectory()) {\n      throw new PathIsDirectoryException(path);\n    }\n\n    // Update all directories. If we find a file we know it's an invalid write.\n    const fragments = split(path);\n    let curr: Path = normalize('/');\n    for (const fr of fragments) {\n      curr = join(curr, fr);\n      const maybeStats = this._cache.get(fr);\n      if (maybeStats) {\n        if (maybeStats.isFile()) {\n          throw new PathIsFileException(curr);\n        }\n      } else {\n        this._cache.set(curr, this._newDirStats());\n      }\n    }\n\n    // Create the stats.\n    const stats: Stats<SimpleMemoryHostStats> = this._newFileStats(content, old);\n    this._cache.set(path, stats);\n    this._updateWatchers(path, old ? HostWatchEventType.Changed : HostWatchEventType.Created);\n  }\n  protected _read(path: Path): FileBuffer {\n    path = this._toAbsolute(path);\n    const maybeStats = this._cache.get(path);\n    if (!maybeStats) {\n      throw new FileDoesNotExistException(path);\n    } else if (maybeStats.isDirectory()) {\n      throw new PathIsDirectoryException(path);\n    } else if (!maybeStats.content) {\n      throw new PathIsDirectoryException(path);\n    } else {\n      return maybeStats.content;\n    }\n  }\n  protected _delete(path: Path): void {\n    path = this._toAbsolute(path);\n    if (this._isDirectory(path)) {\n      for (const [cachePath] of this._cache.entries()) {\n        if (cachePath.startsWith(path + NormalizedSep) || cachePath === path) {\n          this._cache.delete(cachePath);\n        }\n      }\n    } else {\n      this._cache.delete(path);\n    }\n    this._updateWatchers(path, HostWatchEventType.Deleted);\n  }\n  protected _rename(from: Path, to: Path): void {\n    from = this._toAbsolute(from);\n    to = this._toAbsolute(to);\n    if (!this._cache.has(from)) {\n      throw new FileDoesNotExistException(from);\n    } else if (this._cache.has(to)) {\n      throw new FileAlreadyExistException(to);\n    }\n\n    if (this._isDirectory(from)) {\n      for (const path of this._cache.keys()) {\n        if (path.startsWith(from + NormalizedSep)) {\n          const content = this._cache.get(path);\n          if (content) {\n            // We don't need to clone or extract the content, since we're moving files.\n            this._cache.set(join(to, NormalizedSep, path.slice(from.length)), content);\n          }\n        }\n      }\n    } else {\n      const content = this._cache.get(from);\n      if (content) {\n        const fragments = split(to);\n        const newDirectories: Path[] = [];\n        let curr: Path = normalize('/');\n        for (const fr of fragments) {\n          curr = join(curr, fr);\n          const maybeStats = this._cache.get(fr);\n          if (maybeStats) {\n            if (maybeStats.isFile()) {\n              throw new PathIsFileException(curr);\n            }\n          } else {\n            newDirectories.push(curr);\n          }\n        }\n        for (const newDirectory of newDirectories) {\n          this._cache.set(newDirectory, this._newDirStats());\n        }\n        this._cache.delete(from);\n        this._cache.set(to, content);\n      }\n    }\n\n    this._updateWatchers(from, HostWatchEventType.Renamed);\n  }\n\n  protected _list(path: Path): PathFragment[] {\n    path = this._toAbsolute(path);\n    if (this._isFile(path)) {\n      throw new PathIsFileException(path);\n    }\n\n    const fragments = split(path);\n    const result = new Set<PathFragment>();\n    if (path !== NormalizedRoot) {\n      for (const p of this._cache.keys()) {\n        if (p.startsWith(path + NormalizedSep)) {\n          result.add(split(p)[fragments.length]);\n        }\n      }\n    } else {\n      for (const p of this._cache.keys()) {\n        if (p.startsWith(NormalizedSep) && p !== NormalizedRoot) {\n          result.add(split(p)[1]);\n        }\n      }\n    }\n\n    return [...result];\n  }\n\n  protected _exists(path: Path): boolean {\n    return !!this._cache.get(this._toAbsolute(path));\n  }\n  protected _isDirectory(path: Path): boolean {\n    const maybeStats = this._cache.get(this._toAbsolute(path));\n\n    return maybeStats ? maybeStats.isDirectory() : false;\n  }\n  protected _isFile(path: Path): boolean {\n    const maybeStats = this._cache.get(this._toAbsolute(path));\n\n    return maybeStats ? maybeStats.isFile() : false;\n  }\n\n  protected _stat(path: Path): Stats<SimpleMemoryHostStats> | null {\n    const maybeStats = this._cache.get(this._toAbsolute(path));\n\n    if (!maybeStats) {\n      return null;\n    } else {\n      return maybeStats;\n    }\n  }\n\n  protected _watch(path: Path, options?: HostWatchOptions): Observable<HostWatchEvent> {\n    path = this._toAbsolute(path);\n\n    const subject = new Subject<HostWatchEvent>();\n    let maybeWatcherArray = this._watchers.get(path);\n    if (!maybeWatcherArray) {\n      maybeWatcherArray = [];\n      this._watchers.set(path, maybeWatcherArray);\n    }\n\n    maybeWatcherArray.push([options || {}, subject]);\n\n    return subject.asObservable();\n  }\n\n  write(path: Path, content: FileBuffer): Observable<void> {\n    return new Observable<void>((obs) => {\n      this._write(path, content);\n      obs.next();\n      obs.complete();\n    });\n  }\n\n  read(path: Path): Observable<FileBuffer> {\n    return new Observable<FileBuffer>((obs) => {\n      const content = this._read(path);\n      obs.next(content);\n      obs.complete();\n    });\n  }\n\n  delete(path: Path): Observable<void> {\n    return new Observable<void>((obs) => {\n      this._delete(path);\n      obs.next();\n      obs.complete();\n    });\n  }\n\n  rename(from: Path, to: Path): Observable<void> {\n    return new Observable<void>((obs) => {\n      this._rename(from, to);\n      obs.next();\n      obs.complete();\n    });\n  }\n\n  list(path: Path): Observable<PathFragment[]> {\n    return new Observable<PathFragment[]>((obs) => {\n      obs.next(this._list(path));\n      obs.complete();\n    });\n  }\n\n  exists(path: Path): Observable<boolean> {\n    return new Observable<boolean>((obs) => {\n      obs.next(this._exists(path));\n      obs.complete();\n    });\n  }\n\n  isDirectory(path: Path): Observable<boolean> {\n    return new Observable<boolean>((obs) => {\n      obs.next(this._isDirectory(path));\n      obs.complete();\n    });\n  }\n\n  isFile(path: Path): Observable<boolean> {\n    return new Observable<boolean>((obs) => {\n      obs.next(this._isFile(path));\n      obs.complete();\n    });\n  }\n\n  // Some hosts may not support stat.\n  stat(path: Path): Observable<Stats<{}> | null> | null {\n    return new Observable<Stats<{}> | null>((obs) => {\n      obs.next(this._stat(path));\n      obs.complete();\n    });\n  }\n\n  watch(path: Path, options?: HostWatchOptions): Observable<HostWatchEvent> | null {\n    return this._watch(path, options);\n  }\n\n  reset(): void {\n    this._cache.clear();\n    this._watchers.clear();\n  }\n}\n"]}
\No newline at end of file