diff --git a/.gitignore b/.gitignore index 7c520063..21699a78 100644 --- a/.gitignore +++ b/.gitignore @@ -1,10 +1,13 @@ out/ -junit +tmp/ +*.tgz +*.zip # ide files .project .settings .idea/ +/.vscode *.iml # node specific files @@ -28,8 +31,8 @@ npm-debug.log .tool-versions # instanbul code coverage… -coverage +.nyc_output/ +coverage/ /examples/tmp-docs -.nyc_output +junit/ test/fixtures/cyclic-out/ -/.vscode diff --git a/.npmignore b/.npmignore index b7655d32..9d2986fb 100644 --- a/.npmignore +++ b/.npmignore @@ -1,7 +1,13 @@ +out/ +tmp/ +*.tgz +*.zip + # ide files .project .settings .idea/ +.vscode/ *.iml # node specific files @@ -21,3 +27,20 @@ npm-debug.log # mac files .DS_Store + +.tool-versions + +# instanbul code coverage… +.nyc_output/ +coverage/ +/examples/ +junit/ +test/ +tsconfig.json + +# CI/CD +.circleci/ +.github/ +.renovaterc.json +.tidelift.yml +crowdin.yml diff --git a/examples/generated-schemas/subdir.schema.json b/examples/generated-schemas/subdir.schema.json new file mode 100644 index 00000000..aaf60194 --- /dev/null +++ b/examples/generated-schemas/subdir.schema.json @@ -0,0 +1 @@ +{"meta:license":["Copyright 2017 Adobe Systems Incorporated. All rights reserved.","This file is licensed to you under the Apache License, Version 2.0 (the 'License');","you may not use this file except in compliance with the License. You may obtain a copy","of the License at http://www.apache.org/licenses/LICENSE-2.0"],"$schema":"http://json-schema.org/draft-06/schema#","$id":"https://example.com/schemas/subdir/subdir","title":"Subdir","type":"object","description":"A schema in a sub directory","definitions":{"content":{"properties":{"id":{"type":"string","format":"uri","description":"A unique identifier given to every addressable thing.","version":"1.0.0","testProperty":"test"}}}},"allOf":[{"$ref":"#/definitions/content"}]} diff --git a/lib/formatInfo.js b/lib/formatInfo.js index d8dee485..3544c8ae 100644 --- a/lib/formatInfo.js +++ b/lib/formatInfo.js @@ -49,24 +49,24 @@ function iscustom(schema) { function getdefined(schema) { if (schema[s.parent]) { return { - text: `${path.basename(schema[s.filename])}*`, - link: schema[s.filename], + text: `${schema[s.filename]}*`, + link: schema[s.fullpath], }; } return { - text: path.basename(schema[s.filename]), - link: schema[s.filename], + text: schema[s.filename], + link: schema[s.fullpath], }; } function plaindescription(schema) { try { - if (schema[s.filename] && !schema[s.parent]) { - const filename = path.resolve( - path.dirname(schema[s.filename]), + if (schema[s.fullpath] && !schema[s.parent]) { + const fullPath = path.resolve( + path.dirname(schema[s.fullpath]), schema[s.filename].replace(/\..*$/, '.description.md'), ); - const longdesc = fs.readFileSync(filename); + const longdesc = fs.readFileSync(fullPath); return longdesc.toString(); } } catch { diff --git a/lib/index.js b/lib/index.js index 82dfc2a8..c2b06c66 100644 --- a/lib/index.js +++ b/lib/index.js @@ -9,17 +9,15 @@ * OF ANY KIND, either express or implied. See the License for the specific language * governing permissions and limitations under the License. */ - +/* @ts-check */ const yargs = require('yargs'); -const nodepath = require('path'); - const fs = require('fs'); const readdirp = require('readdirp'); const logger = require('@adobe/helix-log'); const { iter, pipe, filter, map, obj, } = require('ferrum'); -const npath = require('path'); +const nodepath = require('path'); const { i18nConfig } = require('es2015-i18n-tag'); const traverse = require('./traverseSchema'); const build = require('./markdownBuilder'); @@ -30,46 +28,134 @@ const { writeSchema } = require('./writeSchema'); const { info, error, debug } = logger; -function jsonschema2md(schema, { - schemaPath, - out, - meta, - schemaOut, - includeReadme, - links, - i18n, - language, - exampleFormat, - includeProperties, - header, - skipProperties, -}) { - const outOrDefault = out || nodepath.resolve(nodepath.join('.', 'out')); +/** + * @typedef {import("../types/api").JsonSchema} JsonSchema + */ +/** + * @typedef {import("../types/api").SchemaList} SchemaList + */ +/** + * @typedef {import("../types/api").SchemaContent} SchemaContent + */ +/** + * @typedef {import("../types/api").SchemaFiles} SchemaFiles + */ +/** + * @typedef {import("../types/api").GeneratedOutput} GeneratedOutput + */ + +/** + * Public API for jsonschema2md that can be used to turn JSON Schema files + * into readable Markdown documentation. + * @param {JsonSchema | SchemaFiles} schema JSON Schema input to get data from + * @param {Object} options Additional options for generation + * @param {string} [options.schemaPath] - (optional) Path to directory containing all JSON Schemas + * or a single JSON Schema file. This will be considered as the baseURL. + * @param {string} [options.outDir] - (optional) Path to output directory. Generating files will + * be skipped if directory is not specified. + * @param {{ [key:string]: string }} [options.metadata] - (optional) Add metadata elements to + * .md files. + * @param {string} [options.schemaOut] - (optional) Output JSON Schema files including + * description and validated examples in the specified folder. + * @param {boolean} [options.includeReadme=true] - (optional) Generate a README.md file in the + * output directory. + * @param {{ [key:string]: string }[]} [options.links] - (optional) Add this file as a link + * explaining the specified attribute. + * @param {string} [options.i18n="locales/"] - (optional) Path to a locales folder with + * JSON files used for internationalization. + * @param {"en_US" | "de"} [options.language="en_US"] - (optional) Selected language. + * @param {"json" | "yaml"} [options.exampleFormat="json"] - (optional) How to format examples. + * @param {string[]} [options.includeProperties=[]] - (optional) Name of custom properties + * which should be also in the description of an element. + * @param {boolean} [options.header=true] - (optional) Whether or not to include the header + * in markdown. + * @param {string[]} [options.skipProperties=[]] - (optional) Name of a default property to + * skip in markdown. + * @returns {GeneratedOutput} List of raw markdown that were generated from input schema. + */ +function jsonschema2md(schema, options) { + const { + schemaPath, + outDir, + metadata, + schemaOut, + includeReadme, + links, + i18n, + language, + exampleFormat, + includeProperties, + header, + skipProperties, + } = options; + if (!schema || typeof schema !== 'object') { + throw Error('Input is not valid. Provide JSON schema either as Object or Array.'); + } const locales = i18n || nodepath.resolve(__dirname, 'locales'); // eslint-disable-next-line import/no-dynamic-require, global-require i18nConfig(require(nodepath.resolve(locales, `${language || 'en_US'}.json`))); + let out = outDir; + if (options.out) { + // eslint-disable-next-line no-console + console.warn("Options 'out' has been deprecated. Please, use 'outDir' instead"); + out = options.out; + } + let meta = metadata; + if (options.meta) { + // eslint-disable-next-line no-console + console.warn("Options 'meta' has been deprecated. Please, use 'metadata' instead"); + meta = options.meta; + } - const schemas = [].concat(schema); - if (schemaOut) { - console.log('writing schemas'); - writeSchema({ - schemadir: schemaOut, - origindir: schemaPath, - })(schemas); + /** @type {SchemaFiles} */ + let normalized; + if (Array.isArray(schema)) { + normalized = schema; + } else { + normalized = [{ + fileName: 'definition.schema.json', + content: schema, + }]; } + const schemaLoader = loader(); + + // collect data about the schemas and turn everything into a big object + const schemas = pipe( + normalized, + // Checking if data contains the file path or its contents (JSON schema) + map(({ fileName, fullPath, content }) => { + if (!content && fullPath) { + // eslint-disable-next-line import/no-dynamic-require, global-require + return schemaLoader(fullPath, require(fullPath)); + } + return schemaLoader(fileName, content); + }), + traverse, + ); + + /** + * @type {GeneratedOutput} + */ + const output = {}; + + console.log('preparing schemas...'); + output.schema = writeSchema({ + schemadir: schemaOut, + origindir: schemaPath, + })(schemas); + if (includeReadme) { - console.log('writing README'); - pipe( + console.log('preparing README...'); + output.readme = pipe( schemas, // build readme readme({ - readme: includeReadme, + readme: true, }), writereadme({ - readme: includeReadme, - out: outOrDefault, + out, info, error, debug, @@ -78,8 +164,8 @@ function jsonschema2md(schema, { ); } - console.log('writing documentation'); - const markdown = pipe( + console.log('preparing documentation...'); + output.markdown = pipe( schemas, // generate Markdown ASTs build({ @@ -89,21 +175,24 @@ function jsonschema2md(schema, { exampleFormat, skipProperties, rewritelinks: (origin) => { - const mddir = outOrDefault; + const mddir = out; + if (!mddir) { + return origin; + } const srcdir = schemaPath; const schemadir = schemaOut || schemaPath; - const target = npath.relative( + const target = nodepath.relative( mddir, - npath.resolve(schemadir, npath.relative(srcdir, origin)), - ).split(npath.sep).join(npath.posix.sep); + nodepath.resolve(schemadir, nodepath.relative(srcdir, origin)), + ).split(nodepath.sep).join(nodepath.posix.sep); return target; }, }), // write to files writemarkdown({ - out: outOrDefault, + out, info, error, debug, @@ -111,9 +200,14 @@ function jsonschema2md(schema, { }), ); - return markdown; + return output; } +/** + * Main function used in the CLI. + * @param {{ [key:string]: unknown }} args CLI arguments from user input + * @returns The generated Markdown files to the specified output directory + */ async function main(args) { // parse/process command line arguments const { argv } = yargs(args) @@ -198,8 +292,8 @@ async function main(args) { ); const schemaPath = argv.d; - const out = argv.o; - const meta = argv.m; + const outDir = argv.o; + const metadata = argv.m; const schemaOut = argv.x !== '-' ? argv.x : null; const includeReadme = !argv.n; const i18n = argv.i; @@ -211,27 +305,23 @@ async function main(args) { const schemaExtension = argv.e; - const schemaloader = loader(); - // list all schema files in the specified directory - const schemafiles = await readdirp.promise(schemaPath, { root: schemaPath, fileFilter: `*.${schemaExtension}` }); + const schemaFiles = await readdirp.promise(schemaPath, { root: schemaPath, fileFilter: `*.${schemaExtension}` }); - console.log(`loading ${schemafiles.length} schemas`); + console.log(`loading ${schemaFiles.length} schemas`); - // then collect data about the schemas and turn everything into a big object - const loadedschemas = pipe( - schemafiles, - map((schema) => schema.fullPath), - // eslint-disable-next-line import/no-dynamic-require, global-require - map((path) => schemaloader(require(path), path)), - // find contained schemas - traverse, - ); + /** + * @type {SchemaList[]} + * */ + const schemas = schemaFiles.map((schema) => ({ + fileName: schema.basename, + fullPath: schema.fullPath, + })); - jsonschema2md(loadedschemas, { + jsonschema2md(schemas, { schemaPath, - out, - meta, + outDir, + metadata, schemaOut, includeReadme, links, diff --git a/lib/markdownBuilder.js b/lib/markdownBuilder.js index 428cb09e..87675e31 100644 --- a/lib/markdownBuilder.js +++ b/lib/markdownBuilder.js @@ -18,7 +18,7 @@ const { blockquote, } = require('mdast-builder'); const i18n = require('es2015-i18n-tag').default; -const ghslugger = require('github-slugger'); +const GhSlugger = require('github-slugger'); const yaml = require('js-yaml'); const s = require('./symbols'); const { gentitle } = require('./formattingTools'); @@ -853,7 +853,7 @@ function build({ console.log('generating markdown'); return (schemas) => foldl(schemas, {}, (pv, schema) => { - const slugger = ghslugger(); + const slugger = new GhSlugger(); // eslint-disable-next-line no-param-reassign pv[schema[s.slug]] = root([ // todo add more elements diff --git a/lib/readmeBuilder.js b/lib/readmeBuilder.js index 6fed624f..2a8214d9 100644 --- a/lib/readmeBuilder.js +++ b/lib/readmeBuilder.js @@ -58,9 +58,9 @@ function build({ readme = true }) { const bytype = (type) => flist(pipe( schemas, - filter((schema) => schema[keyword`type`] === type), // remove schemas without matching type - filter((schema) => !!schema[s.parent]), // keep only schemas with a parent - filter((schema) => !schema.$ref), // it is not a reference + filter((schema) => schema[keyword`type`] === type), // remove schemas without matching type + filter((schema) => !!schema[s.parent]), // keep only schemas with a parent + filter((schema) => !schema.$ref), // it is not a reference mapSort((schema) => gentitle(schema[s.titles], schema[keyword`type`])), map((schema) => listItem(paragraph([ link(`./${schema[s.slug]}.md`, gendescription(schema), [text(gentitle(schema[s.titles], schema[keyword`type`]))]), diff --git a/lib/schemaProxy.js b/lib/schemaProxy.js index 2010be24..4ffef902 100644 --- a/lib/schemaProxy.js +++ b/lib/schemaProxy.js @@ -9,7 +9,8 @@ * OF ANY KIND, either express or implied. See the License for the specific language * governing permissions and limitations under the License. */ -const ghslugger = require('github-slugger'); +// @ts-check +const GhSlugger = require('github-slugger'); const path = require('path'); const { URL } = require('url'); const { formatmeta } = require('./formatInfo'); @@ -18,6 +19,10 @@ const { keyword } = require('./keywords'); const myslug = Symbol('myslug'); +/** + * @param {string} file + * @param {number} [num=1] + */ function loadExamples(file, num = 1) { const examplefile = path.resolve(path.dirname(file), path.basename(file).replace(/\..*$/, `.example.${num}.json`)); try { @@ -30,13 +35,14 @@ function loadExamples(file, num = 1) { } const handler = ({ - root = '', filename = '.', schemas, parent, slugger, + root = '', fullpath = null, filename = '.', schemas, parent = null, slugger, }) => { const meta = {}; meta[symbols.parent] = () => parent; meta[symbols.pointer] = () => root; meta[symbols.filename] = () => filename; + meta[symbols.fullpath] = () => fullpath; meta[symbols.id] = (target) => { // if the schema has it's own ID, use it if (target[keyword`$id`]) { @@ -62,7 +68,7 @@ const handler = ({ }; meta[symbols.resolve] = (target, prop, receiver) => (proppath) => { - // console.log('trying to resolve', proppath, 'in', receiver[symbols.filename]); + // console.log('trying to resolve', proppath, 'in', receiver[symbols.fullpath]); if (proppath === undefined) { return receiver; } @@ -78,7 +84,7 @@ const handler = ({ meta[symbols.slug] = (target, prop, receiver) => { if (!receiver[myslug] && !parent && receiver[symbols.filename]) { // eslint-disable-next-line no-param-reassign - receiver[myslug] = slugger.slug(path.basename(receiver[symbols.filename]).replace(/\..*$/, '')); + receiver[myslug] = slugger.slug(receiver[symbols.filename].replace(/\..*$/, '')); } if (!receiver[myslug]) { const parentslug = parent[symbols.slug]; @@ -106,13 +112,13 @@ const handler = ({ } const retval = Reflect.get(target, prop, receiver); - if (retval === undefined && prop === keyword`examples` && !receiver[symbols.parent]) { - return loadExamples(receiver[symbols.filename], 1); + if (retval === undefined && receiver[symbols.fullpath] && prop === keyword`examples` && !receiver[symbols.parent]) { + return loadExamples(receiver[symbols.fullpath], 1); } if (typeof retval === 'object' && retval !== null) { if (retval[keyword`$ref`]) { const [uri, pointer] = retval.$ref.split('#'); - // console.log('resolving ref', uri, pointer, 'from', receiver[symbols.filename]); + // console.log('resolving ref', uri, pointer, 'from', receiver[symbols.fullpath]); const basedoc = uri || receiver[symbols.id]; let referenced = null; @@ -121,7 +127,7 @@ const handler = ({ if (schemas.known[basedoc]) { referenced = schemas.known[basedoc][symbols.resolve](pointer); } else if (path.parse(basedoc)) { - const basepath = path.dirname(meta[symbols.filename]()); + const basepath = path.dirname(meta[symbols.fullpath]()); let reldoc = uri; // if uri is a URI then only try to resolve it locally if the scheme is 'file:' @@ -160,6 +166,7 @@ const handler = ({ const subschema = new Proxy(retval, handler({ root: `${root}/${prop}`, parent: receiver, + fullpath, filename, schemas, slugger, @@ -188,18 +195,22 @@ module.exports.loader = () => { files: {}, }; - const slugger = ghslugger(); + const slugger = new GhSlugger(); - return (schema, filename) => { - // console.log('loading', filename); - const proxied = new Proxy(schema, handler({ filename, schemas, slugger })); + return (/** @type {string} */ name, /** @type {any} */ schema) => { + // console.log('loading', name); + const filename = path.basename(name); + const fullpath = name === filename ? undefined : name; + const proxied = new Proxy(schema, handler({ + filename, fullpath, schemas, slugger, + })); schemas.loaded.push(proxied); if (proxied[keyword`$id`]) { // stow away the schema for lookup schemas.known[proxied[keyword`$id`]] = proxied; } - schemas.files[filename] = proxied; + schemas.files[fullpath || filename] = proxied; return proxied; }; diff --git a/lib/symbols.js b/lib/symbols.js index 522f5518..0a3cd6d7 100644 --- a/lib/symbols.js +++ b/lib/symbols.js @@ -9,9 +9,18 @@ * OF ANY KIND, either express or implied. See the License for the specific language * governing permissions and limitations under the License. */ + +/** + * @typedef {import("../types/api").UniqueSymbols} UniqueSymbols + */ + +/** + * @type {UniqueSymbols} + * */ const symbols = { pointer: Symbol('pointer'), filename: Symbol('filename'), + fullpath: Symbol('fullpath'), id: Symbol('id'), titles: Symbol('titles'), resolve: Symbol('resolve'), diff --git a/lib/writeMarkdown.js b/lib/writeMarkdown.js index 212e4403..fa00854f 100644 --- a/lib/writeMarkdown.js +++ b/lib/writeMarkdown.js @@ -9,6 +9,7 @@ * OF ANY KIND, either express or implied. See the License for the specific language * governing permissions and limitations under the License. */ +// @ts-check const { each, pairs } = require('ferrum'); const stringify = require('remark-stringify'); const unified = require('unified'); @@ -17,6 +18,29 @@ const path = require('path'); const fs = require('fs-extra'); const yaml = require('js-yaml'); +/** + * @typedef {import("../types/api").MarkdownAst} MarkdownAst + */ +/** + * @typedef {import("../types/api").MarkdownContent} MarkdownContent + */ +/** + * @typedef {import("../types/api").ReadmeContent} ReadmeContent + */ +/** + * @typedef {{ [name: string]: MarkdownAst }} MarkdownAstFiles + */ + +/** + * Write the Markdown to filesystem or an object + * @param {Object} options + * @param {string} [options.out] - (optional) Directory where the files will be saved + * @param {Function} [options.error] - (optional) Error logger + * @param {Function} [options.info] - (optional) Info logger + * @param {Function} [options.debug] - (optional) Debug logger + * @param {{ [key: string]: string }} [options.meta] - (optional) Metadata to be added to Markdown + * @returns {(astFiles: MarkdownAstFiles) => MarkdownContent[]} + */ function writemarkdown({ out, error, meta, }) { @@ -24,62 +48,87 @@ function writemarkdown({ .use(gfm) .use(stringify); - fs.mkdirpSync(out); - return (schemas) => { - each(pairs(schemas), ([name, markdown]) => { - const fileName = path.resolve(out, `${name}.md`); + /** @type {MarkdownContent[]} */ + const output = []; + each(pairs(schemas), (tuple) => { + /** @type {[ string, MarkdownAst ]} */ + const [name, markdownAst] = tuple; // add YAML frontmatter - const output = (!meta ? '' : '---\n') + const content = (!meta ? '' : '---\n') + (!meta ? '' : yaml.dump(meta)) + (!meta ? '' : '---\n\n') - + processor.stringify(markdown); + + processor.stringify(markdownAst); - fs.writeFile(fileName, output, (err) => { - if (err) { + const fileName = `${name}.md`; + let fullPath; + if (out) { + fullPath = path.resolve(out, fileName); + try { + fs.outputFileSync(fullPath, content); + } catch (err) { error(err); } // info(`${fileName} created`); + } + output.push({ + fileName, + fullPath, + markdownAst, + content, }); }); - return schemas; + return output; }; } +/** + * Write the Readme Markdown to filesystem or an object + * @param {Object} options + * @param {string} [options.out] - (optional) Directory where the files will be saved + * @param {Function} [options.error] - (optional) Error logger + * @param {Function} [options.info] - (optional) Info logger + * @param {Function} [options.debug] - (optional) Debug logger + * @param {{ [key: string]: string }} [options.meta] - (optional) Metadata to be added to Markdown + * @returns {(markdownAst: MarkdownAst) => ReadmeContent} + */ function writereadme({ - out, error, info, meta, readme, + out, error, info, meta, }) { const processor = unified() .use(gfm) .use(stringify); - if (readme) { - fs.mkdirpSync(out); + return (markdownAst) => { + // add YAML frontmatter + const content = (!meta ? '' : '---\n') + + (!meta ? '' : yaml.dump(meta)) + + (!meta ? '' : '---\n\n') - return (readmeast) => { - const fileName = path.resolve(out, 'README.md'); - // add YAML frontmatter - const output = (!meta ? '' : '---\n') - + (!meta ? '' : yaml.dump(meta)) - + (!meta ? '' : '---\n\n') - - + processor.stringify(readmeast); + + processor.stringify(markdownAst); - fs.writeFile(fileName, output, (err) => { - if (err) { - error(err); - } - info(`${fileName} created`); - }); + const fileName = 'README.md'; + let fullPath; + if (out) { + fullPath = path.resolve(out, fileName); + try { + fs.outputFileSync(fullPath, content); + } catch (err) { + error(err); + } + info(`${fileName} created`); + } - return fileName; + return { + fileName, + fullPath, + markdownAst, + content, }; - } else { - return (args) => args; - } + }; } module.exports = { writemarkdown, writereadme }; diff --git a/lib/writeSchema.js b/lib/writeSchema.js index 75a0e8be..6b79b345 100644 --- a/lib/writeSchema.js +++ b/lib/writeSchema.js @@ -9,33 +9,61 @@ * OF ANY KIND, either express or implied. See the License for the specific language * governing permissions and limitations under the License. */ +// @ts-check const fs = require('fs-extra'); const path = require('path'); const s = require('./symbols'); -function writeSchema({ schemadir, origindir }) { - const targetpath = (filename) => path.resolve(schemadir, path.relative(origindir, filename)); +/** + * @typedef {import("../types/api").ExtendedJsonSchema} ExtendedJsonSchema + */ +/** + * @typedef {import("../types/api").SchemaContent} SchemaContent + */ +/** + * Write the JSON Schemas to filesystem or an object + * @param {Object} options + * @param {string} [options.schemadir] - (optional) Directory where the files will be saved + * @param {string} [options.origindir] - (optional) Directory where the files were loaded from + * @returns {(schemas: ExtendedJsonSchema[]) => SchemaContent[]} + */ +function writeSchema({ schemadir, origindir }) { return (schemas) => { - console.log('writing schemas to', schemadir); + // eslint-disable-next-line no-console + console.log('preparing schemas'); const realschemas = Object.values(schemas).filter((schema) => !schema[s.parent]); - realschemas.forEach((schema) => { - // console.log('writing', schema[s.filename], 'to ', targetpath(schema[s.filename])); - const filename = targetpath(schema[s.filename]); - const dirname = path.dirname(filename); + return realschemas.map((schema) => { + const fileName = schema[s.filename]; + const inputPath = schema[s.fullpath] || (origindir && path.join(origindir, fileName)); + let content = schema; - const out = fs.readJSONSync(schema[s.filename]); - if (schema[s.meta] && schema[s.meta].description) { - // copy description from external file - out.description = schema[s.meta].description; + if (inputPath) { + content = fs.readJsonSync(inputPath); + if (schema[s.meta] && schema[s.meta].description) { + // copy description from external file + content.description = schema[s.meta].description; + } + if (schema.examples && Array.isArray(schema.examples) && schema.examples.length > 0) { + // copy examples from external files + content.examples = [...schema.examples]; + } } - if (schema.examples && Array.isArray(schema.examples) && schema.examples.length > 0) { - // copy examples from external files - out.examples = [...schema.examples]; + + let fullPath; + if (schemadir) { + // eslint-disable-next-line no-console + console.log('writing schemas to', schemadir); + fullPath = path.resolve(schemadir, fileName); + fs.outputJsonSync(fullPath, content); } - fs.mkdirpSync(dirname); - fs.writeJsonSync(filename, out); + + return { + fileName, + fullPath, + content, + }; }); }; } diff --git a/package-lock.json b/package-lock.json index b7aa327e..fdec08d3 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1044,6 +1044,32 @@ "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", "dev": true }, + "@types/fs-extra": { + "version": "9.0.12", + "resolved": "https://registry.npmjs.org/@types/fs-extra/-/fs-extra-9.0.12.tgz", + "integrity": "sha512-I+bsBr67CurCGnSenZZ7v94gd3tc3+Aj2taxMT4yu4ABLuOgOjeFxX3dokG24ztSRg5tnT00sL8BszO7gSMoIw==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/github-slugger": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@types/github-slugger/-/github-slugger-1.3.0.tgz", + "integrity": "sha512-J/rMZa7RqiH/rT29TEVZO4nBoDP9XJOjnbbIofg7GQKs4JIduEO3WLpte+6WeUz/TcrXKlY+bM7FYrp8yFB+3g==", + "dev": true + }, + "@types/js-yaml": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@types/js-yaml/-/js-yaml-4.0.2.tgz", + "integrity": "sha512-KbeHS/Y4R+k+5sWXEYzAZKuB1yQlZtEghuhRxrVRLaqhtoG5+26JwQsa4HyS3AWX8v1Uwukma5HheduUDskasA==", + "dev": true + }, + "@types/json-schema": { + "version": "7.0.8", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.8.tgz", + "integrity": "sha512-YSBPTLTVm2e2OoQIDYx8HaeWJ5tTToLH67kXR7zYNGupXMEHa2++G8k+DczX2cFVgalypqtyZIcU19AFcmOpmg==" + }, "@types/json5": { "version": "0.0.29", "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", @@ -1051,9 +1077,9 @@ "dev": true }, "@types/mdast": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/@types/mdast/-/mdast-3.0.3.tgz", - "integrity": "sha512-SXPBMnFVQg1s00dlMCc/jCdvPqdE4mXaMMCeRlxLDmTAEoegHT53xKtkDnzDTOcmMHUfcjyf36/YYZ6SxRdnsw==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@types/mdast/-/mdast-3.0.4.tgz", + "integrity": "sha512-gIdhbLDFlspL53xzol2hVzrXAbzt71erJHoOwQZWssjaiouOotf03lNtMmFm9VfFkvnLWccSVjUAZGQ5Kqw+jA==", "requires": { "@types/unist": "*" } @@ -1063,6 +1089,12 @@ "resolved": "https://registry.npmjs.org/@types/minimist/-/minimist-1.2.1.tgz", "integrity": "sha512-fZQQafSREFyuZcdWFAExYjBiCL7AUCdgsk80iO0q4yihYYdcIiH28CcuPTGFgLOCC8RlW49GSQxdHwZP+I7CNg==" }, + "@types/node": { + "version": "16.0.1", + "resolved": "https://registry.npmjs.org/@types/node/-/node-16.0.1.tgz", + "integrity": "sha512-hBOx4SUlEPKwRi6PrXuTGw1z6lz0fjsibcWCM378YxsSu/6+C30L6CR49zIBKHiwNWCYIcOLjg4OHKZaFeLAug==", + "dev": true + }, "@types/normalize-package-data": { "version": "2.4.0", "resolved": "https://registry.npmjs.org/@types/normalize-package-data/-/normalize-package-data-2.4.0.tgz", @@ -1085,6 +1117,21 @@ "resolved": "https://registry.npmjs.org/@types/unist/-/unist-2.0.3.tgz", "integrity": "sha512-FvUupuM3rlRsRtCN+fDudtmytGO6iHJuuRKS1Ss0pG5z8oX0diNEw94UEL7hgDbpN94rgaK5R7sWm6RrSkZuAQ==" }, + "@types/yargs": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.2.tgz", + "integrity": "sha512-JhZ+pNdKMfB0rXauaDlrIvm+U7V4m03PPOSVoPS66z8gf+G4Z/UW8UlrVIj2MRQOBzuoEvYtjS0bqYwnpZaS9Q==", + "dev": true, + "requires": { + "@types/yargs-parser": "*" + } + }, + "@types/yargs-parser": { + "version": "20.2.1", + "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-20.2.1.tgz", + "integrity": "sha512-7tFImggNeNBVMsn0vLrpn1H1uPrUBdnARPTpZoitY37ZrdJREzf7I16tMrlK3hen349gr1NYh8CmZQa7CTG6Aw==", + "dev": true + }, "@ungap/promise-all-settled": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/@ungap/promise-all-settled/-/promise-all-settled-1.1.2.tgz", @@ -6092,6 +6139,11 @@ "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==" }, + "json-schema": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.5.tgz", + "integrity": "sha512-gWJOWYFrhQ8j7pVm0EM8Slr+EPVq1Phf6lvzvD/WCeqkrx/f2xBI0xOsRRS9xCn3I4vKtP519dvs3TP09r24wQ==" + }, "json-schema-traverse": { "version": "0.4.1", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", @@ -12808,6 +12860,12 @@ "is-typedarray": "^1.0.0" } }, + "typescript": { + "version": "4.3.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.3.5.tgz", + "integrity": "sha512-DqQgihaQ9cUrskJo9kIyW/+g0Vxsk8cDtZ52a3NGh0YNTfpUSArXSohyUGnvbPazEPLu398C0UxmKSOrPumUzA==", + "dev": true + }, "uglify-js": { "version": "3.13.6", "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.13.6.tgz", diff --git a/package.json b/package.json index 7a2b9108..3b213d7d 100644 --- a/package.json +++ b/package.json @@ -3,26 +3,31 @@ "description": "Validate and document complex JSON Schemas the easy way.", "version": "1.1.1", "main": "lib/index.js", + "typings": "types", "bin": { "jsonschema2md": "./cli.js" }, "scripts": { - "semantic-release": "semantic-release", + "semantic-release": "npm run emit-types && npx semantic-release", "commit": "git-cz", "lint": "eslint .", "lint-fix": "eslint . --fix", "start": "node cli.js", "examples": "node cli.js -d examples/schemas/ -o examples/docs/ -x examples/generated-schemas/", "test": " nyc --reporter=text --reporter=lcov --check-coverage --branches 80 --statements 95 --lines 95 mocha && node lib/locales/update.js", - "test-ci": "nyc --reporter=text --reporter=lcov --check-coverage --branches 80 --statements 95 --lines 95 mocha --reporter xunit --reporter-options output=./junit/test-results.xml; codecov" + "test-ci": "nyc --reporter=text --reporter=lcov --check-coverage --branches 80 --statements 95 --lines 95 mocha --reporter xunit --reporter-options output=./junit/test-results.xml; codecov", + "emit-types": "npx -p typescript tsc" }, "dependencies": { "@adobe/helix-log": "5.0.5", + "@types/json-schema": "^7.0.8", + "@types/mdast": "^3.0.4", "es2015-i18n-tag": "1.6.1", "ferrum": "1.9.2", "fs-extra": "10.0.0", "github-slugger": "1.4.0", "js-yaml": "4.1.0", + "json-schema": "^0.2.5", "mdast-builder": "1.1.1", "mdast-util-to-string": "2.0.0", "readdirp": "3.6.0", @@ -37,6 +42,11 @@ "@semantic-release/changelog": "5.0.1", "@semantic-release/git": "10.0.0", "@semantic-release/github": "8.0.0", + "@types/fs-extra": "^9.0.12", + "@types/github-slugger": "^1.3.0", + "@types/js-yaml": "^4.0.2", + "@types/node": "^16.0.1", + "@types/yargs": "^17.0.2", "codecov": "3.8.3", "commitizen": "4.2.4", "cz-conventional-changelog": "3.3.0", @@ -53,6 +63,7 @@ "mocha": "9.1.1", "nyc": "15.1.0", "semantic-release": "18.0.0", + "typescript": "^4.3.5", "unist-util-select": "3.0.4" }, "engines": { diff --git a/test/api.test.js b/test/api.test.js index fe5f2e8d..cae7b4ba 100644 --- a/test/api.test.js +++ b/test/api.test.js @@ -14,12 +14,12 @@ const assert = require('assert'); const fs = require('fs-extra'); const path = require('path'); -const { loadschemas } = require('./testUtils'); +const { assertMarkdown, loadSchemas } = require('./testUtils'); -const example = require('./fixtures/example/example.schema.json'); +const example = require('./fixtures/example/api.schema.json'); const { jsonschema2md } = require('../lib/index'); -describe('Testing Main API', () => { +describe('Testing Public API', () => { beforeEach(async () => { await fs.remove(path.resolve(__dirname, '..', 'tmp')); }); @@ -28,28 +28,99 @@ describe('Testing Main API', () => { await fs.remove(path.resolve(__dirname, '..', 'tmp')); }); - it('Main API processes readme-1 directory', async () => { - const schemas = await loadschemas('readme-1'); - const res = jsonschema2md(schemas, { + it('Public API processes multiple schemas with full path', async () => { + const schemasFiles = await loadSchemas('readme-1'); + const result = jsonschema2md(schemasFiles, { + links: { abstract: 'fooabstract.html' }, + header: true, + includeReadme: true, + }); + // console.log('done!', res); + assert(result === Object(result)); + assert.ok(result.readme.content); + assertMarkdown(result.readme.markdownAst) + .contains('# README') + .contains('The schemas linked above') + .fuzzy` +## Top-level Schemas + +* [Abstract](./abstract.md "This is an abstract schema") – ${null} +* [Complex References](./complex.md "This is an example schema that uses types defined in other schemas") – ${null} +* [Simple](./simple.md "This is a very simple example of a JSON schema") – ${null} +`; + assert.strictEqual(result.markdown.length, 29); + }); + + it('Public API processes multiple schemas with content', async () => { + const schemasFiles = await loadSchemas('readme-1'); + const schemas = schemasFiles.map(({ fileName, fullPath }) => ({ + fileName, + // eslint-disable-next-line global-require, import/no-dynamic-require + content: require(fullPath), + })); + + const result = jsonschema2md(schemas, { schemaPath: path.resolve(__dirname, 'fixtures/readme-1'), - out: 'tmp', + outDir: 'tmp', schemaOut: 'tmp', includeReadme: true, }); - // console.log('done!', res); - assert(res === Object(res)); const readme = await fs.stat(path.resolve(__dirname, '..', 'tmp', 'README.md')); assert.ok(readme.isFile()); + assert.notStrictEqual(readme.size, 0); + assert.ok(result.readme.content); + assertMarkdown(result.readme.markdownAst) + .contains('# README') + .contains('The schemas linked above'); + assert.strictEqual(result.schema.length, 3); + assert.strictEqual(result.markdown.length, 28); }); - it('Main API processes example schema', async () => { - const res = jsonschema2md(example, { + it('Public API processes from single schema', async () => { + const result = jsonschema2md(example, { + includeReadme: true, out: 'tmp', + meta: { + key: 'value', + }, + }); + // console.log('done!', result); + assert(result === Object(result)); + assert.ok(result.readme.content); + assertMarkdown(result.readme.markdownAst) + .contains('# README') + .matches(/Top-level Schemas/) + .contains('https://example.com/schemas/example-api'); + assert.strictEqual(result.markdown.length, 7); + assertMarkdown(result.markdown[0].markdownAst) + .contains('# Example API Properties') + .contains('## foo') + .contains('## bar') + .contains('## baz') + .contains('properties within properties') + .contains('## examples'); + }); + + it('Public API with invalid schema', async () => { + try { + jsonschema2md('test', { + outDir: 'tmp', + includeReadme: true, + }); + } catch (e) { + assert.strictEqual(e.message, 'Input is not valid. Provide JSON schema either as Object or Array.'); + } + }); + + it('Public API with unsupported output directory', async () => { + const outDir = path.resolve(__dirname, '..', 'tmp'); + await fs.ensureDir(outDir); + await fs.chmod(outDir, 0o400); + jsonschema2md(example, { + outDir, includeReadme: true, }); - // console.log('done!', res); - assert(res === Object(res)); - const readme = await fs.stat(path.resolve(__dirname, '..', 'tmp', 'README.md')); - assert.ok(readme.isFile()); + const files = await fs.readdir(outDir); + assert.strictEqual(files.length, 0); }); }); diff --git a/test/cyclicSchemaIntegration.test.js b/test/cyclicSchemaIntegration.test.js index 8ad282e0..129e57ef 100644 --- a/test/cyclicSchemaIntegration.test.js +++ b/test/cyclicSchemaIntegration.test.js @@ -41,9 +41,9 @@ describe('Integration Test: Cyclic References', () => { const myloader = loader(); console.log('Loading Schemas'); - proxiedone = myloader(one, path.resolve(__dirname, 'fixtures', 'cyclic', 'one.schema.json')); - proxiedtwo = myloader(two, path.resolve(__dirname, 'fixtures', 'cyclic', 'two.schema.json')); - proxiedthree = myloader(three, path.resolve(__dirname, 'fixtures', 'cyclic', 'three.schema.json')); + proxiedone = myloader(path.resolve(__dirname, 'fixtures', 'cyclic', 'one.schema.json'), one); + proxiedtwo = myloader(path.resolve(__dirname, 'fixtures', 'cyclic', 'two.schema.json'), two); + proxiedthree = myloader(path.resolve(__dirname, 'fixtures', 'cyclic', 'three.schema.json'), three); console.log('Traversing Schemas'); diff --git a/test/fixtures/example/api.schema.json b/test/fixtures/example/api.schema.json new file mode 100644 index 00000000..3abf0b3c --- /dev/null +++ b/test/fixtures/example/api.schema.json @@ -0,0 +1,68 @@ +{ + "meta:license":[ + "Copyright 2017 Adobe Systems Incorporated. All rights reserved.", + "This file is licensed to you under the Apache License, Version 2.0 (the 'License');", + "you may not use this file except in compliance with the License. You may obtain a copy", + "of the License at http://www.apache.org/licenses/LICENSE-2.0" + ], + "$schema":"http://json-schema.org/draft-06/schema#", + "$id":"https://example.com/schemas/example-api", + "title":"Example API", + "type":"object", + "description":"This is an example schema with examples. Too many examples? There can never be too many examples!", + "properties":{ + "foo":{ + "type":"string", + "description":"A simple string.", + "examples":[ + "bar" + ], + "version":"1.0.0", + "testProperty":"test" + }, + "bar":{ + "type":"string", + "description":"A simple string.", + "examples":[ + "bar", + "baz" + ], + "version":"1.0.0", + "testProperty":"test" + }, + "baz":{ + "anyOf":[ + { + "$ref":"#/properties/foo" + }, + { + "$ref":"#/properties/bar" + } + ] + }, + "properties":{ + "type":"object", + "description":"properties within properties" + }, + "required":{ + "type":"array", + "examples":[ + [ + { + "properies":[ + + ] + } + ] + ] + }, + "examples":{ + "type":"number", + "examples":[ + 1, + 2, + 3 + ] + } + } +} diff --git a/test/fixtures/example/example.schema.json b/test/fixtures/example/proxy.schema.json similarity index 94% rename from test/fixtures/example/example.schema.json rename to test/fixtures/example/proxy.schema.json index 0693e584..deed510b 100644 --- a/test/fixtures/example/example.schema.json +++ b/test/fixtures/example/proxy.schema.json @@ -6,8 +6,8 @@ "of the License at http://www.apache.org/licenses/LICENSE-2.0" ], "$schema":"http://json-schema.org/draft-06/schema#", - "$id":"https://example.com/schemas/example", - "title":"Example", + "$id":"https://example.com/schemas/example-proxy", + "title":"Example Proxy", "type":"object", "description":"This is an example schema with examples. Too many examples? There can never be too many examples!", "properties":{ diff --git a/test/fixtures/example/traverse.schema.json b/test/fixtures/example/traverse.schema.json new file mode 100644 index 00000000..6f5cb16d --- /dev/null +++ b/test/fixtures/example/traverse.schema.json @@ -0,0 +1,76 @@ +{ + "meta:license":[ + "Copyright 2017 Adobe Systems Incorporated. All rights reserved.", + "This file is licensed to you under the Apache License, Version 2.0 (the 'License');", + "you may not use this file except in compliance with the License. You may obtain a copy", + "of the License at http://www.apache.org/licenses/LICENSE-2.0" + ], + "$schema":"http://json-schema.org/draft-06/schema#", + "$id":"https://example.com/schemas/example-traverse", + "title":"Example Traverse", + "type":"object", + "description":"This is an example schema with examples. Too many examples? There can never be too many examples!", + "properties":{ + "foo":{ + "type":"string", + "description":"A simple string.", + "examples":[ + "bar" + ], + "version":"1.0.0", + "testProperty":"test" + }, + "bar":{ + "type":"string", + "description":"A simple string.", + "examples":[ + "bar", + "baz" + ], + "version":"1.0.0", + "testProperty":"test" + }, + "zip":{ + "type":"object", + "title":"An object" + }, + "zup":{ + "type":"object", + "title":"An object" + }, + "baz":{ + "anyOf":[ + { + "$ref":"#/properties/foo" + }, + { + "$ref":"#/properties/bar" + } + ] + }, + "properties":{ + "type":"object", + "description":"properties within properties" + }, + "required":{ + "type":"array", + "examples":[ + [ + { + "properies":[ + + ] + } + ] + ] + }, + "examples":{ + "type":"number", + "examples":[ + 1, + 2, + 3 + ] + } + } +} diff --git a/test/markdownBuilder.test.js b/test/markdownBuilder.test.js index 1c106d7f..05018d09 100644 --- a/test/markdownBuilder.test.js +++ b/test/markdownBuilder.test.js @@ -11,7 +11,7 @@ */ /* eslint-env mocha */ /* eslint-disable no-unused-expressions */ -const { assertMarkdown, loadschemas } = require('./testUtils'); +const { assertMarkdown, traverseSchemas } = require('./testUtils'); const build = require('../lib/markdownBuilder'); @@ -19,7 +19,7 @@ describe('Testing Markdown Builder: content', () => { let results; before(async () => { - const schemas = await loadschemas('content'); + const schemas = await traverseSchemas('content'); const builder = build({ header: false }); results = builder(schemas); }); @@ -45,7 +45,7 @@ describe('Testing Markdown Builder: not', () => { let results; before(async () => { - const schemas = await loadschemas('not'); + const schemas = await traverseSchemas('not'); const builder = build({ header: false }); results = builder(schemas); }); @@ -66,7 +66,7 @@ describe('Testing Markdown Builder: nullable', () => { let results; before(async () => { - const schemas = await loadschemas('nullable'); + const schemas = await traverseSchemas('nullable'); const builder = build({ header: false }); results = builder(schemas); }); @@ -81,7 +81,7 @@ describe('Testing Markdown Builder: title', () => { let results; before(async () => { - const schemas = await loadschemas('title'); + const schemas = await traverseSchemas('title'); const builder = build({ header: false }); results = builder(schemas); }); @@ -101,7 +101,7 @@ describe('Testing Markdown Builder: type', () => { let results; before(async () => { - const schemas = await loadschemas('type'); + const schemas = await traverseSchemas('type'); const builder = build({ header: false }); results = builder(schemas); }); @@ -123,7 +123,7 @@ describe('Testing Markdown Builder: format', () => { let results; before(async () => { - const schemas = await loadschemas('format'); + const schemas = await traverseSchemas('format'); const builder = build({ header: false }); results = builder(schemas); }); @@ -150,7 +150,7 @@ describe('Testing Markdown Builder: YAML examples', () => { let results; before(async () => { - const schemas = await loadschemas('format'); + const schemas = await traverseSchemas('format'); const builder = build({ header: false, exampleFormat: 'yaml' }); results = builder(schemas); }); @@ -171,7 +171,7 @@ describe('Testing Markdown Builder: enums', () => { let results; before(async () => { - const schemas = await loadschemas('enums'); + const schemas = await traverseSchemas('enums'); const builder = build({ header: true, includeProperties: ['foo', 'bar'] }); results = builder(schemas); }); @@ -195,7 +195,7 @@ describe('Testing Markdown Builder: null', () => { let results; before(async () => { - const schemas = await loadschemas('null'); + const schemas = await traverseSchemas('null'); const builder = build({ header: true }); results = builder(schemas); }); @@ -212,7 +212,7 @@ describe('Testing Markdown Builder: additionalprops', () => { let results; before(async () => { - const schemas = await loadschemas('additionalprops'); + const schemas = await traverseSchemas('additionalprops'); const builder = build({ header: true }); results = builder(schemas); }); @@ -236,7 +236,7 @@ describe('Testing Markdown Builder: types', () => { let results; before(async () => { - const schemas = await loadschemas('types'); + const schemas = await traverseSchemas('types'); const builder = build({ header: true }); results = builder(schemas); }); @@ -271,7 +271,7 @@ describe('Testing Markdown Builder: identifiable', () => { let results; before(async () => { - const schemas = await loadschemas('identifiable'); + const schemas = await traverseSchemas('identifiable'); const builder = build({ header: true }); results = builder(schemas); }); @@ -287,7 +287,7 @@ describe('Testing Markdown Builder: arrays', () => { let results; before(async () => { - const schemas = await loadschemas('arrays'); + const schemas = await traverseSchemas('arrays'); const builder = build({ header: true }); results = builder(schemas); }); @@ -330,7 +330,7 @@ describe('Testing Markdown Builder: stringformats', () => { let results; before(async () => { - const schemas = await loadschemas('stringformats'); + const schemas = await traverseSchemas('stringformats'); const builder = build({ header: true }); results = builder(schemas); }); @@ -356,7 +356,7 @@ describe('Testing Markdown Builder: readme-1', () => { let results; before(async () => { - const schemas = await loadschemas('readme-1'); + const schemas = await traverseSchemas('readme-1'); const builder = build({ header: true, links: { abstract: 'fooabstract.html' } }); results = builder(schemas); }); @@ -416,7 +416,7 @@ describe('Testing Markdown Builder: Skip properties', () => { let schemas; before(async () => { - schemas = await loadschemas('skipproperties'); + schemas = await traverseSchemas('skipproperties'); }); it('Skipped properties exist', () => { diff --git a/test/readmeBuilder.test.js b/test/readmeBuilder.test.js index 4ab54c64..5c58e06e 100644 --- a/test/readmeBuilder.test.js +++ b/test/readmeBuilder.test.js @@ -12,7 +12,7 @@ /* eslint-env mocha */ /* eslint-disable no-unused-expressions */ const assert = require('assert'); -const { assertMarkdown, loadschemas } = require('./testUtils'); +const { assertMarkdown, traverseSchemas } = require('./testUtils'); const build = require('../lib/readmeBuilder'); const { loader } = require('../lib/schemaProxy'); @@ -28,7 +28,7 @@ describe('Testing Readme Builder', () => { }); it('Readme Builder builds a README for type', async () => { - const schemas = await loadschemas('type'); + const schemas = await traverseSchemas('type'); const builder = build({ readme: true }); const result = builder(schemas); @@ -38,7 +38,7 @@ describe('Testing Readme Builder', () => { }); it('Readme Builder builds a medium README for multiple Schemas', async () => { - const schemas = await loadschemas('readme-1'); + const schemas = await traverseSchemas('readme-1'); const builder = build({ readme: true }); const result = builder(schemas); @@ -59,7 +59,7 @@ describe('Testing Readme Builder', () => { const builder = build({ readme: true }); const schemaloader = loader(); const schemas = [ - schemaloader({ + schemaloader('example.schema.json', { type: 'object', title: 'Test Schema', description: 'Not much', @@ -76,7 +76,7 @@ describe('Testing Readme Builder', () => { title: 'An Array', }, }, - }, 'example.schema.json'), + }), ]; const result = builder(schemas); diff --git a/test/schemaProxy.test.js b/test/schemaProxy.test.js index 33242202..24998f68 100644 --- a/test/schemaProxy.test.js +++ b/test/schemaProxy.test.js @@ -17,14 +17,14 @@ const { loader, pointer, filename, id, titles, resolve, slug, meta, } = require('../lib/schemaProxy'); -const example = require('./fixtures/example/example.schema.json'); +const example = require('./fixtures/example/proxy.schema.json'); const referencing = { $schema: 'http://json-schema.org/draft-06/schema#', $id: 'https://example.com/schemas/referencing', title: 'Referencing', properties: { - $ref: 'https://example.com/schemas/example#/properties', + $ref: 'https://example.com/schemas/example-proxy#/properties', zap: { type: 'boolean', }, @@ -33,23 +33,23 @@ const referencing = { describe('Testing Schema Proxy', () => { it('Schema Proxy creates a JSON schema', () => { - const proxied = loader()(example, 'example.schema.json'); + const proxied = loader()('proxy.schema.json`', example); - assert.equal(proxied.title, 'Example'); + assert.equal(proxied.title, 'Example Proxy'); assert.equal(proxied.properties.foo.type, 'string'); }); it('Schema Proxy loads multiple JSON schemas', () => { const myloader = loader(); - const proxied1 = myloader(example, 'example.schema.json'); - const proxied2 = myloader(referencing, 'referencing.schema.json'); + const proxied1 = myloader('proxy.schema.json', example); + const proxied2 = myloader('referencing.schema.json', referencing); - assert.equal(proxied1.title, 'Example'); + assert.equal(proxied1.title, 'Example Proxy'); assert.equal(proxied2.$id, 'https://example.com/schemas/referencing'); }); it('Schema Proxy creates a JSON schema with Pointers', () => { - const proxied = loader()(example, 'example.schema.json'); + const proxied = loader()('proxy.schema.json', example); assert.equal(proxied[pointer], ''); assert.equal(proxied.properties[pointer], '/properties'); @@ -59,39 +59,39 @@ describe('Testing Schema Proxy', () => { }); it('Schema Proxy creates a JSON schema with ID References', () => { - const proxied = loader()(example, 'example.schema.json'); + const proxied = loader()('proxy.schema.json', example); - assert.equal(proxied[id], 'https://example.com/schemas/example'); + assert.equal(proxied[id], 'https://example.com/schemas/example-proxy'); assert.equal(proxied.properties[pointer], '/properties'); - assert.equal(proxied.properties[id], 'https://example.com/schemas/example'); + assert.equal(proxied.properties[id], 'https://example.com/schemas/example-proxy'); assert.equal(proxied.properties[pointer], '/properties'); - assert.equal(proxied.properties.foo[id], 'https://example.com/schemas/example'); - assert.equal(proxied['meta:license'][id], 'https://example.com/schemas/example'); - assert.equal(proxied.properties.baz.anyOf[0][id], 'https://example.com/schemas/example'); + assert.equal(proxied.properties.foo[id], 'https://example.com/schemas/example-proxy'); + assert.equal(proxied['meta:license'][id], 'https://example.com/schemas/example-proxy'); + assert.equal(proxied.properties.baz.anyOf[0][id], 'https://example.com/schemas/example-proxy'); }); it('Schema Proxy creates a JSON schema with Filename References', () => { - const proxied = loader()(example, 'example.schema.json'); + const proxied = loader()('proxy.schema.json', example); - assert.equal(proxied[filename], 'example.schema.json'); - assert.equal(proxied.properties[filename], 'example.schema.json'); - assert.equal(proxied.properties.foo[filename], 'example.schema.json'); - assert.equal(proxied['meta:license'][filename], 'example.schema.json'); - assert.equal(proxied.properties.baz.anyOf[0][filename], 'example.schema.json'); + assert.equal(proxied[filename], 'proxy.schema.json'); + assert.equal(proxied.properties[filename], 'proxy.schema.json'); + assert.equal(proxied.properties.foo[filename], 'proxy.schema.json'); + assert.equal(proxied['meta:license'][filename], 'proxy.schema.json'); + assert.equal(proxied.properties.baz.anyOf[0][filename], 'proxy.schema.json'); }); it('Schema Proxy creates a JSON schema with title References', () => { - const proxied = loader()(example, 'example.schema.json'); + const proxied = loader()('proxy.schema.json', example); - assert.deepStrictEqual(proxied[titles], ['Example']); - assert.deepStrictEqual(proxied.properties.zip[titles], ['Example', undefined, 'An object']); + assert.deepStrictEqual(proxied[titles], ['Example Proxy']); + assert.deepStrictEqual(proxied.properties.zip[titles], ['Example Proxy', undefined, 'An object']); }); it('Schema proxy resolves JSON Pointers', () => { const myloader = loader(); - const proxied1 = myloader(example, 'example.schema.json'); + const proxied1 = myloader('proxy.schema.json', example); assert.deepStrictEqual(proxied1.properties, proxied1[resolve]('/properties')); assert.deepStrictEqual(proxied1.properties.foo, proxied1[resolve]('/properties/foo')); @@ -100,8 +100,8 @@ describe('Testing Schema Proxy', () => { it('Schema proxy resolves Reference Pointers', () => { const myloader = loader(); - myloader(example, 'example.schema.json'); - const proxied2 = myloader(referencing, 'referencing.schema.json'); + myloader('proxy.schema.json', example); + const proxied2 = myloader('referencing.schema.json', referencing); assert.deepStrictEqual(new Set(Object.keys(proxied2.properties)), new Set([ '$ref', 'zap', // the two properties from the original declaration @@ -111,16 +111,16 @@ describe('Testing Schema Proxy', () => { it('Schema proxy generates unique names', () => { const myloader = loader(); - const proxied1 = myloader(example, 'example.schema.json'); - const proxied2 = myloader(referencing, 'referencing.schema.json'); - const proxied3 = myloader({ + const proxied1 = myloader('proxy.schema.json', example); + const proxied2 = myloader('referencing.schema.json', referencing); + const proxied3 = myloader('anotherreference.schema.json', { title: 'Referencing', - }, 'anotherreference.schema.json'); + }); - assert.equal(proxied1[slug], 'example'); + assert.equal(proxied1[slug], 'proxy'); - assert.equal(proxied1.properties.zip[slug], 'example-properties-an-object'); - assert.equal(proxied1.properties.zup[slug], 'example-properties-an-object-1'); // avoid duplicates + assert.equal(proxied1.properties.zip[slug], 'proxy-properties-an-object'); + assert.equal(proxied1.properties.zup[slug], 'proxy-properties-an-object-1'); // avoid duplicates assert.equal(proxied2[slug], 'referencing'); assert.equal(proxied2[slug], 'referencing'); // make sure the slug stays stable @@ -132,7 +132,7 @@ describe('Testing Schema Proxy', () => { const examplefile = path.resolve(__dirname, '..', 'examples', 'schemas', 'definitions.schema.json'); // eslint-disable-next-line import/no-dynamic-require, global-require - const exampleschema = myloader(require(examplefile), examplefile); + const exampleschema = myloader(examplefile, require(examplefile)); assert.equal(exampleschema[meta].shortdescription, 'This is an example of using a definitions object within a schema'); }); @@ -152,7 +152,7 @@ describe('Testing Schema Proxy', () => { const schemas = files.map((file) => { const fname = path.resolve(__dirname, '..', 'examples', 'schemas', file); // eslint-disable-next-line import/no-dynamic-require, global-require - return myloader(require(fname), fname); + return myloader(fname, require(fname)); }); assert.equal(schemas[0][slug], 'abstract'); @@ -178,7 +178,7 @@ describe('Testing Schema Proxy', () => { const schemas = files.map((file) => { const fname = path.resolve(__dirname, 'fixtures', 'deadref', file); // eslint-disable-next-line import/no-dynamic-require, global-require - return myloader(require(fname), fname); + return myloader(fname, require(fname)); }); assert.equal(schemas[0][slug], 'deadref'); diff --git a/test/testUtils.js b/test/testUtils.js index 081dd3ca..d0574b43 100644 --- a/test/testUtils.js +++ b/test/testUtils.js @@ -73,16 +73,24 @@ ${inspect(node)}`); return null; } -async function loadschemas(dir) { +async function loadSchemas(dir) { + const schemaDir = path.resolve(__dirname, 'fixtures', dir); + const schemas = await readdirp.promise(schemaDir, { fileFilter: '*.schema.json' }); + + return schemas.map((schema) => ({ + fileName: schema.basename, + fullPath: schema.fullPath, + })); +} + +async function traverseSchemas(dir) { + const schemas = await loadSchemas(dir); const schemaloader = loader(); - const schemadir = path.resolve(__dirname, 'fixtures', dir); - const schemas = await readdirp.promise(schemadir, { fileFilter: '*.schema.json' }); - return traverse(schemas - .map(({ fullPath }) => schemaloader( - // eslint-disable-next-line global-require, import/no-dynamic-require - require(fullPath), fullPath, - ))); + return traverse(schemas.map(({ fullPath }) => schemaloader( + // eslint-disable-next-line global-require, import/no-dynamic-require + fullPath, require(fullPath), + ))); } -module.exports = { assertMarkdown, loadschemas }; +module.exports = { assertMarkdown, loadSchemas, traverseSchemas }; diff --git a/test/testintegration.test.js b/test/testintegration.test.js index 42a82be9..fc7fee2a 100644 --- a/test/testintegration.test.js +++ b/test/testintegration.test.js @@ -50,10 +50,11 @@ describe('Integration Test', () => { }); ['arrays', 'cyclic'].forEach((dir) => it(`CLI processes ${dir} directory`, async () => { - const res = await main((`jsonschema2md -d test/fixtures/${dir} -o tmp -x tmp`).split(' ')); + const res = await main((`jsonschema2md -d test/fixtures/${dir} -o tmp -x tmp -m input1=test1 -m input2=test2`).split(' ')); console.log('done!', res); const readme = await fs.stat(path.resolve(__dirname, '..', 'tmp', 'README.md')); assert.ok(readme.isFile()); + assert.notStrictEqual(readme.size, 0); })); ['json-logic-js/schemas'].forEach((dir) => it(`CLI processes ${dir} directory`, async () => { @@ -61,5 +62,6 @@ describe('Integration Test', () => { console.log('done!', res); const readme = await fs.stat(path.resolve(__dirname, '..', 'tmp', 'README.md')); assert.ok(readme.isFile()); + assert.notStrictEqual(readme.size, 0); })); }); diff --git a/test/traverseSchema.test.js b/test/traverseSchema.test.js index 51cf9892..299c0005 100644 --- a/test/traverseSchema.test.js +++ b/test/traverseSchema.test.js @@ -18,78 +18,16 @@ const { loader, filename, } = require('../lib/schemaProxy'); +const example = require('./fixtures/example/traverse.schema.json'); const traverse = require('../lib/traverseSchema'); -const example = { - 'meta:license': [ - 'Copyright 2017 Adobe Systems Incorporated. All rights reserved.', - "This file is licensed to you under the Apache License, Version 2.0 (the 'License');", - 'you may not use this file except in compliance with the License. You may obtain a copy', - 'of the License at http://www.apache.org/licenses/LICENSE-2.0', - ], - $schema: 'http://json-schema.org/draft-06/schema#', - $id: 'https://example.com/schemas/example', - title: 'Example', - type: 'object', - description: - 'This is an example schema with examples. Too many examples? There can never be too many examples!', - properties: { - foo: { - type: 'string', - description: 'A simple string.', - examples: ['bar'], - version: '1.0.0', - testProperty: 'test', - }, - bar: { - type: 'string', - description: 'A simple string.', - examples: ['bar', 'baz'], - version: '1.0.0', - testProperty: 'test', - }, - zip: { - type: 'object', - title: 'An object', - }, - zup: { - type: 'object', - title: 'An object', - }, - baz: { - anyOf: [ - { $ref: '#/properties/foo' }, - { $ref: '#/properties/bar' }, - ], - }, - properties: { - type: 'object', - description: 'properties witin properties', - }, - required: { - type: 'array', - examples: [ - [ - { - properies: [], - }, - ], - ], - }, - examples: { - type: 'number', - examples: [1, 2, 3], - }, - }, -}; - describe('Testing Schema Traversal', () => { it('Schema Traversal generates a list', () => { - const proxied = loader()(example, 'example.schema.json'); + const proxied = loader()('traverse.schema.json', example); const schemas = traverse([proxied]); assert.equal(schemas.length, 11); - assert.equal(schemas[7][filename], 'example.schema.json'); + assert.equal(schemas[7][filename], 'traverse.schema.json'); }); it('Cyclic Schema Traversal generates a list', async () => { @@ -98,9 +36,9 @@ describe('Testing Schema Traversal', () => { const three = await fs.readJson(path.resolve(__dirname, 'fixtures', 'cyclic', 'three.schema.json')); const myloader = loader(); - const proxiedone = myloader(one, path.resolve(__dirname, 'fixtures', 'cyclic', 'one.schema.json')); - const proxiedtwo = myloader(two, path.resolve(__dirname, 'fixtures', 'cyclic', 'two.schema.json')); - const proxiedthree = myloader(three, path.resolve(__dirname, 'fixtures', 'cyclic', 'three.schema.json')); + const proxiedone = myloader(path.resolve(__dirname, 'fixtures', 'cyclic', 'one.schema.json'), one); + const proxiedtwo = myloader(path.resolve(__dirname, 'fixtures', 'cyclic', 'two.schema.json'), two); + const proxiedthree = myloader(path.resolve(__dirname, 'fixtures', 'cyclic', 'three.schema.json'), three); const schemas = traverse([proxiedone, proxiedtwo, proxiedthree]); diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 00000000..620251bf --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,14 @@ +{ + "compilerOptions": { + "target": "es6", + "module": "commonjs", + "moduleResolution": "node", + "allowJs": true, + "declaration": true, + "emitDeclarationOnly": true, + "removeComments": false, + "outDir": "types" + }, + "include": ["lib/index.js"], + "exclude": ["**/locales/**/*"] +} diff --git a/types/api.d.ts b/types/api.d.ts new file mode 100644 index 00000000..d9486275 --- /dev/null +++ b/types/api.d.ts @@ -0,0 +1,72 @@ +/* + * Copyright 2021 Adobe. All rights reserved. + * This file is licensed to you under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. You may obtain a copy + * of the License at http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software distributed under + * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS + * OF ANY KIND, either express or implied. See the License for the specific language + * governing permissions and limitations under the License. + */ + +import { JSONSchema4, JSONSchema6, JSONSchema7 } from "json-schema"; +import { Root } from "mdast"; +import { filename, fullpath } from "../lib/symbols"; + +export type UniqueSymbols = { + readonly pointer: unique symbol; + readonly filename: unique symbol; + readonly fullpath: unique symbol; + readonly id: unique symbol; + readonly titles: unique symbol; + readonly resolve: unique symbol; + readonly slug: unique symbol; + readonly meta: unique symbol; + readonly parent: unique symbol; +}; + +export type JsonSchema = JSONSchema4 | JSONSchema6 | JSONSchema7; +export type SchemaList = { + fileName: string; + fullPath: string; +}; +export type SchemaContent = { + fileName: string; + fullPath?: string; + content: JsonSchema; +}; +export type SchemaFiles = (SchemaList | SchemaContent)[]; +export type ExtendedJsonSchema = JsonSchema & { + [filename]: string; + [fullpath]?: string; +}; + +export type MarkdownAst = Root; +export type MarkdownContent = { + fileName: string; + fullPath: string; + markdownAst?: MarkdownAst; + content?: string; +}; +export type ReadmeContent = MarkdownContent & { + fileName: "README.md"; +}; + +/** + * GeneratedOutput + */ +export type GeneratedOutput = { + /** + * Input JSON schemas + */ + schema: SchemaContent[]; + /** + * Ouput Readme file + */ + readme?: ReadmeContent; + /** + * Output markdown data + */ + markdown: MarkdownContent[]; +}; diff --git a/types/formatInfo.d.ts b/types/formatInfo.d.ts new file mode 100644 index 00000000..293b02b5 --- /dev/null +++ b/types/formatInfo.d.ts @@ -0,0 +1,19 @@ +export function formatmeta(schema: any): { + longcomment: import("unist").Node; + shortcomment: any; + comment: string; + longdescription: import("unist").Node; + shortdescription: any; + description: any; + abstract: boolean; + extensible: boolean; + status: any; + identifiable: string; + custom: boolean; + additional: boolean; + definedin: { + text: any; + link: any; + }; + restrictions: string; +}; diff --git a/types/formattingTools.d.ts b/types/formattingTools.d.ts new file mode 100644 index 00000000..85ca1de1 --- /dev/null +++ b/types/formattingTools.d.ts @@ -0,0 +1,2 @@ +export function gentitle(titles: any, type: any): any; +export function gendescription(schema: any): any; diff --git a/types/index.d.ts b/types/index.d.ts new file mode 100644 index 00000000..f50b0a92 --- /dev/null +++ b/types/index.d.ts @@ -0,0 +1,75 @@ +export type JsonSchema = import("../types/api").JsonSchema; +export type SchemaList = import("../types/api").SchemaList; +export type SchemaContent = import("../types/api").SchemaContent; +export type SchemaFiles = import("../types/api").SchemaFiles; +export type GeneratedOutput = import("../types/api").GeneratedOutput; +/** + * @typedef {import("../types/api").JsonSchema} JsonSchema + */ +/** + * @typedef {import("../types/api").SchemaList} SchemaList + */ +/** + * @typedef {import("../types/api").SchemaContent} SchemaContent + */ +/** + * @typedef {import("../types/api").SchemaFiles} SchemaFiles + */ +/** + * @typedef {import("../types/api").GeneratedOutput} GeneratedOutput + */ +/** + * Public API for jsonschema2md that can be used to turn JSON Schema files + * into readable Markdown documentation. + * @param {JsonSchema | SchemaFiles} schema JSON Schema input to get data from + * @param {Object} options Additional options for generation + * @param {string} [options.schemaPath] - (optional) Path to directory containing all JSON Schemas + * or a single JSON Schema file. This will be considered as the baseURL. + * @param {string} [options.outDir] - (optional) Path to output directory. Generating files will + * be skipped if directory is not specified. + * @param {{ [key:string]: string }} [options.metadata] - (optional) Add metadata elements to + * .md files. + * @param {string} [options.schemaOut] - (optional) Output JSON Schema files including + * description and validated examples in the specified folder. + * @param {boolean} [options.includeReadme=true] - (optional) Generate a README.md file in the + * output directory. + * @param {{ [key:string]: string }[]} [options.links] - (optional) Add this file as a link + * explaining the specified attribute. + * @param {string} [options.i18n="locales/"] - (optional) Path to a locales folder with + * JSON files used for internationalization. + * @param {"en_US" | "de"} [options.language="en_US"] - (optional) Selected language. + * @param {"json" | "yaml"} [options.exampleFormat="json"] - (optional) How to format examples. + * @param {string[]} [options.includeProperties=[]] - (optional) Name of custom properties + * which should be also in the description of an element. + * @param {boolean} [options.header=true] - (optional) Whether or not to include the header + * in markdown. + * @param {string[]} [options.skipProperties=[]] - (optional) Name of a default property to + * skip in markdown. + * @returns {GeneratedOutput} List of raw markdown that were generated from input schema. + */ +export function jsonschema2md(schema: JsonSchema | import("../types/api").SchemaFiles, options: { + schemaPath?: string; + outDir?: string; + metadata?: { + [key: string]: string; + }; + schemaOut?: string; + includeReadme?: boolean; + links?: { + [key: string]: string; + }[]; + i18n?: string; + language?: "en_US" | "de"; + exampleFormat?: "json" | "yaml"; + includeProperties?: string[]; + header?: boolean; + skipProperties?: string[]; +}): GeneratedOutput; +/** + * Main function used in the CLI. + * @param {{ [key:string]: unknown }} args CLI arguments from user input + * @returns The generated Markdown files to the specified output directory + */ +export function main(args: { + [key: string]: unknown; +}): Promise; diff --git a/types/keywords.d.ts b/types/keywords.d.ts new file mode 100644 index 00000000..2edc5a82 --- /dev/null +++ b/types/keywords.d.ts @@ -0,0 +1,2 @@ +export function keyword(str: any): any; +export function report(): Set; diff --git a/types/markdownBuilder.d.ts b/types/markdownBuilder.d.ts new file mode 100644 index 00000000..584730e0 --- /dev/null +++ b/types/markdownBuilder.d.ts @@ -0,0 +1,9 @@ +export = build; +declare function build({ header, links, includeProperties, rewritelinks, exampleFormat, skipProperties, }?: { + header: any; + links?: {}; + includeProperties?: any[]; + rewritelinks?: (x: any) => any; + exampleFormat?: string; + skipProperties?: any[]; +}): (schemas: any) => any; diff --git a/types/readmeBuilder.d.ts b/types/readmeBuilder.d.ts new file mode 100644 index 00000000..417f8594 --- /dev/null +++ b/types/readmeBuilder.d.ts @@ -0,0 +1,6 @@ +export = build; +/** + * Generate the README.md + * @param {object} opts + */ +declare function build({ readme }: object): (schemas: any) => import("unist").Parent; diff --git a/types/schemaProxy.d.ts b/types/schemaProxy.d.ts new file mode 100644 index 00000000..a6134425 --- /dev/null +++ b/types/schemaProxy.d.ts @@ -0,0 +1,14 @@ +declare const _exports: { + pointer: unique symbol; + filename: unique symbol; + fullpath: unique symbol; + id: unique symbol; + titles: unique symbol; + resolve: unique symbol; + slug: unique symbol; + meta: unique symbol; + parent: unique symbol; + loader: () => (name: string, schema: any) => any; +}; +export = _exports; +export function loader(): (name: string, schema: any) => any; diff --git a/types/symbols.d.ts b/types/symbols.d.ts new file mode 100644 index 00000000..1a078989 --- /dev/null +++ b/types/symbols.d.ts @@ -0,0 +1,12 @@ +export = symbols; +/** + * @typedef {import("../types/api").UniqueSymbols} UniqueSymbols + */ +/** + * @type {UniqueSymbols} + * */ +declare const symbols: UniqueSymbols; +declare namespace symbols { + export { UniqueSymbols }; +} +type UniqueSymbols = import("../types/api").UniqueSymbols; diff --git a/types/traverseSchema.d.ts b/types/traverseSchema.d.ts new file mode 100644 index 00000000..6d10d78f --- /dev/null +++ b/types/traverseSchema.d.ts @@ -0,0 +1,7 @@ +export = traverse; +/** + * Traverses a Schema node (containing a JSON Schema) to find sub-schemas, + * which are then emitted as asequence. + * @param {Schema} node + */ +declare function traverse(schemas: any): any[]; diff --git a/types/writeMarkdown.d.ts b/types/writeMarkdown.d.ts new file mode 100644 index 00000000..f39c9452 --- /dev/null +++ b/types/writeMarkdown.d.ts @@ -0,0 +1,56 @@ +export type MarkdownAst = import("../types/api").MarkdownAst; +export type MarkdownContent = import("../types/api").MarkdownContent; +export type ReadmeContent = import("../types/api").ReadmeContent; +export type MarkdownAstFiles = { + [name: string]: import("mdast").Root; +}; +/** + * @typedef {import("../types/api").MarkdownAst} MarkdownAst + */ +/** + * @typedef {import("../types/api").MarkdownContent} MarkdownContent + */ +/** + * @typedef {import("../types/api").ReadmeContent} ReadmeContent + */ +/** + * @typedef {{ [name: string]: MarkdownAst }} MarkdownAstFiles + */ +/** + * Write the Markdown to filesystem or an object + * @param {Object} options + * @param {string} [options.out] - (optional) Directory where the files will be saved + * @param {Function} [options.error] - (optional) Error logger + * @param {Function} [options.info] - (optional) Info logger + * @param {Function} [options.debug] - (optional) Debug logger + * @param {{ [key: string]: string }} [options.meta] - (optional) Metadata to be added to Markdown + * @returns {(astFiles: MarkdownAstFiles) => MarkdownContent[]} + */ +export function writemarkdown({ out, error, meta, }: { + out?: string; + error?: Function; + info?: Function; + debug?: Function; + meta?: { + [key: string]: string; + }; +}): (astFiles: MarkdownAstFiles) => MarkdownContent[]; +/** + * Write the Readme Markdown to filesystem or an object + * @param {Object} options + * @param {string} [options.out] - (optional) Directory where the files will be saved + * @param {Function} [options.error] - (optional) Error logger + * @param {Function} [options.info] - (optional) Info logger + * @param {Function} [options.debug] - (optional) Debug logger + * @param {{ [key: string]: string }} [options.meta] - (optional) Metadata to be added to Markdown + * @returns {(markdownAst: MarkdownAst) => ReadmeContent} + */ +export function writereadme({ out, error, info, meta, }: { + out?: string; + error?: Function; + info?: Function; + debug?: Function; + meta?: { + [key: string]: string; + }; +}): (markdownAst: MarkdownAst) => ReadmeContent; diff --git a/types/writeSchema.d.ts b/types/writeSchema.d.ts new file mode 100644 index 00000000..cb261da0 --- /dev/null +++ b/types/writeSchema.d.ts @@ -0,0 +1,19 @@ +export type ExtendedJsonSchema = import("../types/api").ExtendedJsonSchema; +export type SchemaContent = import("../types/api").SchemaContent; +/** + * @typedef {import("../types/api").ExtendedJsonSchema} ExtendedJsonSchema + */ +/** + * @typedef {import("../types/api").SchemaContent} SchemaContent + */ +/** + * Write the JSON Schemas to filesystem or an object + * @param {Object} options + * @param {string} [options.schemadir] - (optional) Directory where the files will be saved + * @param {string} [options.origindir] - (optional) Directory where the files were loaded from + * @returns {(schemas: ExtendedJsonSchema[]) => SchemaContent[]} + */ +export function writeSchema({ schemadir, origindir }: { + schemadir?: string; + origindir?: string; +}): (schemas: ExtendedJsonSchema[]) => SchemaContent[];