Skip to content

Commit 39ca575

Browse files
committedSep 12, 2020
fix(MongoBinary): improve code
- add tsdoc to many functions - getCachePath: remove being async - getDownloadPath: change the times to be more understandable - getDownloadPath: use "this.getCachePath" instead of "this.cache[]" - getPath: remove unnecessary variables - getPath: add an Error if "binaryPath" is still false-y - improve debug logging
1 parent 82606d8 commit 39ca575

File tree

2 files changed

+55
-29
lines changed

2 files changed

+55
-29
lines changed
 

‎packages/mongodb-memory-server-core/src/util/MongoBinary.ts

+53-26
Original file line numberDiff line numberDiff line change
@@ -32,54 +32,65 @@ export interface MongoBinaryOpts {
3232
export default class MongoBinary {
3333
static cache: MongoBinaryCache = {};
3434

35+
/**
36+
* Probe if the provided "systemBinary" is an existing path
37+
* @param systemBinary The Path to probe for an System-Binary
38+
* @return System Binary path or empty string
39+
*/
3540
static async getSystemPath(systemBinary: string): Promise<string> {
3641
let binaryPath = '';
3742

3843
try {
3944
await promisify(fs.access)(systemBinary);
4045

41-
log(`MongoBinary: found sytem binary path at ${systemBinary}`);
46+
log(`MongoBinary: found sytem binary path at "${systemBinary}"`);
4247
binaryPath = systemBinary;
4348
} catch (err) {
44-
log(`MongoBinary: can't find system binary at ${systemBinary}. ${err.message}`);
49+
log(`MongoBinary: can't find system binary at "${systemBinary}".\n${err.message}`);
4550
}
4651

4752
return binaryPath;
4853
}
4954

50-
static async getCachePath(version: string): Promise<string> {
55+
/**
56+
* Check if specified version already exists in the cache
57+
* @param version The Version to check for
58+
*/
59+
static getCachePath(version: string): string {
5160
return this.cache[version];
5261
}
5362

63+
/**
64+
* Probe download path and download the binary
65+
* @param options Options Configuring which binary to download and to which path
66+
* @returns The BinaryPath the binary has been downloaded to
67+
*/
5468
static async getDownloadPath(options: Required<MongoBinaryOpts>): Promise<string> {
5569
const { downloadDir, platform, arch, version, checkMD5 } = options;
56-
// create downloadDir if not exists
70+
// create downloadDir
5771
await mkdirp(downloadDir);
5872

73+
/** Lockfile path */
5974
const lockfile = path.resolve(downloadDir, `${version}.lock`);
60-
// wait lock
75+
// wait to get a lock
6176
await new Promise((resolve, reject) => {
6277
LockFile.lock(
6378
lockfile,
6479
{
65-
wait: 120000,
80+
wait: 1000 * 12, // 120 seconds
6681
pollPeriod: 100,
67-
stale: 110000,
82+
stale: 1000 * 11, // 110 seconds
6883
retries: 3,
6984
retryWait: 100,
7085
},
7186
(err: any) => {
72-
if (err) {
73-
reject(err);
74-
} else {
75-
resolve();
76-
}
87+
return err ? reject(err) : resolve();
7788
}
7889
);
7990
});
8091

81-
// again check cache, maybe other instance resolve it
82-
if (!this.cache[version]) {
92+
// check cache if it got already added to the cache
93+
if (!this.getCachePath(version)) {
8394
const downloader = new MongoBinaryDownload({
8495
downloadDir,
8596
platform,
@@ -90,16 +101,22 @@ export default class MongoBinary {
90101
this.cache[version] = await downloader.getMongodPath();
91102
}
92103
// remove lock
93-
LockFile.unlock(lockfile, (err: any) => {
104+
LockFile.unlock(lockfile, (err) => {
94105
log(
95106
err
96107
? `MongoBinary: Error when removing download lock ${err}`
97108
: `MongoBinary: Download lock removed`
98109
);
99110
});
100-
return this.cache[version];
111+
return this.getCachePath(version);
101112
}
102113

114+
/**
115+
* Probe all supported paths for an binary and return the binary path
116+
* @param opts Options configuring which binary to search for
117+
* @throws {Error} if no valid BinaryPath has been found
118+
* @return The first found BinaryPath
119+
*/
103120
static async getPath(opts: MongoBinaryOpts = {}): Promise<string> {
104121
const legacyDLDir = path.resolve(os.homedir(), '.cache/mongodb-binaries');
105122

@@ -109,6 +126,7 @@ export default class MongoBinary {
109126
nodeModulesDLDir = path.resolve(nodeModulesDLDir, '..', '..');
110127
}
111128

129+
// "||" is still used here, because it should default if the value is false-y (like an empty string)
112130
const defaultOptions = {
113131
downloadDir:
114132
resolveConfig('DOWNLOAD_DIR') ||
@@ -128,17 +146,16 @@ export default class MongoBinary {
128146
checkMD5: envToBool(resolveConfig('MD5_CHECK') ?? ''),
129147
};
130148

149+
/** Provided Options combined with the Default Options */
131150
const options = { ...defaultOptions, ...opts };
132-
log(`MongoBinary options: ${JSON.stringify(options)}`);
133-
134-
const { version, systemBinary } = options;
151+
log(`MongoBinary options:`, JSON.stringify(options, null, 2));
135152

136153
let binaryPath = '';
137154

138-
if (systemBinary) {
139-
binaryPath = await this.getSystemPath(systemBinary);
155+
if (options.systemBinary) {
156+
binaryPath = await this.getSystemPath(options.systemBinary);
140157
if (binaryPath) {
141-
if (~binaryPath.indexOf(' ')) {
158+
if (binaryPath.indexOf(' ') >= 0) {
142159
binaryPath = `"${binaryPath}"`;
143160
}
144161

@@ -147,30 +164,40 @@ export default class MongoBinary {
147164
.split('\n')[0]
148165
.split(' ')[2];
149166

150-
if (version !== LATEST_VERSION && version !== binaryVersion) {
167+
if (options.version !== LATEST_VERSION && options.version !== binaryVersion) {
151168
// we will log the version number of the system binary and the version requested so the user can see the difference
152169
log(
153170
'MongoMemoryServer: Possible version conflict\n' +
154171
` SystemBinary version: ${binaryVersion}\n` +
155-
` Requested version: ${version}\n\n` +
172+
` Requested version: ${options.version}\n\n` +
156173
' Using SystemBinary!'
157174
);
158175
}
159176
}
160177
}
161178

162179
if (!binaryPath) {
163-
binaryPath = await this.getCachePath(version);
180+
binaryPath = this.getCachePath(options.version);
164181
}
165182

166183
if (!binaryPath) {
167184
binaryPath = await this.getDownloadPath(options);
168185
}
169186

170-
log(`MongoBinary: Mongod binary path: ${binaryPath}`);
187+
if (!binaryPath) {
188+
throw new Error(
189+
`MongoBinary.getPath: could not find an valid binary path! (Got: "${binaryPath}")`
190+
);
191+
}
192+
193+
log(`MongoBinary: Mongod binary path: "${binaryPath}"`);
171194
return binaryPath;
172195
}
173196

197+
/**
198+
* Purpose unkown, not used anywhere
199+
* @param files Array of Strings
200+
*/
174201
static hasValidBinPath(files: string[]): boolean {
175202
if (files.length === 1) {
176203
return true;

‎packages/mongodb-memory-server-core/src/util/__tests__/MongoBinary-test.ts

+2-3
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,6 @@ describe('MongoBinary', () => {
4545

4646
describe('getDownloadPath', () => {
4747
it('should download binary and keep it in cache', async () => {
48-
// download
4948
const version = LATEST_VERSION;
5049
const binPath = await MongoBinary.getPath({
5150
downloadDir: tmpDir.name,
@@ -71,14 +70,14 @@ describe('MongoBinary', () => {
7170
describe('getCachePath', () => {
7271
it('should get the cache', async () => {
7372
MongoBinary.cache['3.4.2'] = '/bin/mongod';
74-
await expect(MongoBinary.getCachePath('3.4.2')).resolves.toEqual('/bin/mongod');
73+
expect(MongoBinary.getCachePath('3.4.2')).toEqual('/bin/mongod');
7574
});
7675
});
7776

7877
describe('getSystemPath', () => {
7978
it('should use system binary if option is passed.', async () => {
8079
const accessSpy = jest.spyOn(fs, 'access');
81-
await MongoBinary.getSystemPath('/usr/bin/mongod');
80+
await MongoBinary.getSystemPath('/usr/bin/mongod'); // ignoring return, because this depends on the host system
8281

8382
expect(accessSpy).toHaveBeenCalledWith('/usr/bin/mongod', expect.any(Function));
8483

0 commit comments

Comments
 (0)
Please sign in to comment.