UNPKG

@angular/cli

Version:
145 lines 20.5 kB
"use strict"; /** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ Object.defineProperty(exports, "__esModule", { value: true }); exports.parseJsonSchemaToOptions = void 0; const core_1 = require("@angular-devkit/core"); async function parseJsonSchemaToOptions(registry, schema, interactive = true) { const options = []; function visitor(current, pointer, parentSchema) { if (!parentSchema) { // Ignore root. return; } else if (pointer.split(/\/(?:properties|items|definitions)\//g).length > 2) { // Ignore subitems (objects or arrays). return; } else if (core_1.json.isJsonArray(current)) { return; } if (pointer.indexOf('/not/') != -1) { // We don't support anyOf/not. throw new Error('The "not" keyword is not supported in JSON Schema.'); } const ptr = core_1.json.schema.parseJsonPointer(pointer); const name = ptr[ptr.length - 1]; if (ptr[ptr.length - 2] != 'properties') { // Skip any non-property items. return; } const typeSet = core_1.json.schema.getTypesOfSchema(current); if (typeSet.size == 0) { throw new Error('Cannot find type of schema.'); } // We only support number, string or boolean (or array of those), so remove everything else. const types = [...typeSet].filter((x) => { switch (x) { case 'boolean': case 'number': case 'string': return true; case 'array': // Only include arrays if they're boolean, string or number. if (core_1.json.isJsonObject(current.items) && typeof current.items.type == 'string' && ['boolean', 'number', 'string'].includes(current.items.type)) { return true; } return false; default: return false; } }); if (types.length == 0) { // This means it's not usable on the command line. e.g. an Object. return; } // Only keep enum values we support (booleans, numbers and strings). const enumValues = ((core_1.json.isJsonArray(current.enum) && current.enum) || []).filter((x) => { switch (typeof x) { case 'boolean': case 'number': case 'string': return true; default: return false; } }); let defaultValue = undefined; if (current.default !== undefined) { switch (types[0]) { case 'string': if (typeof current.default == 'string') { defaultValue = current.default; } break; case 'number': if (typeof current.default == 'number') { defaultValue = current.default; } break; case 'boolean': if (typeof current.default == 'boolean') { defaultValue = current.default; } break; } } const type = types[0]; const $default = current.$default; const $defaultIndex = core_1.json.isJsonObject($default) && $default['$source'] == 'argv' ? $default['index'] : undefined; const positional = typeof $defaultIndex == 'number' ? $defaultIndex : undefined; let required = core_1.json.isJsonArray(schema.required) ? schema.required.includes(name) : false; if (required && interactive && current['x-prompt']) { required = false; } const alias = core_1.json.isJsonArray(current.aliases) ? [...current.aliases].map((x) => '' + x) : current.alias ? ['' + current.alias] : []; const format = typeof current.format == 'string' ? current.format : undefined; const visible = current.visible === undefined || current.visible === true; const hidden = !!current.hidden || !visible; const xUserAnalytics = current['x-user-analytics']; const userAnalytics = typeof xUserAnalytics === 'string' ? xUserAnalytics : undefined; // Deprecated is set only if it's true or a string. const xDeprecated = current['x-deprecated']; const deprecated = xDeprecated === true || typeof xDeprecated === 'string' ? xDeprecated : undefined; const option = { name, description: '' + (current.description === undefined ? '' : current.description), type, default: defaultValue, choices: enumValues.length ? enumValues : undefined, required, alias, format, hidden, userAnalytics, deprecated, positional, }; options.push(option); } const flattenedSchema = await registry.ɵflatten(schema); core_1.json.schema.visitJsonSchema(flattenedSchema, visitor); // Sort by positional and name. return options.sort((a, b) => { if (a.positional) { return b.positional ? a.positional - b.positional : a.name.localeCompare(b.name); } else if (b.positional) { return -1; } return a.name.localeCompare(b.name); }); } exports.parseJsonSchemaToOptions = parseJsonSchemaToOptions; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"json-schema.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular/cli/src/command-builder/utilities/json-schema.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport { json } from '@angular-devkit/core';\nimport yargs from 'yargs';\n\n/**\n * An option description.\n */\nexport interface Option extends yargs.Options {\n  /**\n   * The name of the option.\n   */\n  name: string;\n\n  /**\n   * Whether this option is required or not.\n   */\n  required?: boolean;\n\n  /**\n   * Format field of this option.\n   */\n  format?: string;\n\n  /**\n   * Whether this option should be hidden from the help output. It will still show up in JSON help.\n   */\n  hidden?: boolean;\n\n  /**\n   * If this option can be used as an argument, the position of the argument. Otherwise omitted.\n   */\n  positional?: number;\n\n  /**\n   * Whether or not to report this option to the Angular Team, and which custom field to use.\n   * If this is falsey, do not report this option.\n   */\n  userAnalytics?: string;\n}\n\nexport async function parseJsonSchemaToOptions(\n  registry: json.schema.SchemaRegistry,\n  schema: json.JsonObject,\n  interactive = true,\n): Promise<Option[]> {\n  const options: Option[] = [];\n\n  function visitor(\n    current: json.JsonObject | json.JsonArray,\n    pointer: json.schema.JsonPointer,\n    parentSchema?: json.JsonObject | json.JsonArray,\n  ) {\n    if (!parentSchema) {\n      // Ignore root.\n      return;\n    } else if (pointer.split(/\\/(?:properties|items|definitions)\\//g).length > 2) {\n      // Ignore subitems (objects or arrays).\n      return;\n    } else if (json.isJsonArray(current)) {\n      return;\n    }\n\n    if (pointer.indexOf('/not/') != -1) {\n      // We don't support anyOf/not.\n      throw new Error('The \"not\" keyword is not supported in JSON Schema.');\n    }\n\n    const ptr = json.schema.parseJsonPointer(pointer);\n    const name = ptr[ptr.length - 1];\n\n    if (ptr[ptr.length - 2] != 'properties') {\n      // Skip any non-property items.\n      return;\n    }\n\n    const typeSet = json.schema.getTypesOfSchema(current);\n\n    if (typeSet.size == 0) {\n      throw new Error('Cannot find type of schema.');\n    }\n\n    // We only support number, string or boolean (or array of those), so remove everything else.\n    const types = [...typeSet].filter((x) => {\n      switch (x) {\n        case 'boolean':\n        case 'number':\n        case 'string':\n          return true;\n\n        case 'array':\n          // Only include arrays if they're boolean, string or number.\n          if (\n            json.isJsonObject(current.items) &&\n            typeof current.items.type == 'string' &&\n            ['boolean', 'number', 'string'].includes(current.items.type)\n          ) {\n            return true;\n          }\n\n          return false;\n\n        default:\n          return false;\n      }\n    }) as ('string' | 'number' | 'boolean' | 'array')[];\n\n    if (types.length == 0) {\n      // This means it's not usable on the command line. e.g. an Object.\n      return;\n    }\n\n    // Only keep enum values we support (booleans, numbers and strings).\n    const enumValues = ((json.isJsonArray(current.enum) && current.enum) || []).filter((x) => {\n      switch (typeof x) {\n        case 'boolean':\n        case 'number':\n        case 'string':\n          return true;\n\n        default:\n          return false;\n      }\n    }) as (string | true | number)[];\n\n    let defaultValue: string | number | boolean | undefined = undefined;\n    if (current.default !== undefined) {\n      switch (types[0]) {\n        case 'string':\n          if (typeof current.default == 'string') {\n            defaultValue = current.default;\n          }\n          break;\n        case 'number':\n          if (typeof current.default == 'number') {\n            defaultValue = current.default;\n          }\n          break;\n        case 'boolean':\n          if (typeof current.default == 'boolean') {\n            defaultValue = current.default;\n          }\n          break;\n      }\n    }\n\n    const type = types[0];\n    const $default = current.$default;\n    const $defaultIndex =\n      json.isJsonObject($default) && $default['$source'] == 'argv' ? $default['index'] : undefined;\n    const positional: number | undefined =\n      typeof $defaultIndex == 'number' ? $defaultIndex : undefined;\n\n    let required = json.isJsonArray(schema.required) ? schema.required.includes(name) : false;\n    if (required && interactive && current['x-prompt']) {\n      required = false;\n    }\n\n    const alias = json.isJsonArray(current.aliases)\n      ? [...current.aliases].map((x) => '' + x)\n      : current.alias\n      ? ['' + current.alias]\n      : [];\n    const format = typeof current.format == 'string' ? current.format : undefined;\n    const visible = current.visible === undefined || current.visible === true;\n    const hidden = !!current.hidden || !visible;\n\n    const xUserAnalytics = current['x-user-analytics'];\n    const userAnalytics = typeof xUserAnalytics === 'string' ? xUserAnalytics : undefined;\n\n    // Deprecated is set only if it's true or a string.\n    const xDeprecated = current['x-deprecated'];\n    const deprecated =\n      xDeprecated === true || typeof xDeprecated === 'string' ? xDeprecated : undefined;\n\n    const option: Option = {\n      name,\n      description: '' + (current.description === undefined ? '' : current.description),\n      type,\n      default: defaultValue,\n      choices: enumValues.length ? enumValues : undefined,\n      required,\n      alias,\n      format,\n      hidden,\n      userAnalytics,\n      deprecated,\n      positional,\n    };\n\n    options.push(option);\n  }\n\n  const flattenedSchema = await registry.ɵflatten(schema);\n  json.schema.visitJsonSchema(flattenedSchema, visitor);\n\n  // Sort by positional and name.\n  return options.sort((a, b) => {\n    if (a.positional) {\n      return b.positional ? a.positional - b.positional : a.name.localeCompare(b.name);\n    } else if (b.positional) {\n      return -1;\n    }\n\n    return a.name.localeCompare(b.name);\n  });\n}\n"]}