This repository has been archived by the owner on Jul 21, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 9
/
fs.ts
279 lines (225 loc) · 8.24 KB
/
fs.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
import * as fs from "fs"
import { promisify } from "util"
import { EncodingOptions, ServerProxy } from "../common/proxy"
import { ReadableProxy, WritableProxy } from "./stream"
/* eslint-disable @typescript-eslint/no-explicit-any */
/**
* A serializable version of fs.Stats.
*/
export interface Stats {
dev: number
ino: number
mode: number
nlink: number
uid: number
gid: number
rdev: number
size: number
blksize: number
blocks: number
atimeMs: number
mtimeMs: number
ctimeMs: number
birthtimeMs: number
atime: Date
mtime: Date
ctime: Date
birthtime: Date
_isFile: boolean
_isDirectory: boolean
_isBlockDevice: boolean
_isCharacterDevice: boolean
_isSymbolicLink: boolean
_isFIFO: boolean
_isSocket: boolean
}
export class ReadStreamProxy extends ReadableProxy<fs.ReadStream> {
public constructor(stream: fs.ReadStream) {
super(stream, ["open"])
}
public async close(): Promise<void> {
this.instance.close()
}
public async dispose(): Promise<void> {
this.instance.close()
await super.dispose()
}
}
export class WriteStreamProxy extends WritableProxy<fs.WriteStream> {
public constructor(stream: fs.WriteStream) {
super(stream, ["open"])
}
public async close(): Promise<void> {
this.instance.close()
}
public async dispose(): Promise<void> {
this.instance.close()
await super.dispose()
}
}
export class WatcherProxy extends ServerProxy<fs.FSWatcher> {
public constructor(watcher: fs.FSWatcher) {
super({
bindEvents: ["change", "close", "error"],
doneEvents: ["close", "error"],
instance: watcher,
})
}
public async close(): Promise<void> {
this.instance.close()
}
public async dispose(): Promise<void> {
this.instance.close()
await super.dispose()
}
}
export class FsModuleProxy {
public access(path: fs.PathLike, mode?: number): Promise<void> {
return promisify(fs.access)(path, mode)
}
public appendFile(file: fs.PathLike | number, data: any, options?: fs.WriteFileOptions): Promise<void> {
return promisify(fs.appendFile)(file, data, options)
}
public chmod(path: fs.PathLike, mode: string | number): Promise<void> {
return promisify(fs.chmod)(path, mode)
}
public chown(path: fs.PathLike, uid: number, gid: number): Promise<void> {
return promisify(fs.chown)(path, uid, gid)
}
public close(fd: number): Promise<void> {
return promisify(fs.close)(fd)
}
public copyFile(src: fs.PathLike, dest: fs.PathLike, flags?: number): Promise<void> {
return promisify(fs.copyFile)(src, dest, flags)
}
public async createReadStream(path: fs.PathLike, options?: any): Promise<ReadStreamProxy> {
return new ReadStreamProxy(fs.createReadStream(path, options))
}
public async createWriteStream(path: fs.PathLike, options?: any): Promise<WriteStreamProxy> {
return new WriteStreamProxy(fs.createWriteStream(path, options))
}
public exists(path: fs.PathLike): Promise<boolean> {
return promisify(fs.exists)(path)
}
public fchmod(fd: number, mode: string | number): Promise<void> {
return promisify(fs.fchmod)(fd, mode)
}
public fchown(fd: number, uid: number, gid: number): Promise<void> {
return promisify(fs.fchown)(fd, uid, gid)
}
public fdatasync(fd: number): Promise<void> {
return promisify(fs.fdatasync)(fd)
}
public async fstat(fd: number): Promise<Stats> {
return this.makeStatsSerializable(await promisify(fs.fstat)(fd))
}
public fsync(fd: number): Promise<void> {
return promisify(fs.fsync)(fd)
}
public ftruncate(fd: number, len?: number | null): Promise<void> {
return promisify(fs.ftruncate)(fd, len)
}
public futimes(fd: number, atime: string | number | Date, mtime: string | number | Date): Promise<void> {
return promisify(fs.futimes)(fd, atime, mtime)
}
public lchmod(path: fs.PathLike, mode: string | number): Promise<void> {
return promisify(fs.lchmod)(path, mode)
}
public lchown(path: fs.PathLike, uid: number, gid: number): Promise<void> {
return promisify(fs.lchown)(path, uid, gid)
}
public link(existingPath: fs.PathLike, newPath: fs.PathLike): Promise<void> {
return promisify(fs.link)(existingPath, newPath)
}
public async lstat(path: fs.PathLike): Promise<Stats> {
return this.makeStatsSerializable(await promisify(fs.lstat)(path))
}
public async lstatBatch(args: { path: fs.PathLike }[]): Promise<(Stats | Error)[]> {
return Promise.all(args.map((a) => this.lstat(a.path).catch((e) => e)))
}
public mkdir(path: fs.PathLike, mode: number | string | fs.MakeDirectoryOptions | undefined | null): Promise<void> {
return promisify(fs.mkdir)(path, mode)
}
public mkdtemp(prefix: string, options: EncodingOptions): Promise<string | Buffer> {
return promisify(fs.mkdtemp)(prefix, options)
}
public open(path: fs.PathLike, flags: string | number, mode: string | number | undefined | null): Promise<number> {
return promisify(fs.open)(path, flags, mode)
}
public read(fd: number, length: number, position: number | null): Promise<{ bytesRead: number; buffer: Buffer }> {
const buffer = Buffer.alloc(length)
return promisify(fs.read)(fd, buffer, 0, length, position)
}
public readFile(path: fs.PathLike | number, options: EncodingOptions): Promise<string | Buffer> {
return promisify(fs.readFile)(path, options)
}
public readdir(path: fs.PathLike, options: EncodingOptions): Promise<Buffer[] | fs.Dirent[] | string[]> {
return promisify(fs.readdir)(path, options)
}
public readdirBatch(
args: { path: fs.PathLike; options: EncodingOptions }[]
): Promise<(Buffer[] | fs.Dirent[] | string[] | Error)[]> {
return Promise.all(args.map((a) => this.readdir(a.path, a.options).catch((e) => e)))
}
public readlink(path: fs.PathLike, options: EncodingOptions): Promise<string | Buffer> {
return promisify(fs.readlink)(path, options)
}
public realpath(path: fs.PathLike, options: EncodingOptions): Promise<string | Buffer> {
return promisify(fs.realpath)(path, options)
}
public rename(oldPath: fs.PathLike, newPath: fs.PathLike): Promise<void> {
return promisify(fs.rename)(oldPath, newPath)
}
public rmdir(path: fs.PathLike): Promise<void> {
return promisify(fs.rmdir)(path)
}
public async stat(path: fs.PathLike): Promise<Stats> {
return this.makeStatsSerializable(await promisify(fs.stat)(path))
}
public async statBatch(args: { path: fs.PathLike }[]): Promise<(Stats | Error)[]> {
return Promise.all(args.map((a) => this.stat(a.path).catch((e) => e)))
}
public symlink(target: fs.PathLike, path: fs.PathLike, type?: fs.symlink.Type | null): Promise<void> {
return promisify(fs.symlink)(target, path, type)
}
public truncate(path: fs.PathLike, len?: number | null): Promise<void> {
return promisify(fs.truncate)(path, len)
}
public unlink(path: fs.PathLike): Promise<void> {
return promisify(fs.unlink)(path)
}
public utimes(path: fs.PathLike, atime: string | number | Date, mtime: string | number | Date): Promise<void> {
return promisify(fs.utimes)(path, atime, mtime)
}
public async write(
fd: number,
buffer: Buffer,
offset?: number,
length?: number,
position?: number
): Promise<{ bytesWritten: number; buffer: Buffer }> {
return promisify(fs.write)(fd, buffer, offset, length, position)
}
public writeFile(path: fs.PathLike | number, data: any, options: EncodingOptions): Promise<void> {
return promisify(fs.writeFile)(path, data, options)
}
public async watch(filename: fs.PathLike, options?: EncodingOptions): Promise<WatcherProxy> {
return new WatcherProxy(fs.watch(filename, options))
}
private makeStatsSerializable(stats: fs.Stats): Stats {
return {
...stats,
/**
* We need to check if functions exist because nexe's implemented FS
* lib doesnt implement fs.stats properly.
*/
_isBlockDevice: stats.isBlockDevice ? stats.isBlockDevice() : false,
_isCharacterDevice: stats.isCharacterDevice ? stats.isCharacterDevice() : false,
_isDirectory: stats.isDirectory(),
_isFIFO: stats.isFIFO ? stats.isFIFO() : false,
_isFile: stats.isFile(),
_isSocket: stats.isSocket ? stats.isSocket() : false,
_isSymbolicLink: stats.isSymbolicLink ? stats.isSymbolicLink() : false,
}
}
}