/*
* Copyright (C) 2023-2024 Yomitan Authors
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
import Ajv from 'ajv';
import {readFileSync} from 'fs';
import {join, dirname as pathDirname} from 'path';
import {createGenerator} from 'ts-json-schema-generator';
import {fileURLToPath} from 'url';
import {describe, expect, test} from 'vitest';
import {parseJson} from '../dev/json.js';
import {getAllFiles} from '../dev/util.js';
const dirname = pathDirname(fileURLToPath(import.meta.url));
const rootDir = join(dirname, '..');
/**
* @param {import('test/json').JsconfigType|undefined} jsconfigType
* @returns {string}
*/
function getJsconfigPath(jsconfigType) {
let path;
switch (jsconfigType) {
case 'dev': path = '../dev/jsconfig.json'; break;
case 'test': path = '../test/jsconfig.json'; break;
default: path = '../jsconfig.json'; break;
}
return join(dirname, path);
}
/**
* @returns {Ajv}
*/
function createAjv() {
return new Ajv({
meta: true,
strictTuples: false,
allowUnionTypes: true
});
}
/**
* @param {string} path
* @param {string} type
* @param {import('test/json').JsconfigType|undefined} jsconfigType
* @returns {import('ajv').ValidateFunction}
*/
function createValidatorFunctionFromTypeScript(path, type, jsconfigType) {
/** @type {import('ts-json-schema-generator/dist/src/Config').Config} */
const config = {
path,
tsconfig: getJsconfigPath(jsconfigType),
type,
jsDoc: 'none',
additionalProperties: false,
minify: false,
expose: 'none',
strictTuples: true
};
const schema = createGenerator(config).createSchema(config.type);
const ajv = createAjv();
return ajv.compile(schema);
}
/**
* @param {string} path
* @returns {import('ajv').ValidateFunction}
*/
function createValidatorFunctionFromSchemaJson(path) {
/** @type {import('ajv').Schema} */
const schema = parseJson(readFileSync(path, {encoding: 'utf8'}));
const ajv = createAjv();
return ajv.compile(schema);
}
/**
* @param {string} value
* @returns {string}
*/
function normalizePathDirectorySeparators(value) {
return value.replace(/\\/g, '/');
}
describe.concurrent('JSON validation', () => {
const ignoreDirectories = new Set([
'builds',
'dictionaries',
'node_modules',
'playwright-report',
'playwright',
'test-results',
'dev/lib',
'test/playwright'
]);
const existingJsonFiles = getAllFiles(rootDir, (path, isDirectory) => {
const fileNameNormalized = normalizePathDirectorySeparators(path);
if (isDirectory) {
return !ignoreDirectories.has(fileNameNormalized);
} else {
return /\.json$/i.test(fileNameNormalized);
}
});
/** @type {Set} */
const existingJsonFileSet = new Set();
for (const path of existingJsonFiles) {
existingJsonFileSet.add(normalizePathDirectorySeparators(path));
}
const jsonFileName = 'json.json';
/** @type {import('test/json').JsonInfo} */
const jsonFileData = parseJson(readFileSync(join(dirname, `data/${jsonFileName}`), {encoding: 'utf8'}));
test(`Each item in ${jsonFileName} must have a unique path`, () => {
/** @type {Set} */
const set = new Set();
for (const {path} of jsonFileData.files) {
set.add(path);
}
expect(set.size).toBe(jsonFileData.files.length);
});
/** @type {Map} */
const jsonFileMap = new Map();
for (const item of jsonFileData.files) {
jsonFileMap.set(item.path, item);
}
// Validate file existance
const requiredFiles = jsonFileData.files.filter((v) => !v.ignore);
test.each(requiredFiles)('File must exist in project: $path', ({path}) => {
expect(existingJsonFileSet.has(path)).toBe(true);
});
// Validate new files
const existingJsonFiles2 = existingJsonFiles.map((path) => ({path: normalizePathDirectorySeparators(path)}));
test.each(existingJsonFiles2)(`File must exist in ${jsonFileName}: $path`, ({path}) => {
expect(jsonFileMap.has(path)).toBe(true);
});
// Validate schemas 1
/** @type {import('test/json').JsonFileParseInfo[]} */
const schemaValidationTargets1 = [];
for (const info of jsonFileData.files) {
if (info.ignore || !existingJsonFileSet.has(info.path)) { continue; }
schemaValidationTargets1.push(info);
}
test.each(schemaValidationTargets1)('Validating file against TypeScript: $path', ({path, typeFile, type, jsconfig}) => {
const validate = createValidatorFunctionFromTypeScript(join(rootDir, typeFile), type, jsconfig);
const data = parseJson(readFileSync(join(rootDir, path), {encoding: 'utf8'}));
const valid = validate(data);
const {errors} = validate;
expect(errors).toBe(null);
expect(valid).toBe(true);
});
// Validate schemas 2
/** @type {{path: string, schema: string}[]} */
const schemaValidationTargets2 = [];
for (const info of jsonFileData.files) {
if (info.ignore || !existingJsonFileSet.has(info.path)) { continue; }
const {schema, path} = info;
if (typeof schema !== 'string') { continue; }
schemaValidationTargets2.push({schema, path});
}
test.each(schemaValidationTargets2)('Validating file against schema: $path', ({path, schema}) => {
const validate = createValidatorFunctionFromSchemaJson(join(rootDir, schema));
const data = parseJson(readFileSync(join(rootDir, path), {encoding: 'utf8'}));
const valid = validate(data);
const {errors} = validate;
expect(errors).toBe(null);
expect(valid).toBe(true);
});
});