Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Component best practices- async flags check #73

Merged
merged 90 commits into from
May 29, 2024
Merged
Show file tree
Hide file tree
Changes from 77 commits
Commits
Show all changes
90 commits
Select commit Hold shift + click to select a range
e2bab81
feat: Analyze async flags
d3xter666 Apr 15, 2024
46b4622
feat: Implement manifest checks
d3xter666 Apr 15, 2024
6e32f58
feat: Extend linting further
d3xter666 Apr 16, 2024
cda4880
fix: JSON "parse" improvement
d3xter666 Apr 16, 2024
c980c07
feat: Handle variations for error messaging
d3xter666 Apr 16, 2024
7d5711d
feat: Checks for inline manifest
d3xter666 Apr 18, 2024
4e8d607
fix: Tests
d3xter666 Apr 18, 2024
3280f54
fix: Merge conflicts
d3xter666 Apr 18, 2024
a3899a0
fix: Reset Component.js
d3xter666 Apr 18, 2024
2a8c08e
fix: Tests
d3xter666 Apr 18, 2024
a02fbad
refactor: Restructure rules tests to use a common execution
d3xter666 Apr 19, 2024
baaeb79
refactor: Linter helper
d3xter666 Apr 19, 2024
9358dd4
feat: Add tests for Best practices
d3xter666 Apr 19, 2024
7524834
fix: Tests
d3xter666 Apr 19, 2024
daeb315
refactor: Automatically resolve BestPractices subdirs
d3xter666 Apr 19, 2024
badf560
feat: Add more tests for Best Practices
d3xter666 Apr 19, 2024
1f998ba
fix: Detection of unset values
d3xter666 Apr 19, 2024
acd3472
refactor: Improve messaging
d3xter666 Apr 19, 2024
89b1261
fix: Eslint
d3xter666 Apr 19, 2024
1611ad9
fix: Update snapshots
d3xter666 Apr 19, 2024
5dc3f12
refactor: Interfaces are part of metadata
d3xter666 Apr 22, 2024
894029c
refactor: Look into hierarchy
d3xter666 Apr 24, 2024
b88f623
fix: Extract and check against real module dependency var
d3xter666 Apr 24, 2024
aedea65
refactor: Allow sourcefile tests for a whole dir
d3xter666 Apr 25, 2024
079444b
test: Add BestPractices samples
d3xter666 Apr 25, 2024
8bb2b38
fix: Allow filtering for BestPractices tests
d3xter666 Apr 25, 2024
c78eaca
fix: Analyze only Component.js files
d3xter666 Apr 25, 2024
b95900c
test: Add negative test case
d3xter666 Apr 25, 2024
ddf092d
fix: Test Component's namespace
d3xter666 Apr 25, 2024
e94be3f
fix: Tests
d3xter666 Apr 25, 2024
455267a
fix: Eslint
d3xter666 Apr 25, 2024
6199213
test: Add positive case
d3xter666 Apr 25, 2024
aa6de7c
fix: Update snapshots
d3xter666 Apr 25, 2024
d64b365
fix: Windows paths for tests
d3xter666 Apr 25, 2024
c2f2ece
fix: Paths on windows
d3xter666 Apr 25, 2024
292334a
refactor: Cleanup obsolete code
d3xter666 Apr 25, 2024
5a9aa70
refactor: Cleanup code
d3xter666 Apr 26, 2024
1b1a277
refactor: Always try to process the manifest
d3xter666 Apr 26, 2024
c410868
refactor: Detect sap/ui/core/UIComponent dependency
d3xter666 Apr 26, 2024
95d29c4
refactor: Lint Message
d3xter666 Apr 26, 2024
52dd0d5
fix: Update messages & tests
d3xter666 Apr 26, 2024
5338f9c
feat: Detect manifest.json section in Component.js
d3xter666 Apr 26, 2024
05442d5
fix: Detect missing manifest declaration in Component.js
d3xter666 Apr 26, 2024
1091024
fix: Linting test filenames
d3xter666 Apr 26, 2024
546cb41
fix: Eslint
d3xter666 Apr 26, 2024
29f8cb7
fix: Paths for windows
d3xter666 Apr 26, 2024
1c6ee18
feat: Handle ES6 interface implementation
d3xter666 May 7, 2024
4d9e33a
test: Checks for inheritance in ES6 components
d3xter666 May 7, 2024
e2238e9
refactor: Renames & docs
d3xter666 May 8, 2024
013122f
fix: Eslint issues
d3xter666 May 8, 2024
af1d2f9
fix: Handle quoted properties
d3xter666 May 15, 2024
29fa1db
test: Add case for missing view & routing
d3xter666 May 15, 2024
a7a307e
refactor: Migrate checks to enum
d3xter666 May 15, 2024
79cdbe7
fix: Reset result template
d3xter666 May 15, 2024
89f11f8
refactor: Remove redundant comments
d3xter666 May 15, 2024
bd2f0da
refactor: Rename BestPractices
d3xter666 May 15, 2024
9679220
refactor: Exit early
d3xter666 May 15, 2024
0cd5a7f
refactor: Remove redundant checks
d3xter666 May 15, 2024
a8e62d2
refactor: Resolve SourceFile
d3xter666 May 15, 2024
da987fd
refactor: Cleanup redundant code
d3xter666 May 15, 2024
cacf900
refactor: Improve messaging
d3xter666 May 15, 2024
4d8a01c
fix: Eslint
d3xter666 May 15, 2024
c30a8c9
fix: Rename bestPractices.ts
d3xter666 May 15, 2024
adae2ac
refactor: Extract virBasePath from filePaths
d3xter666 May 16, 2024
b3c1bb5
feat: Provide links to manifest.json when needed
d3xter666 May 17, 2024
168fd14
fix: Link resolver for LinterContext
d3xter666 May 17, 2024
697869c
test: Update snapshots with the new changes
d3xter666 May 17, 2024
f840097
fix: Eslint
d3xter666 May 17, 2024
c9319d9
fix: LintContext fix links resolve
d3xter666 May 17, 2024
49d4e33
refactor: Improve messaging
d3xter666 May 17, 2024
34f6698
refactor: Improve runtime performance by checking Component.js
d3xter666 May 17, 2024
1bb546d
refactor: Cleanup
d3xter666 May 17, 2024
e670ecc
test: Add case for obsolete async flag
d3xter666 May 17, 2024
ca2115e
test: Update snapshots
d3xter666 May 17, 2024
417a94b
fix: Correct logic for reporting
d3xter666 May 17, 2024
604ef22
test: Add test cases for single async flags
d3xter666 May 17, 2024
6d63c56
fix: Report missing manifest definition only when there is a manifest…
d3xter666 May 20, 2024
77e5369
deps: Rebuild api-json deps from UI5 1.120.13
d3xter666 May 27, 2024
0b01e1b
refactor(tests): Rename BestPractices to AsyncComponentFlags, use dir…
RandomByte May 23, 2024
84dfaf2
refactor: Rename bestPractices module to asyncComponentFlags, also ch…
RandomByte May 23, 2024
d2eabaa
test(AsyncComponentFlags): Add more test cases
RandomByte May 23, 2024
e7a8955
refactor(asyncComponentFlags): Remove usage of internal API, improve …
RandomByte May 24, 2024
608dd67
refactor(asyncComponentFlags): Rename types, use boolean for async in…
RandomByte May 24, 2024
a718397
refactor(asyncComponentFlags): Simplify async flag merge logic
RandomByte May 24, 2024
992dee4
test(AsyncComponentFlags): Move parent components into subdirs
RandomByte May 27, 2024
f4be3cb
fix: Skip depcheck for fake modules in testing
d3xter666 May 28, 2024
85fda50
refactor: Improve file path handling
RandomByte May 28, 2024
7790b4a
refactor: Update message texts
RandomByte May 28, 2024
f136e64
refactor: Fix message text
RandomByte May 28, 2024
8e42467
refactor: Revert "refactor: Extract virBasePath from filePaths"
RandomByte May 28, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions src/linter/LinterContext.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import {AbstractAdapter, AbstractReader} from "@ui5/fs";
import {createReader} from "@ui5/fs/resourceFactory";
import {resolveLinks} from "../formatter/lib/resolveLinks.js";

