-
Notifications
You must be signed in to change notification settings - Fork 3
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* refactor: rename `ndjson` to `jsonl` in utils * chore: fix linting issues * refactor: rename default stats file extension to `.jsonl` * test: add `utils/stats` tests * test: fix issue when initializing the fixture files
- Loading branch information
Showing
10 changed files
with
227 additions
and
65 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -4,6 +4,9 @@ | |
/build | ||
/webui/dist | ||
|
||
# test fixtures | ||
*.temp.jsonl | ||
|
||
# dependencies | ||
node_modules/ | ||
npm-debug.log* | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
import { describe, expect, it, mock } from 'bun:test'; | ||
import fs from 'fs'; | ||
import path from 'path'; | ||
|
||
import { appendJsonLine, forEachJsonLines, parseJsonLine } from '../jsonl'; | ||
|
||
describe('forEachJsonLines', () => { | ||
it('iterates each line of file', async () => { | ||
const lines: string[] = []; | ||
await forEachJsonLines(fixture('specification'), (content) => { | ||
lines.push(content); | ||
}); | ||
|
||
expect(lines).toEqual([ | ||
expect.stringContaining('Gilbert'), | ||
expect.stringContaining('Alexa'), | ||
expect.stringContaining('May'), | ||
expect.stringContaining('Deloise'), | ||
]); | ||
}); | ||
|
||
it('iterates each line with line numbers starting from 1', async () => { | ||
const onReadLine = mock(); | ||
await forEachJsonLines(fixture('specification'), onReadLine); | ||
|
||
// Callback is invoked with (content, line, reader) => ... | ||
expect(onReadLine).not.toHaveBeenCalledWith(expect.any(String), 0, expect.any(Object)); | ||
expect(onReadLine).toHaveBeenCalledWith(expect.any(String), 1, expect.any(Object)); | ||
expect(onReadLine).toHaveBeenCalledWith(expect.any(String), 2, expect.any(Object)); | ||
expect(onReadLine).toHaveBeenCalledWith(expect.any(String), 3, expect.any(Object)); | ||
expect(onReadLine).toHaveBeenCalledWith(expect.any(String), 4, expect.any(Object)); | ||
}); | ||
}); | ||
|
||
describe('parseJsonLine', () => { | ||
it('parses a single line from file', async () => { | ||
expect(await parseJsonLine(fixture('specification'), 1)).toMatchObject({ name: 'Gilbert' }); | ||
expect(await parseJsonLine(fixture('specification'), 2)).toMatchObject({ name: 'Alexa' }); | ||
expect(await parseJsonLine(fixture('specification'), 3)).toMatchObject({ name: 'May' }); | ||
expect(await parseJsonLine(fixture('specification'), 4)).toMatchObject({ name: 'Deloise' }); | ||
}); | ||
|
||
it('throws if single line is not found', async () => { | ||
await expect(parseJsonLine(fixture('specification'), 99999)).rejects.toThrow( | ||
'Line 99999 not found in file' | ||
); | ||
}); | ||
}); | ||
|
||
describe('appendJsonLine', () => { | ||
it('appends a single line to file', async () => { | ||
const file = fixture('append-single', { temporary: true }); | ||
await appendJsonLine(file, { name: 'Gilbert' }); | ||
await expect(fs.promises.readFile(file, 'utf-8')).resolves.toBe('{"name":"Gilbert"}\n'); | ||
}); | ||
|
||
it('appends multiple lines to file', async () => { | ||
const file = fixture('append-multiple', { temporary: true }); | ||
const data = [ | ||
{ name: 'Gilbert', list: ['some-list'] }, | ||
{ name: 'Alexa', nested: { nested: true, list: ['other', 'items'] } }, | ||
{ name: 'May', names: 1 }, | ||
{ name: 'Deloise', simple: true }, | ||
]; | ||
|
||
for (const item of data) { | ||
await appendJsonLine(file, item); | ||
} | ||
|
||
await expect(fs.promises.readFile(file, 'utf-8')).resolves.toBe( | ||
data.map((item) => JSON.stringify(item) + '\n').join('') | ||
); | ||
}); | ||
}); | ||
|
||
/** | ||
* Get the file path to a fixture, by name. | ||
* This automatically adds the required `.jsonl` or `.temp.jsonl` extension. | ||
* Use `temporary: true` to keep it out of the repository, and reset the content automatically. | ||
*/ | ||
function fixture(name: string, { temporary = false }: { temporary?: boolean } = {}) { | ||
const file = temporary | ||
? path.join(__dirname, 'fixtures/jsonl', `${name}.temp.jsonl`) | ||
: path.join(__dirname, 'fixtures/jsonl', `${name}.jsonl`); | ||
|
||
fs.mkdirSync(path.dirname(file), { recursive: true }); | ||
|
||
if (temporary) { | ||
fs.writeFileSync(file, ''); | ||
} | ||
|
||
return file; | ||
} |
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,119 @@ | ||
import { describe, expect, it } from 'bun:test'; | ||
import fs from 'fs'; | ||
import path from 'path'; | ||
|
||
import { name, version } from '../../../package.json'; | ||
import { AtlasValidationError } from '../errors'; | ||
import { getStatsPath, getStatsMetdata, createStatsFile, validateStatsFile } from '../stats'; | ||
|
||
describe('getStatsPath', () => { | ||
it('returns default path `<project>/.expo/stats.jsonl`', () => { | ||
expect(getStatsPath('<project>')).toBe('<project>/.expo/stats.jsonl'); | ||
}); | ||
}); | ||
|
||
describe('getStatsMetadata', () => { | ||
it('returns package name and version', () => { | ||
expect(getStatsMetdata()).toMatchObject({ name, version }); | ||
}); | ||
}); | ||
|
||
describe('createStatsFile', () => { | ||
it('creates a stats file with the correct metadata', async () => { | ||
const file = fixture('create-metadata', { temporary: true }); | ||
await createStatsFile(file); | ||
await expect(fs.promises.readFile(file, 'utf8')).resolves.toBe( | ||
JSON.stringify({ name, version }) + '\n' | ||
); | ||
}); | ||
|
||
it('overwrites invalid stats file', async () => { | ||
const file = fixture('create-invalid', { temporary: true }); | ||
await fs.promises.writeFile(file, JSON.stringify({ name, version: '0.0.0' }) + '\n'); | ||
await createStatsFile(file); | ||
await expect(fs.promises.readFile(file, 'utf8')).resolves.toBe( | ||
JSON.stringify({ name, version }) + '\n' | ||
); | ||
}); | ||
|
||
it('reuses valid stats file', async () => { | ||
const file = fixture('create-valid', { temporary: true }); | ||
await fs.promises.writeFile(file, JSON.stringify({ name, version }) + '\n'); | ||
await createStatsFile(file); | ||
await expect(fs.promises.readFile(file, 'utf-8')).resolves.toBe( | ||
JSON.stringify({ name, version }) + '\n' | ||
); | ||
}); | ||
}); | ||
|
||
describe('validateStatsFile', () => { | ||
it('passes for valid stats file', async () => { | ||
const file = fixture('validate-valid', { temporary: true }); | ||
await createStatsFile(file); | ||
await expect(validateStatsFile(file)).resolves.pass(); | ||
}); | ||
|
||
it('fails for non-existing stats file', async () => { | ||
await expect(validateStatsFile('./this-file-does-not-exists')).rejects.toThrow( | ||
AtlasValidationError | ||
); | ||
}); | ||
|
||
it('fails for invalid stats file', async () => { | ||
const file = fixture('validate-invalid', { temporary: true }); | ||
await fs.promises.writeFile(file, JSON.stringify({ name, version: '0.0.0' }) + '\n'); | ||
await expect(validateStatsFile(file)).rejects.toThrow(AtlasValidationError); | ||
}); | ||
|
||
it('skips validation when EXPO_ATLAS_NO_STATS_VALIDATION is true-ish', async () => { | ||
using _env = env('EXPO_ATLAS_NO_STATS_VALIDATION', 'true'); | ||
const file = fixture('validate-skip-invalid', { temporary: true }); | ||
await fs.promises.writeFile(file, JSON.stringify({ name, version: '0.0.0' }) + '\n'); | ||
await expect(validateStatsFile(file)).resolves.pass(); | ||
}); | ||
}); | ||
|
||
/** | ||
* Get the file path to a fixture, by name. | ||
* This automatically adds the required `.jsonl` or `.temp.jsonl` extension. | ||
* Use `temporary: true` to keep it out of the repository, and reset the content automatically. | ||
*/ | ||
function fixture(name: string, { temporary = false }: { temporary?: boolean } = {}) { | ||
const file = temporary | ||
? path.join(__dirname, 'fixtures/stats', `${name}.temp.jsonl`) | ||
: path.join(__dirname, 'fixtures/stats', `${name}.jsonl`); | ||
|
||
fs.mkdirSync(path.dirname(file), { recursive: true }); | ||
|
||
if (temporary) { | ||
fs.writeFileSync(file, ''); | ||
} | ||
|
||
return file; | ||
} | ||
|
||
/** | ||
* Change the environment variable for the duration of a test. | ||
* This uses explicit resource management to revert the environment variable after the test. | ||
*/ | ||
function env(key: string, value?: string): { key: string; value?: string } & Disposable { | ||
const original = process.env[key]; | ||
|
||
if (value === undefined) { | ||
delete process.env[key]; | ||
} else { | ||
process.env[key] = value; | ||
} | ||
|
||
return { | ||
key, | ||
value, | ||
[Symbol.dispose]() { | ||
if (original === undefined) { | ||
delete process.env[key]; | ||
} else { | ||
process.env[key] = value; | ||
} | ||
}, | ||
}; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,10 +1,10 @@ | ||
import { boolish } from 'getenv'; | ||
|
||
export const env = { | ||
get EXPO_DEBUG() { | ||
return boolish('EXPO_DEBUG', false); | ||
get EXPO_ATLAS_DEBUG() { | ||
return boolish('EXPO_ATLAS_DEBUG', false); | ||
}, | ||
get EXPO_NO_STATS_VALIDATION() { | ||
return boolish('EXPO_NO_STATS_VALIDATION', false); | ||
get EXPO_ATLAS_NO_STATS_VALIDATION() { | ||
return boolish('EXPO_ATLAS_NO_STATS_VALIDATION', false); | ||
}, | ||
}; |
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters