UNPKG

projen

Version:

CDK for software projects

221 lines • 33.1 kB
"use strict"; var _a, _b; Object.defineProperty(exports, "__esModule", { value: true }); exports.Pom = exports.ChecksumPolicy = exports.UpdatePolicy = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const _resolve_1 = require("../_resolve"); const component_1 = require("../component"); const dependencies_1 = require("../dependencies"); const semver_1 = require("../util/semver"); const xmlfile_1 = require("../xmlfile"); const POM_XML_ATTRS = { "@xsi:schemaLocation": "http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd", "@xmlns": "http://maven.apache.org/POM/4.0.0", "@xmlns:xsi": "http://www.w3.org/2001/XMLSchema-instance", }; class UpdatePolicy { /** * Updates at an interval of X minutes. */ static interval(minutes) { return `interval:${minutes}`; } } exports.UpdatePolicy = UpdatePolicy; _a = JSII_RTTI_SYMBOL_1; UpdatePolicy[_a] = { fqn: "projen.java.UpdatePolicy", version: "0.95.2" }; UpdatePolicy.ALWAYS = "always"; UpdatePolicy.DAILY = "daily"; UpdatePolicy.NEVER = "never"; var ChecksumPolicy; (function (ChecksumPolicy) { ChecksumPolicy["IGNORE"] = "ignore"; ChecksumPolicy["FAIL"] = "fail"; ChecksumPolicy["WARN"] = "warn"; })(ChecksumPolicy || (exports.ChecksumPolicy = ChecksumPolicy = {})); /** * A Project Object Model or POM is the fundamental unit of work in Maven. It is * an XML file that contains information about the project and configuration * details used by Maven to build the project. */ class Pom extends component_1.Component { constructor(project, options) { super(project); this.properties = {}; this.repositories = []; this.pluginRepositories = []; this.fileName = "pom.xml"; this.groupId = options.groupId; this.artifactId = options.artifactId; this.version = options.version; this.packaging = options.packaging ?? "jar"; this.name = project.name; this.description = options.description; this.url = options.url; this.parentPom = options.parentPom; new xmlfile_1.XmlFile(project, this.fileName, { obj: () => this.synthPom() }); } /** * Adds a key/value property to the pom. * @param key the key * @param value the value */ addProperty(key, value) { this.properties[key] = value; } /** * Adds a runtime dependency. * * @param spec Format `<groupId>/<artifactId>@<semver>` */ addDependency(spec) { this.project.deps.addDependency(spec, dependencies_1.DependencyType.RUNTIME); } /** * Adds a test dependency. * * @param spec Format `<groupId>/<artifactId>@<semver>` */ addTestDependency(spec) { this.project.deps.addDependency(spec, dependencies_1.DependencyType.TEST); } /** * Adds a build plugin to the pom. * * The plug in is also added as a BUILD dep to the project. * * @param spec dependency spec (`group/artifact@version`) * @param options plugin options */ addPlugin(spec, options = {}) { for (const dep of options.dependencies ?? []) { this.project.deps.addDependency(dep, dependencies_1.DependencyType.BUILD); } return this.project.deps.addDependency(spec, dependencies_1.DependencyType.BUILD, options); } /** * Adds a repository to the pom * @param repository the repository to add */ addRepository(repository) { this.repositories.push(repository); } /* * Adds a repository for plugins to the pom * @param repository the repository to add */ addPluginRepository(repository) { this.pluginRepositories.push(repository); } synthPom() { return (0, _resolve_1.resolve)({ project: { ...POM_XML_ATTRS, modelVersion: "4.0.0", groupId: this.groupId, artifactId: this.artifactId, version: this.version, packaging: this.packaging, name: this.name, description: this.description, url: this.url, properties: this.properties, parent: this.parentPom, ...this.synthRepositories(), ...this.synthDependencies(), ...this.synthPluginRepositories(), }, }, { omitEmpty: true }); } synthDependencies() { const deps = this.project.deps.all; if (deps.length === 0) { return; } const dependencies = []; const plugins = []; for (const dep of deps) { switch (dep.type) { case dependencies_1.DependencyType.PEER: case dependencies_1.DependencyType.RUNTIME: dependencies.push(mavenCoords(dep)); break; case dependencies_1.DependencyType.TEST: dependencies.push({ ...mavenCoords(dep), scope: "test", }); break; // build maps to plugins case dependencies_1.DependencyType.BUILD: plugins.push({ ...mavenCoords(dep), ...pluginConfig(dep.metadata), }); break; default: throw new Error(`unsupported dependency type: ${dep.type}`); } } return { build: { plugins: { plugin: plugins } }, dependencies: { dependency: dependencies }, }; } synthRepositories() { if (this.repositories.length === 0) { return; } return { repositories: { repository: this.repositories }, }; } synthPluginRepositories() { if (this.pluginRepositories.length === 0) { return; } return { pluginRepositories: { pluginRepository: this.pluginRepositories }, }; } } exports.Pom = Pom; _b = JSII_RTTI_SYMBOL_1; Pom[_b] = { fqn: "projen.java.Pom", version: "0.95.2" }; /** * Parses maven groupId and artifactId from a dependency name. * * name <=> <groupId>/<artifactId> * version <=> <version> */ function mavenCoords(dep) { const name = dep.name; const parts = name.split("/"); if (parts.length !== 2) { throw new Error(`invalid maven coordinates in dependency named "${name}". format is "<groupId>/<artifactId>". For example "org.junit.jupiter/junit-jupiter-engine"`); } return { groupId: parts[0], artifactId: parts[1], version: dep.version ? (0, semver_1.toMavenVersionRange)(dep.version) : undefined, }; } function pluginConfig(options = {}) { return { configuration: options.configuration, dependencies: options.dependencies && options.dependencies.length > 0 ? { dependency: options.dependencies?.map((d) => mavenCoords(dependencies_1.Dependencies.parseDependency(d))), } : undefined, executions: { execution: options.executions?.map((e) => ({ id: e.id, goals: e.goals.map((goal) => ({ goal })), phase: e.phase, configuration: e.configuration, })), }, }; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pom.js","sourceRoot":"","sources":["../../src/java/pom.ts"],"names":[],"mappings":"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","sourcesContent":["import { resolve } from \"../_resolve\";\nimport { Component } from \"../component\";\nimport {\n  Dependencies,\n  DependencyCoordinates,\n  DependencyType,\n} from \"../dependencies\";\nimport { Project } from \"../project\";\nimport { toMavenVersionRange } from \"../util/semver\";\nimport { XmlFile } from \"../xmlfile\";\n\nconst POM_XML_ATTRS = {\n  \"@xsi:schemaLocation\":\n    \"http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd\",\n  \"@xmlns\": \"http://maven.apache.org/POM/4.0.0\",\n  \"@xmlns:xsi\": \"http://www.w3.org/2001/XMLSchema-instance\",\n};\n\n/**\n * Options for `Pom`.\n */\nexport interface PomOptions {\n  /**\n   * This is generally unique amongst an organization or a project. For example,\n   * all core Maven artifacts do (well, should) live under the groupId\n   * org.apache.maven. Group ID's do not necessarily use the dot notation, for\n   * example, the junit project. Note that the dot-notated groupId does not have\n   * to correspond to the package structure that the project contains. It is,\n   * however, a good practice to follow. When stored within a repository, the\n   * group acts much like the Java packaging structure does in an operating\n   * system. The dots are replaced by OS specific directory separators (such as\n   * '/' in Unix) which becomes a relative directory structure from the base\n   * repository. In the example given, the org.codehaus.mojo group lives within\n   * the directory $M2_REPO/org/codehaus/mojo.\n   * @default \"org.acme\"\n   * @featured\n   */\n  readonly groupId: string;\n\n  /**\n   * The artifactId is generally the name that the project is known by. Although\n   * the groupId is important, people within the group will rarely mention the\n   * groupId in discussion (they are often all be the same ID, such as the\n   * MojoHaus project groupId: org.codehaus.mojo). It, along with the groupId,\n   * creates a key that separates this project from every other project in the\n   * world (at least, it should :) ). Along with the groupId, the artifactId\n   * fully defines the artifact's living quarters within the repository. In the\n   * case of the above project, my-project lives in\n   * $M2_REPO/org/codehaus/mojo/my-project.\n   * @default \"my-app\"\n   * @featured\n   */\n  readonly artifactId: string;\n\n  /**\n   * This is the last piece of the naming puzzle. groupId:artifactId denotes a\n   * single project but they cannot delineate which incarnation of that project\n   * we are talking about. Do we want the junit:junit of 2018 (version 4.12), or\n   * of 2007 (version 3.8.2)? In short: code changes, those changes should be\n   * versioned, and this element keeps those versions in line. It is also used\n   * within an artifact's repository to separate versions from each other.\n   * my-project version 1.0 files live in the directory structure\n   * $M2_REPO/org/codehaus/mojo/my-project/1.0.\n   * @default \"0.1.0\"\n   * @featured\n   */\n  readonly version: string;\n\n  /**\n   * Project packaging format.\n   *\n   * @default \"jar\"\n   */\n  readonly packaging?: string;\n\n  /**\n   * Description of a project is always good. Although this should not replace\n   * formal documentation, a quick comment to any readers of the POM is always\n   * helpful.\n   *\n   * @default undefined\n   * @featured\n   */\n  readonly description?: string;\n\n  /**\n   * The URL, like the name, is not required. This is a nice gesture for\n   * projects users, however, so that they know where the project lives.\n   *\n   * @default undefined\n   * @featured\n   */\n  readonly url?: string;\n\n  /**\n   * A Parent Pom can be used to have a child project inherit\n   * properties/plugins/ect in order to reduce duplication and keep standards\n   * across a large amount of repos\n   *\n   * @default undefined\n   * @featured\n   */\n  readonly parentPom?: ParentPom;\n}\n\n/*\n * Represents a Parent Maven Pom. A parent maven pom can provide things like dependencies and plugin lists/configuration.\n * That can be helpful for trying to control configuration across a large swath of programs especially for things like\n * enterprise configurations of plugins.\n * @see https://maven.apache.org/guides/introduction/introduction-to-the-pom.html#project-inheritance\n */\nexport interface ParentPom {\n  /**\n   * Parent Pom Group ID\n   */\n  readonly groupId?: string;\n  /**\n   * Parent Pom Artifact ID\n   */\n  readonly artifactId?: string;\n  /**\n   * Parent Pom Version\n   */\n  readonly version?: string;\n  /**\n   * Parent Pom Relative path from the current pom\n   */\n  readonly relativePath?: string;\n}\n\n/**\n * Represents a Maven repository\n * @see https://maven.apache.org/guides/introduction/introduction-to-repositories.html\n */\nexport interface MavenRepository {\n  /**\n   * The identifier for the repository\n   */\n  readonly id: string;\n  /**\n   * The url of the repository\n   */\n  readonly url: string;\n  /**\n   * The name of the repository\n   */\n  readonly name?: string;\n  /**\n   * The layout of the repository\n   */\n  readonly layout?: string;\n  /**\n   * Repository Policy for Snapshots\n   */\n  readonly snapshots?: MavenRepositoryPolicy;\n  /**\n   * Repository Policy for Releases\n   */\n  readonly releases?: MavenRepositoryPolicy;\n}\n\n/**\n * Represents a Maven Repository Policy\n * @see https://maven.apache.org/settings.html#repositories\n */\nexport interface MavenRepositoryPolicy {\n  /*\n   * Whether this repository is enabled  for the respective type (`releases` or `snapshots`).\n   */\n  readonly enabled?: boolean;\n\n  /**\n   * Update Policy\n   * This element specifies how often updates should attempt to occur. Maven will compare the local POM's timestamp (stored in a repository's maven-metadata file) to the remote.\n   * @default UpdatePolicy.DAILY\n   */\n  readonly updatePolicy?: UpdatePolicy;\n\n  /**\n   * Checksum Policy\n   * When Maven deploys files to the repository, it also deploys corresponding checksum files.\n   */\n  readonly checksumPolicy?: ChecksumPolicy;\n}\n\nexport class UpdatePolicy {\n  static readonly ALWAYS = \"always\";\n  static readonly DAILY = \"daily\";\n  static readonly NEVER = \"never\";\n\n  /**\n   * Updates at an interval of X minutes.\n   */\n  static interval(minutes: number) {\n    return `interval:${minutes}`;\n  }\n}\n\nexport enum ChecksumPolicy {\n  IGNORE = \"ignore\",\n  FAIL = \"fail\",\n  WARN = \"warn\",\n}\n\n/**\n * A Project Object Model or POM is the fundamental unit of work in Maven. It is\n * an XML file that contains information about the project and configuration\n * details used by Maven to build the project.\n */\nexport class Pom extends Component {\n  /**\n   * The name of the pom file.\n   */\n  public readonly fileName: string;\n\n  /**\n   * Maven group ID.\n   */\n  public readonly groupId: string;\n\n  /**\n   * Maven artifact ID.\n   */\n  public readonly artifactId: string;\n\n  /**\n   * Project version.\n   */\n  public readonly version: string;\n\n  /**\n   * Maven packaging format.\n   */\n  public readonly packaging: string;\n\n  /**\n   * Project display name.\n   */\n  public readonly name?: string;\n\n  /**\n   * Project description.\n   */\n  public readonly description?: string;\n\n  /**\n   * Project URL.\n   */\n  public readonly url?: string;\n\n  /*\n   * Project Parent POM\n   */\n  private readonly parentPom?: ParentPom;\n\n  private readonly properties: Record<string, any> = {};\n\n  private readonly repositories: MavenRepository[] = [];\n\n  private readonly pluginRepositories: MavenRepository[] = [];\n\n  constructor(project: Project, options: PomOptions) {\n    super(project);\n\n    this.fileName = \"pom.xml\";\n    this.groupId = options.groupId;\n    this.artifactId = options.artifactId;\n    this.version = options.version;\n    this.packaging = options.packaging ?? \"jar\";\n    this.name = project.name;\n    this.description = options.description;\n    this.url = options.url;\n    this.parentPom = options.parentPom;\n\n    new XmlFile(project, this.fileName, { obj: () => this.synthPom() });\n  }\n\n  /**\n   * Adds a key/value property to the pom.\n   * @param key the key\n   * @param value the value\n   */\n  public addProperty(key: string, value: string) {\n    this.properties[key] = value;\n  }\n\n  /**\n   * Adds a runtime dependency.\n   *\n   * @param spec Format `<groupId>/<artifactId>@<semver>`\n   */\n  public addDependency(spec: string) {\n    this.project.deps.addDependency(spec, DependencyType.RUNTIME);\n  }\n\n  /**\n   * Adds a test dependency.\n   *\n   * @param spec Format `<groupId>/<artifactId>@<semver>`\n   */\n  public addTestDependency(spec: string) {\n    this.project.deps.addDependency(spec, DependencyType.TEST);\n  }\n\n  /**\n   * Adds a build plugin to the pom.\n   *\n   * The plug in is also added as a BUILD dep to the project.\n   *\n   * @param spec dependency spec (`group/artifact@version`)\n   * @param options plugin options\n   */\n  public addPlugin(spec: string, options: PluginOptions = {}) {\n    for (const dep of options.dependencies ?? []) {\n      this.project.deps.addDependency(dep, DependencyType.BUILD);\n    }\n    return this.project.deps.addDependency(spec, DependencyType.BUILD, options);\n  }\n\n  /**\n   * Adds a repository to the pom\n   * @param repository the repository to add\n   */\n  public addRepository(repository: MavenRepository) {\n    this.repositories.push(repository);\n  }\n\n  /*\n   * Adds a repository for plugins to the pom\n   * @param repository the repository to add\n   */\n  public addPluginRepository(repository: MavenRepository) {\n    this.pluginRepositories.push(repository);\n  }\n\n  private synthPom() {\n    return resolve(\n      {\n        project: {\n          ...POM_XML_ATTRS,\n          modelVersion: \"4.0.0\",\n          groupId: this.groupId,\n          artifactId: this.artifactId,\n          version: this.version,\n          packaging: this.packaging,\n          name: this.name,\n          description: this.description,\n          url: this.url,\n          properties: this.properties,\n          parent: this.parentPom,\n          ...this.synthRepositories(),\n          ...this.synthDependencies(),\n          ...this.synthPluginRepositories(),\n        },\n      },\n      { omitEmpty: true }\n    );\n  }\n\n  private synthDependencies() {\n    const deps = this.project.deps.all;\n    if (deps.length === 0) {\n      return;\n    }\n\n    const dependencies: any[] = [];\n    const plugins: any[] = [];\n\n    for (const dep of deps) {\n      switch (dep.type) {\n        case DependencyType.PEER:\n        case DependencyType.RUNTIME:\n          dependencies.push(mavenCoords(dep));\n          break;\n\n        case DependencyType.TEST:\n          dependencies.push({\n            ...mavenCoords(dep),\n            scope: \"test\",\n          });\n          break;\n\n        // build maps to plugins\n        case DependencyType.BUILD:\n          plugins.push({\n            ...mavenCoords(dep),\n            ...pluginConfig(dep.metadata as PluginOptions),\n          });\n          break;\n\n        default:\n          throw new Error(`unsupported dependency type: ${dep.type}`);\n      }\n    }\n\n    return {\n      build: { plugins: { plugin: plugins } },\n      dependencies: { dependency: dependencies },\n    };\n  }\n\n  private synthRepositories() {\n    if (this.repositories.length === 0) {\n      return;\n    }\n\n    return {\n      repositories: { repository: this.repositories },\n    };\n  }\n\n  private synthPluginRepositories() {\n    if (this.pluginRepositories.length === 0) {\n      return;\n    }\n\n    return {\n      pluginRepositories: { pluginRepository: this.pluginRepositories },\n    };\n  }\n}\n\n/**\n * Options for Maven plugins.\n */\nexport interface PluginOptions {\n  /**\n   * Plugin key/value configuration\n   * @default {}\n   */\n  readonly configuration?: { [key: string]: any };\n\n  /**\n   * Plugin executions\n   * @default []\n   */\n  readonly executions?: PluginExecution[];\n\n  /**\n   * You could configure the dependencies for the plugin.\n   *\n   * Dependencies are in `<groupId>/<artifactId>@<semver>` format.\n   *\n   * @default []\n   */\n  readonly dependencies?: string[];\n}\n\n/**\n * Plugin execution definition.\n */\nexport interface PluginExecution {\n  /**\n   * The ID.\n   */\n  readonly id: string;\n\n  /**\n   * Which Maven goals this plugin should be associated with.\n   */\n  readonly goals: string[];\n\n  /**\n   * The phase in which the plugin should execute.\n   */\n  readonly phase?: string;\n\n  /**\n   * Execution key/value configuration\n   * @default {}\n   */\n  readonly configuration?: { [key: string]: any };\n}\n\n/**\n * Parses maven groupId and artifactId from a dependency name.\n *\n *     name    <=> <groupId>/<artifactId>\n *     version <=> <version>\n */\nfunction mavenCoords(dep: DependencyCoordinates) {\n  const name = dep.name;\n  const parts = name.split(\"/\");\n  if (parts.length !== 2) {\n    throw new Error(\n      `invalid maven coordinates in dependency named \"${name}\". format is \"<groupId>/<artifactId>\". For example \"org.junit.jupiter/junit-jupiter-engine\"`\n    );\n  }\n\n  return {\n    groupId: parts[0],\n    artifactId: parts[1],\n    version: dep.version ? toMavenVersionRange(dep.version) : undefined,\n  };\n}\n\nfunction pluginConfig(options: PluginOptions = {}) {\n  return {\n    configuration: options.configuration,\n    dependencies:\n      options.dependencies && options.dependencies.length > 0\n        ? {\n            dependency: options.dependencies?.map((d) =>\n              mavenCoords(Dependencies.parseDependency(d))\n            ),\n          }\n        : undefined,\n    executions: {\n      execution: options.executions?.map((e) => ({\n        id: e.id,\n        goals: e.goals.map((goal) => ({ goal })),\n        phase: e.phase,\n        configuration: e.configuration,\n      })),\n    },\n  };\n}\n"]}