From f9b0f966bb5458f1cdf39f027204381d75de1768 Mon Sep 17 00:00:00 2001 From: Merlin Beutlberger Date: Fri, 13 Dec 2024 12:37:29 +0100 Subject: [PATCH] feat(linter): Extend project initialization fallbacks Enhance compatibility for legacy UI5 projects that do not include a ui5.yaml and do not use current best practice directory structures like 'webapp' for apps and 'src'/'test' for libraries. In addition to those directories we now also check and eventually use the following directories: Applications: * src/main/webapp * WebContent Libraries: * src/main/jslib (+ src/test/jslib) * src/main/uilib (+ src/test/uilib) * src/main/js (+ src/test/js) --- src/linter/linter.ts | 98 ++++++++--- .../legacy.app.a/WebContent/manifest.json | 7 + .../src/main/webapp/manifest.json | 7 + .../legacy.lib.a/src/main/jslib/.library | 7 + .../legacy.lib.b/src/main/uilib/.library | 7 + .../legacy.lib.c/src/main/js/.library | 7 + test/lib/linter/linter.ts | 154 +++++++++++++++++- 7 files changed, 266 insertions(+), 21 deletions(-) create mode 100644 test/fixtures/linter/projects/legacy-dirs/legacy.app.a/WebContent/manifest.json create mode 100644 test/fixtures/linter/projects/legacy-dirs/legacy.app.b/src/main/webapp/manifest.json create mode 100644 test/fixtures/linter/projects/legacy-dirs/legacy.lib.a/src/main/jslib/.library create mode 100644 test/fixtures/linter/projects/legacy-dirs/legacy.lib.b/src/main/uilib/.library create mode 100644 test/fixtures/linter/projects/legacy-dirs/legacy.lib.c/src/main/js/.library diff --git a/src/linter/linter.ts b/src/linter/linter.ts index 0d3dd8b0f..973afa6ab 100644 --- a/src/linter/linter.ts +++ b/src/linter/linter.ts @@ -188,26 +188,38 @@ async function getProjectGraph(rootDir: string, ui5Config?: string | object): Pr rootConfigPath = ui5YamlPath; } else { if (ui5Config) throw new Error(`Unable to find UI5 config file '${ui5Config}'`); - const isApp = await dirExists(path.join(rootDir, "webapp")); - if (isApp) { - rootConfiguration = { - specVersion: "3.0", - type: "application", - metadata: { - name: "ui5-linter-target", - }, - }; - } else { - const isLibrary = await dirExists(path.join(rootDir, "src")); - if (isLibrary) { - rootConfiguration = { - specVersion: "3.0", - type: "library", - metadata: { - name: "ui5-linter-target", - }, - }; - } + + const dirChecks = await Promise.all([ + dirExists(path.join(rootDir, "webapp")), + dirExists(path.join(rootDir, "src", "main", "webapp")), + dirExists(path.join(rootDir, "WebContent")), + dirExists(path.join(rootDir, "src", "main", "jslib")), + dirExists(path.join(rootDir, "src", "main", "js")), + dirExists(path.join(rootDir, "src", "main", "uilib")), + dirExists(path.join(rootDir, "src")), + ]); + + if (dirChecks[0]) { + // Common app with webapp folder + rootConfiguration = createProjectConfig("application", "webapp"); + } else if (dirChecks[1]) { + // Legacy app with src/main/webapp folder + rootConfiguration = createProjectConfig("application", "src/main/webapp"); + } else if (dirChecks[2]) { + // Legacy app with WebContent folder + rootConfiguration = createProjectConfig("application", "WebContent"); + } else if (dirChecks[3]) { + // Library with src/main/jslib folder + rootConfiguration = createProjectConfig("library", "src/main/jslib", "src/test/jslib"); + } else if (dirChecks[4]) { + // Library with src/main/js folder + rootConfiguration = createProjectConfig("library", "src/main/js", "src/test/js"); + } else if (dirChecks[5]) { + // Library with src/main/uilib folder + rootConfiguration = createProjectConfig("library", "src/main/uilib", "src/test/uilib"); + } else if (dirChecks[6]) { + // Library with src folder + rootConfiguration = createProjectConfig("library", "src", "test"); } } @@ -231,6 +243,47 @@ async function getProjectGraph(rootDir: string, ui5Config?: string | object): Pr }); } +interface ProjectConfig { + specVersion: string; + type: string; + metadata: { + name: string; + }; + resources?: { + configuration: { + paths: { + webapp?: string; + src?: string; + test?: string; + }; + }; + }; +} + +function createProjectConfig(projectType: string, projectSrcPath?: string, projectTestPath?: string): ProjectConfig { + let resourcesConfig: ProjectConfig["resources"] = { + configuration: { + paths: {}, + }, + }; + if (projectType === "application") { + resourcesConfig.configuration.paths.webapp = projectSrcPath ?? "webapp"; + } else if (projectType === "library") { + resourcesConfig.configuration.paths.src = projectSrcPath ?? "src"; + resourcesConfig.configuration.paths.test = projectTestPath ?? "test"; + } else { + // Do not set a resources configuration for other project types + resourcesConfig = undefined; + } + return { + specVersion: "4.0", + type: projectType, + metadata: { + name: "ui5-linter-project", + }, + resources: resourcesConfig, + }; +} /** * Normalize provided virtual paths to the original file paths */ @@ -389,3 +442,8 @@ export function mergeIgnorePatterns(options: LinterOptions, config: UI5LintConfi ...(options.ignorePatterns ?? []), // CLI patterns go after config patterns ].filter(($) => $); } + +// Export local function for testing only +export const __localFunctions__ = (process.env.NODE_ENV === "test") ? + {getProjectGraph} : + /* istanbul ignore next */ undefined; diff --git a/test/fixtures/linter/projects/legacy-dirs/legacy.app.a/WebContent/manifest.json b/test/fixtures/linter/projects/legacy-dirs/legacy.app.a/WebContent/manifest.json new file mode 100644 index 000000000..3ff5a6225 --- /dev/null +++ b/test/fixtures/linter/projects/legacy-dirs/legacy.app.a/WebContent/manifest.json @@ -0,0 +1,7 @@ +{ + "_version": "1.12.0", + "sap.app": { + "id": "com.app", + "type": "application" + } +} diff --git a/test/fixtures/linter/projects/legacy-dirs/legacy.app.b/src/main/webapp/manifest.json b/test/fixtures/linter/projects/legacy-dirs/legacy.app.b/src/main/webapp/manifest.json new file mode 100644 index 000000000..3ff5a6225 --- /dev/null +++ b/test/fixtures/linter/projects/legacy-dirs/legacy.app.b/src/main/webapp/manifest.json @@ -0,0 +1,7 @@ +{ + "_version": "1.12.0", + "sap.app": { + "id": "com.app", + "type": "application" + } +} diff --git a/test/fixtures/linter/projects/legacy-dirs/legacy.lib.a/src/main/jslib/.library b/test/fixtures/linter/projects/legacy-dirs/legacy.lib.a/src/main/jslib/.library new file mode 100644 index 000000000..61286d35e --- /dev/null +++ b/test/fixtures/linter/projects/legacy-dirs/legacy.lib.a/src/main/jslib/.library @@ -0,0 +1,7 @@ + + + library.with.custom.paths + SAP SE + ${version} + ${copyright} + diff --git a/test/fixtures/linter/projects/legacy-dirs/legacy.lib.b/src/main/uilib/.library b/test/fixtures/linter/projects/legacy-dirs/legacy.lib.b/src/main/uilib/.library new file mode 100644 index 000000000..61286d35e --- /dev/null +++ b/test/fixtures/linter/projects/legacy-dirs/legacy.lib.b/src/main/uilib/.library @@ -0,0 +1,7 @@ + + + library.with.custom.paths + SAP SE + ${version} + ${copyright} + diff --git a/test/fixtures/linter/projects/legacy-dirs/legacy.lib.c/src/main/js/.library b/test/fixtures/linter/projects/legacy-dirs/legacy.lib.c/src/main/js/.library new file mode 100644 index 000000000..61286d35e --- /dev/null +++ b/test/fixtures/linter/projects/legacy-dirs/legacy.lib.c/src/main/js/.library @@ -0,0 +1,7 @@ + + + library.with.custom.paths + SAP SE + ${version} + ${copyright} + diff --git a/test/lib/linter/linter.ts b/test/lib/linter/linter.ts index dafbd7544..1fd717b85 100644 --- a/test/lib/linter/linter.ts +++ b/test/lib/linter/linter.ts @@ -2,6 +2,7 @@ import anyTest, {TestFn} from "ava"; import sinonGlobal, {SinonStub} from "sinon"; import path from "node:path"; import {fileURLToPath} from "node:url"; +import esmock from "esmock"; import { createTestsForFixtures, assertExpectedLintResults, esmockDeprecationText, preprocessLintResultsForSnapshot, @@ -28,7 +29,7 @@ test.after.always((t) => { t.context.sinon.restore(); }); -// Define tests for reach file in the fixtures/linter/general directory +// Define tests for each file in the fixtures/linter/general directory createTestsForFixtures(fixturesGeneralPath); // Test project fixtures individually @@ -272,3 +273,154 @@ test.serial("lint: com.ui5.troublesome.app with custom UI5 config which does NOT ui5Config, }), {message: `Unable to find UI5 config file '${ui5Config}'`}); }); + +test.serial("lint: getProjectGraph with different directory structures", async (t) => { + const graphFromObjectStub = t.context.sinon.stub().resolves(); + const {__localFunctions__} = await esmock("../../../src/linter/linter.js", { + "@ui5/project/graph": { + graphFromObject: graphFromObjectStub, + }, + }); + + const {getProjectGraph} = __localFunctions__; + + const basePath = path.join(fixturesProjectsPath, "legacy-dirs"); + + // Legacy app structures + const appA = path.join(basePath, "legacy.app.a"); + const appB = path.join(basePath, "legacy.app.b"); + const libA = path.join(basePath, "legacy.lib.a"); + const libB = path.join(basePath, "legacy.lib.b"); + const libC = path.join(basePath, "legacy.lib.c"); + + await getProjectGraph(appA); + await getProjectGraph(appB); + await getProjectGraph(libA); + await getProjectGraph(libB); + await getProjectGraph(libC); + + t.is(graphFromObjectStub.callCount, 5); + t.deepEqual(graphFromObjectStub.getCall(0).args[0], { + dependencyTree: { + dependencies: [], + id: "ui5-linter-target", + path: appA, + version: "1.0.0", + }, + resolveFrameworkDependencies: false, + rootConfigPath: undefined, + rootConfiguration: { + metadata: { + name: "ui5-linter-project", + }, + resources: { + configuration: { + paths: { + webapp: "WebContent", + }, + }, + }, + specVersion: "4.0", + type: "application", + }, + }); + t.deepEqual(graphFromObjectStub.getCall(1).args[0], { + dependencyTree: { + dependencies: [], + id: "ui5-linter-target", + path: appB, + version: "1.0.0", + }, + resolveFrameworkDependencies: false, + rootConfigPath: undefined, + rootConfiguration: { + metadata: { + name: "ui5-linter-project", + }, + resources: { + configuration: { + paths: { + webapp: "src/main/webapp", + }, + }, + }, + specVersion: "4.0", + type: "application", + }, + }); + t.deepEqual(graphFromObjectStub.getCall(2).args[0], { + dependencyTree: { + dependencies: [], + id: "ui5-linter-target", + path: libA, + version: "1.0.0", + }, + resolveFrameworkDependencies: false, + rootConfigPath: undefined, + rootConfiguration: { + metadata: { + name: "ui5-linter-project", + }, + resources: { + configuration: { + paths: { + src: "src/main/jslib", + test: "src/test/jslib", + }, + }, + }, + specVersion: "4.0", + type: "library", + }, + }); + t.deepEqual(graphFromObjectStub.getCall(3).args[0], { + dependencyTree: { + dependencies: [], + id: "ui5-linter-target", + path: libB, + version: "1.0.0", + }, + resolveFrameworkDependencies: false, + rootConfigPath: undefined, + rootConfiguration: { + metadata: { + name: "ui5-linter-project", + }, + resources: { + configuration: { + paths: { + src: "src/main/uilib", + test: "src/test/uilib", + }, + }, + }, + specVersion: "4.0", + type: "library", + }, + }); + t.deepEqual(graphFromObjectStub.getCall(4).args[0], { + dependencyTree: { + dependencies: [], + id: "ui5-linter-target", + path: libC, + version: "1.0.0", + }, + resolveFrameworkDependencies: false, + rootConfigPath: undefined, + rootConfiguration: { + metadata: { + name: "ui5-linter-project", + }, + resources: { + configuration: { + paths: { + src: "src/main/js", + test: "src/test/js", + }, + }, + }, + specVersion: "4.0", + type: "library", + }, + }); +});