Skip to content

Commit a15d53e

Browse files
authoredJun 19, 2024··
fix(component-store): resolve issues in migration script to v18 (#4403)
1 parent 67a5a93 commit a15d53e

File tree

2 files changed

+118
-6
lines changed

2 files changed

+118
-6
lines changed
 

‎modules/component-store/migrations/18_0_0-beta/index.spec.ts

+95
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import {
55
import { createWorkspace } from '@ngrx/schematics-core/testing';
66
import { tags } from '@angular-devkit/core';
77
import * as path from 'path';
8+
import { LogEntry } from '@angular-devkit/core/src/logger';
89

910
describe('ComponentStore Migration to 18.0.0-beta', () => {
1011
const collectionPath = path.join(__dirname, '../migration.json');
@@ -117,6 +118,100 @@ export class MyStore extends ComponentStore {
117118
});
118119
});
119120

121+
it('should work with prior import from same namespace', async () => {
122+
const input = tags.stripIndent`
123+
import { ComponentStore, provideComponentStore } from '@ngrx/component-store';
124+
import { tapResponse } from '@ngrx/component-store';
125+
126+
export class MyStore extends ComponentStore {}
127+
`;
128+
const output = tags.stripIndent`
129+
import { ComponentStore, provideComponentStore } from '@ngrx/component-store';
130+
import { tapResponse } from '@ngrx/operators';
131+
132+
export class MyStore extends ComponentStore {}
133+
`;
134+
await verifySchematic(input, output);
135+
});
136+
137+
it('should operate on multiple files', async () => {
138+
const inputMainOne = tags.stripIndent`
139+
import { ComponentStore, tapResponse } from '@ngrx/component-store';
140+
import { concatLatestFrom } from '@ngrx/operators';
141+
142+
@Injectable()
143+
export class MyStore extends ComponentStore {
144+
145+
}
146+
`;
147+
148+
const outputMainOne = tags.stripIndent`
149+
import { ComponentStore } from '@ngrx/component-store';
150+
import { concatLatestFrom, tapResponse } from '@ngrx/operators';
151+
152+
@Injectable()
153+
export class MyStore extends ComponentStore {
154+
155+
}
156+
`;
157+
158+
const inputMainTwo = tags.stripIndent`
159+
import { tapResponse } from "@ngrx/component-store";
160+
161+
@Injectable()
162+
export class MyStore extends ComponentStore {
163+
164+
}
165+
`;
166+
const outputMainTwo = tags.stripIndent`
167+
import { tapResponse } from '@ngrx/operators';
168+
169+
@Injectable()
170+
export class MyStore extends ComponentStore {
171+
172+
}
173+
`;
174+
appTree.create('mainOne.ts', inputMainOne);
175+
appTree.create('mainTwo.ts', inputMainTwo);
176+
177+
const tree = await schematicRunner.runSchematic(
178+
`ngrx-component-store-migration-18-beta`,
179+
{},
180+
appTree
181+
);
182+
183+
const actualMainOne = tree.readContent('mainOne.ts');
184+
const actualMainTwo = tree.readContent('mainTwo.ts');
185+
186+
expect(actualMainOne).toBe(outputMainOne);
187+
expect(actualMainTwo).toBe(outputMainTwo);
188+
});
189+
190+
it('should report a warning on multiple imports of tapResponse', async () => {
191+
const input = tags.stripIndent`
192+
import { tapResponse } from '@ngrx/component-store';
193+
import { tapResponse, ComponentStore } from '@ngrx/component-store';
194+
195+
class SomeEffects {}
196+
`;
197+
198+
appTree.create('main.ts', input);
199+
const logEntries: LogEntry[] = [];
200+
schematicRunner.logger.subscribe((logEntry) => logEntries.push(logEntry));
201+
await schematicRunner.runSchematic(
202+
`ngrx-component-store-migration-18-beta`,
203+
{},
204+
appTree
205+
);
206+
207+
expect(logEntries).toHaveLength(1);
208+
expect(logEntries[0]).toMatchObject({
209+
message:
210+
'[@ngrx/component-store] Skipping because of multiple `tapResponse` imports',
211+
level: 'info',
212+
});
213+
});
214+
120215
it('should add @ngrx/operators if they are missing', async () => {
121216
const originalPackageJson = JSON.parse(
122217
appTree.readContent('/package.json')

‎modules/component-store/migrations/18_0_0-beta/index.ts

+23-6
Original file line numberDiff line numberDiff line change
@@ -17,30 +17,46 @@ import { createRemoveChange } from '../../schematics-core/utility/change';
1717

1818
export function migrateTapResponseImport(): Rule {
1919
return (tree: Tree, ctx: SchematicContext) => {
20-
const changes: Change[] = [];
2120
addPackageToPackageJson(tree, 'dependencies', '@ngrx/operators', '^18.0.0');
2221

2322
visitTSSourceFiles(tree, (sourceFile) => {
2423
const importDeclarations = new Array<ts.ImportDeclaration>();
2524
getImportDeclarations(sourceFile, importDeclarations);
2625

27-
const componentStoreImportsAndDeclaration = importDeclarations
26+
const componentStoreImportsAndDeclarations = importDeclarations
2827
.map((componentStoreImportDeclaration) => {
2928
const componentStoreImports = getComponentStoreNamedBinding(
3029
componentStoreImportDeclaration
3130
);
3231
if (componentStoreImports) {
33-
return { componentStoreImports, componentStoreImportDeclaration };
32+
if (
33+
componentStoreImports.elements.some(
34+
(element) => element.name.getText() === 'tapResponse'
35+
)
36+
) {
37+
return { componentStoreImports, componentStoreImportDeclaration };
38+
}
39+
return undefined;
3440
} else {
3541
return undefined;
3642
}
3743
})
38-
.find(Boolean);
44+
.filter(Boolean);
3945

40-
if (!componentStoreImportsAndDeclaration) {
46+
if (componentStoreImportsAndDeclarations.length === 0) {
47+
return;
48+
} else if (componentStoreImportsAndDeclarations.length > 1) {
49+
ctx.logger.info(
50+
'[@ngrx/component-store] Skipping because of multiple `tapResponse` imports'
51+
);
4152
return;
4253
}
4354

55+
const [componentStoreImportsAndDeclaration] =
56+
componentStoreImportsAndDeclarations;
57+
if (!componentStoreImportsAndDeclaration) {
58+
return;
59+
}
4460
const { componentStoreImports, componentStoreImportDeclaration } =
4561
componentStoreImportsAndDeclaration;
4662

@@ -53,6 +69,7 @@ export function migrateTapResponseImport(): Rule {
5369
.map((element) => element.name.getText())
5470
.join(', ');
5571

72+
const changes: Change[] = [];
5673
// Remove `tapResponse` from @ngrx/component-store and leave the other imports
5774
if (otherComponentStoreImports) {
5875
changes.push(
@@ -107,7 +124,7 @@ export function migrateTapResponseImport(): Rule {
107124
new InsertChange(
108125
sourceFile.fileName,
109126
componentStoreImportDeclaration.getEnd() + 1,
110-
`${newOperatorsImport}\n`
127+
`${newOperatorsImport}\n` // not os-independent for snapshot tests
111128
)
112129
);
113130
}

0 commit comments

Comments
 (0)
Please sign in to comment.