UNPKG

@atomist/sdm

Version:

Atomist Software Delivery Machine SDK

101 lines (95 loc) 3.74 kB
/* * Copyright © 2020 Atomist, Inc. * * Licensed 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 CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { Java9FileParser, KotlinFileParser } from "@atomist/antlr"; import { ProjectAsync } from "@atomist/automation-client/lib/project/Project"; import { gather, matches } from "@atomist/automation-client/lib/tree/ast/astUtils"; import { logger } from "@atomist/automation-client/lib/util/logger"; import * as _ from "lodash"; import { JavaSourceFiles, KotlinSourceFiles } from "./javaProjectUtils"; import { JavaPackage } from "./query/javaPathExpressions"; /** * Path expression using the Kotlin grammar for a Java package declaration */ export const KotlinPackage = "//packageHeader//identifier"; /** * Represents Java project structure (nested packages following Java naming conventions) * which can be inferred from project contents. * Also works for Kotlin. */ export class JavaProjectStructure { /** * Find root Java or Kotlin package * @param {ProjectAsync} p * @return {Promise<JavaProjectStructure>} */ public static async infer(p: ProjectAsync): Promise<JavaProjectStructure> { // Treat Java and Kotlin as one const packages = ( await gather(p, { parseWith: Java9FileParser, globPatterns: JavaSourceFiles, pathExpression: JavaPackage, // TODO workaround for recursive ANTLR bug mapper: m => m.$value.replace(/.*package (.*);/, "$1"), }) ).concat( ( await matches(p, { parseWith: KotlinFileParser, globPatterns: KotlinSourceFiles, pathExpression: KotlinPackage, }) ).map(m => m.$value), ); const uniquePackages = _.uniq(packages); if (uniquePackages.length === 0) { return undefined; } if (uniquePackages.length === 1) { const jps = new JavaProjectStructure(uniquePackages[0]); logger.debug("Successful JavaProjectStructure inference on %j: Sole package is %j", p.id, jps); return jps; } const longestPrefix = sharedStart(uniquePackages); if (!!longestPrefix) { const jps = new JavaProjectStructure(longestPrefix.replace(/\.$/, "")); logger.debug("Successful JavaProjectStructure inference on %j: Shortest path is %j", p.id, jps); return jps; } else { logger.debug("Unsuccessful JavaProjectStructure inference on %j", p.id); return undefined; } } /** * @param applicationPackage The first Java package found in the project. */ private constructor(public readonly applicationPackage: string) {} } // Taken from https://stackoverflow.com/questions/1916218/find-the-longest-common-starting-substring-in-a-set-of-strings function sharedStart(array: string[]): string { const A = array.concat().sort(); if (!A) { return ""; } const a1 = A[0]; const a2 = A[A.length - 1]; const L = a1.length; let i = 0; while (i < L && a1.charAt(i) === a2.charAt(i)) { i++; } return a1.substring(0, i); }