export type FilePath = string; // Platform-dependent path
export type ResourcePath = string; // Always POSIX
Expand Down Expand Up @@ -156,6 +157,10 @@ export default class LinterContext {
}

addLintingMessage(resourcePath: ResourcePath, message: LintMessage) {
if (message.messageDetails) {
message.messageDetails = resolveLinks(message.messageDetails);
}

this.getLintingMessages(resourcePath).push(message);
}

Expand Down
11 changes: 7 additions & 4 deletions src/linter/linter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -87,16 +87,19 @@ export async function lintProject({
export async function lintFile({
rootDir, pathsToLint, namespace, reportCoverage, includeMessageDetails,
}: LinterOptions): Promise<LintResult[]> {
const reader = createReader({
fsBasePath: rootDir,
virBasePath: "/",
});
let resolvedFilePaths;
let virBasePath = "";
if (pathsToLint?.length) {
const absoluteFilePaths = resolveFilePaths(rootDir, pathsToLint);
resolvedFilePaths = transformFilePathsToVirtualPaths(
absoluteFilePaths, rootDir, "/", rootDir);
// Extract the (virtual) path from the filename
virBasePath = resolvedFilePaths[0].split("/").slice(0, -1).join("/");
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What was the reason for adae2ac ?

I'm wondering whether this approach still works if the first path in the array is deeply nested. For example /resources/my/app/controller/MyController.js would lead to virtual base path /resources/my/app/controller/. Which does not match for other paths like /resources/my/app/view/MyView.view.xml

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The reason for that change and the related ones is that for the test cases it used to read files one by one and didn't have the context for the parent components, for example.

The problem lies in the fact that without a /resource prefix, the files were not found. To be honest, I'm not sure for the core reason behnid this behaviour, but I guess it might be something with the Reader

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmmm, you're right about the deeply nested components 🤔

But, how to extract the namespace without breaking the signature of the function?
Maybe, reuse the logic from lintProject(), but then we need a project, not a single file...

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I reverted the commit and all tests are still green 😅

Copy link
Contributor Author

@d3xter666 d3xter666 May 28, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unfortunately, it’s not just that change 😅
I have tried several things to provide a better handling of those resources.

Maybe, directly comparing/reseting the file with the main branch would be easier 😅

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hm, not sure I understand the implications...

I just pushed the revert which restores the behavior I think is better (and should work for deep paths). Let's discuss it based on that 👍

Copy link
Contributor Author

@d3xter666 d3xter666 May 28, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please take a look here: src/linter/linter.ts

This is the original change compared to the main branch.
The change is working pretty well, except that it cannot handle /test-resources/ and that's why I needed to find a better way of doing it.
Please take a look at the conversation we had with @flovogt on that matter here: https://github.com/SAP/ui5-linter/pull/73/files#r1601613818

But, if we take a look at the implementation of the lintProject() method, it again relies only on the /resources path, ignoring the /test-resources.

If this is ok, then we can leave it as it is now :)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Got it, thanks. Since the lintFile API is currently only used in our tests, I think we can safely go with the current state and ignore the test-resources case for now 👍

}
const reader = createReader({
fsBasePath: rootDir,
virBasePath: `${virBasePath}/`,
});

const res = await lint(reader, {
rootDir,
Expand Down
16 changes: 15 additions & 1 deletion src/linter/ui5Types/SourceFileLinter.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import ts, {Identifier} from "typescript";
import SourceFileReporter from "./SourceFileReporter.js";
import LinterContext, {ResourcePath, CoverageCategory, LintMessageSeverity} from "../LinterContext.js";
import analyzeComponentJson from "./bestPractices.js";

interface DeprecationInfo {
symbol: ts.Symbol;
Expand All @@ -16,11 +17,12 @@ export default class SourceFileLinter {
#boundVisitNode: (node: ts.Node) => void;
#reportCoverage: boolean;
#messageDetails: boolean;
#manifestContent: string | undefined;

constructor(
context: LinterContext, resourcePath: ResourcePath, sourceFile: ts.SourceFile, sourceMap: string | undefined,
checker: ts.TypeChecker, reportCoverage: boolean | undefined = false,
messageDetails: boolean | undefined = false
messageDetails: boolean | undefined = false, manifestContent?: string | undefined
) {
this.#resourcePath = resourcePath;
this.#sourceFile = sourceFile;
Expand All @@ -30,6 +32,7 @@ export default class SourceFileLinter {
this.#boundVisitNode = this.visitNode.bind(this);
this.#reportCoverage = reportCoverage;
this.#messageDetails = messageDetails;
this.#manifestContent = manifestContent;
}

// eslint-disable-next-line @typescript-eslint/require-await
Expand Down Expand Up @@ -67,6 +70,17 @@ export default class SourceFileLinter {
node as (ts.PropertyAccessExpression | ts.ElementAccessExpression)); // Check for deprecation
} else if (node.kind === ts.SyntaxKind.ImportDeclaration) {
this.analyzeImportDeclaration(node as ts.ImportDeclaration); // Check for deprecation
} else if (node.kind === ts.SyntaxKind.ExpressionWithTypeArguments &&
ts.isSourceFile(this.#sourceFile) &&
RandomByte marked this conversation as resolved.
Show resolved Hide resolved
this.#sourceFile.fileName.endsWith("/Component.js")) {
RandomByte marked this conversation as resolved.
Show resolved Hide resolved
RandomByte marked this conversation as resolved.
Show resolved Hide resolved
analyzeComponentJson({
node: node as ts.ExpressionWithTypeArguments,
manifestContent: this.#manifestContent,
resourcePath: this.#resourcePath,
reporter: this.#reporter,
context: this.#context,
checker: this.#checker,
});
}

// Traverse the whole AST from top to bottom
Expand Down
12 changes: 10 additions & 2 deletions src/linter/ui5Types/TypeLinter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import SourceFileLinter from "./SourceFileLinter.js";
import {taskStart} from "../../util/perf.js";
import {getLogger} from "@ui5/logger";
import LinterContext, {LinterParameters} from "../LinterContext.js";
// import {Project} from "@ui5/project";
import path from "node:path/posix";
import {AbstractAdapter} from "@ui5/fs";
import {createAdapter, createResource} from "@ui5/fs/resourceFactory";

Expand Down Expand Up @@ -103,12 +103,20 @@ export default class TypeChecker {
if (!sourceMap) {
log.verbose(`Failed to get source map for ${sourceFile.fileName}`);
}
let manifestContent;
if (sourceFile.fileName.endsWith("/Component.js")) {
d3xter666 marked this conversation as resolved.
Show resolved Hide resolved
const res = await this.#workspace.byPath(path.dirname(sourceFile.fileName) + "/manifest.json");
if (res) {
manifestContent = await res.getString();
}
}
const linterDone = taskStart("Type-check resource", sourceFile.fileName, true);
const linter = new SourceFileLinter(
this.#context,
sourceFile.fileName, sourceFile,
sourceMap,
checker, reportCoverage, messageDetails
checker, reportCoverage, messageDetails,
manifestContent
);
await linter.lint();
linterDone();
Expand Down
Loading