angular-cli icon indicating copy to clipboard operation
angular-cli copied to clipboard

ng add fails to find app.module.ts when using custom tsconfig.json paths

Open seangwright opened this issue 6 years ago • 9 comments

Bug Report or Feature Request (mark with an x)

- [X ] bug report -> please search issues before submitting
- [X ] feature request

Whether this is a bug or a feature depends on whether or not custom paths in tsconfig.json are supported with the Cli.

Command (mark with an x)

- [X ] add

Versions

Angular CLI: 7.0.2
Node: 10.8.0
OS: win32 x64
Angular: 7.0.0
... animations, common, compiler, compiler-cli, core, forms ... http, language-service, platform-browser
... platform-browser-dynamic, router

Repro steps

  • ng new app-name (pick defaults)
  • Add "paths": { "@app/*": ["src/app/*"] } to tsconfig.json
  • Update main.ts to use this new path import { AppModule } from '@app/app.module'; import { environment } from './environments/environment';
  • ng add @angular/material
  • Select Yes for ? Set up browser animations for Angular Material?

Repository all set up to reproduce this error is available here. Just clone and type ng add @angular/material.

The log given by the failure

Could not read Angular module file: /src/@app/app.module.ts

Desired functionality

Cli handles custom paths in tsconfig.json.

Mention any other details that might be useful

Here is where material tries to get the app.module.ts path https://github.com/angular/material2/blob/985774a4eaa14d1dcbf1ad96ab176043d38f433e/src/lib/schematics/ng-add/setup-project.ts#L56

Here is where the path to app.module.ts is being calculated in the schematics code https://github.com/angular/angular-cli/blob/82f2bda2f59e4665611d1a75b51a62bae21ac340/packages/schematics/angular/utility/ng-ast-utils.ts#L78

I believe the above link is what is causing the problem.

I've always preferred absolute urls in Angular apps compared to the relative paths (which become very difficult to read for nested components).

Adding custom paths to tsconfig.json works pretty well to solve this problem. VSCode works fine with it and the Cli has no other issues I've run into.

If using paths isn't officially supported does the Angular team recommend using relative paths like import { MyApiService } from '../../../../../my-api.service'?

seangwright avatar Oct 24 '18 20:10 seangwright

Relates to https://github.com/angular/angular-cli/issues/12729

alan-agius4 avatar Oct 25 '18 05:10 alan-agius4

Did anyone find a workaround for this issue? I'm trying to add angular material to an existing project and run into this issue. Can someone help me to get it working?

serridavide avatar Nov 30 '20 08:11 serridavide

I just wanted to bump this and say that with Nx and people learning TS more, these custom path mappings are very common. I would suggest moving the frequency of this issue up from low to medium.

This is something that Angular Material users have been running into for 3 years now when using our ng add schematic.

Also based on my investigation in https://github.com/angular/angular-cli/issues/20118, this is still occuring with 11.2.1.

Splaktar avatar Feb 22 '21 21:02 Splaktar

I had a look at this and continuing on what @Splaktar mentioned in #20118, in https://github.com/angular/components/blob/8f558eeb7e475da54ad69e85ec07229fb4b0c3e9/src/cdk/schematics/utils/ast/ng-module-imports.ts#L16 the tree.read is invoked with an inexistent path. Schematics are not aware of specific TypeScript path mappings logic as it works similar to the Node.JS file system and hence it is correctly failing.

The module needs to be provided prior to passing it to tree.read. Since, the module resolution logic, is within the Angular Components schematic, it does appear to me that this is more of an Angular Material bug, where path mappings are not supported by the ng-add schematic.

I did a high level example on what needs to be done in the schematic:

const workspace = await getWorkspace(host);
const project = getProjectFromWorkspace(workspace, options.project);
let appModulePath = getAppModulePath(host, getProjectMainFile(project));

if (!host.exits(appModulePath)) {
  const tsConfig = project.targets.get('build')?.options?.tsConfig;
  if (typeof tsConfig === 'string') {
    const { config } = ts.readConfigFile(tsConfig, host.read);
    const { resolvedModule } = ts.resolveModuleName(appModulePath, getProjectMainFile(project), config.options, {
      readFile: function (fileName: string): string {
        return host.read(fileName).toString().replace(/^\uFEFF/, '');
      },
      directoryExists: function (directoryName: string): boolean {
        // When the path is file getDir will throw.
        try {
          const dir = host.getDir(directoryName);

          return !!(dir.subdirs.length || dir.subfiles.length);
        } catch {
          return false;
        }
      },
      fileExists: function (fileName: string): boolean {
        return host.exists(fileName);
      },
      realpath: function (path: string): string {
        return path;
      },
      getCurrentDirectory: function () {
        return host.root.path;
      },
    });

    appModulePath = resolvedModule.resolvedFileName;
  }
}

Alternatively, for a simpler approach an addition option can be added that would allow the users to provide the module path.

ng add @angular/material --module /src/app/app.module.ts
const workspace = await getWorkspace(host);
const project = getProjectFromWorkspace(workspace, options.project);
let appModulePath = options.module ?? getAppModulePath(host, getProjectMainFile(project));

alan-agius4 avatar Feb 23 '21 10:02 alan-agius4

Did anyone find a workaround for this issue? I'm trying to add angular material to an existing project and run into this issue. Can someone help me to get it working?

Found a temporary solution: In your main.ts change the import path of AppModule.

For me it looks like this: import { AppModule } from '@app/app.module'; -> import { AppModule } from './app/app.module';

Absolute path didn't work:

import { AppModule } from 'src/app/app.module';
Could not read Angular module file: /src/src/app/app.module.ts

ThisIsIvan avatar Jun 21 '23 10:06 ThisIsIvan

This still happens on Angular 17

Setup to fail:

  • In your tsconfig.json add a custom path:
...
    "paths": {
      "@app/*": ["app/*"],
...
  • Then use it in your main.ts:
import { bootstrapApplication } from '@angular/platform-browser';

import { AppComponent } from '@app/app.component';
import { appConfig } from '@app/app.config';

bootstrapApplication(AppComponent, appConfig).catch((err) => console.error(err));
  • For example, try to add ssr package: ng add @angular/ssr

The command fails:

Cannot statically analyze bootstrapApplication call in src/main.ts

  • If you change main.ts to:
import { bootstrapApplication } from '@angular/platform-browser';

import { AppComponent } from './app/app.component';
import { appConfig } from './app/app.config';

bootstrapApplication(AppComponent, appConfig).catch((err) => console.error(err));

It works @alan-agius4

Sergiobop avatar Nov 19 '23 11:11 Sergiobop

By default Angular 17 create a project that use standalone. You can change that putting false when you create your project. ng new nameproject --standalone=false

AngelCorre avatar Dec 29 '23 03:12 AngelCorre

I can confirm this. I'm having the same issue right now:

Cannot statically analyze bootstrapApplication

this happens when I run:

ng add @angular/ssr

EDIT:

I had to change:

import { appConfig } from "app/app.config";

to:

import { appConfig } from "./app/app.config";

adding a ./ to the path

JonasDev17 avatar Jan 22 '24 07:01 JonasDev17

6 years later and still hasn't been fixed. I can confirm the same behaviour.

Updated the path in main.ts only for the AppModule and finished successfully.

STotev avatar Feb 09 '24 10:02 STotev