-
Notifications
You must be signed in to change notification settings - Fork 2.4k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(nest): add v10 migrations for tsconfig & CacheModule
- Loading branch information
1 parent
16b9b20
commit cacb2e0
Showing
3 changed files
with
387 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
139 changes: 139 additions & 0 deletions
139
packages/nest/src/migrations/update-16-4-0-cache-manager/nestjs-10-updates.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,139 @@ | ||
import { | ||
Tree, | ||
addDependenciesToPackageJson, | ||
createProjectGraphAsync, | ||
formatFiles, | ||
getProjects, | ||
joinPathFragments, | ||
updateJson, | ||
visitNotIgnoredFiles, | ||
} from '@nx/devkit'; | ||
import { NormalizedWebpackExecutorOptions } from '@nx/webpack'; | ||
import { tsquery } from '@phenomnomnominal/tsquery'; | ||
import { forEachExecutorOptions } from '@nx/devkit/src/generators/executor-options-utils'; | ||
import { | ||
ImportDeclaration, | ||
VariableStatement, | ||
ScriptTarget, | ||
isVariableStatement, | ||
} from 'typescript'; | ||
|
||
const JS_TS_FILE_MATCHER = /\.[jt]sx?$/; | ||
|
||
const importMatch = | ||
':matches(ImportDeclaration, VariableStatement):has(Identifier[name="CacheModule"], Identifier[name="CacheModule"]):has(StringLiteral[value="@nestjs/common"])'; | ||
|
||
export async function updateNestJs10(tree: Tree) { | ||
const nestProjects = await getNestProejcts(); | ||
if (nestProjects.length === 0) { | ||
return; | ||
} | ||
|
||
let installCacheModuleDeps = false; | ||
const projects = getProjects(tree); | ||
|
||
for (const projectName of nestProjects) { | ||
const projectConfig = projects.get(projectName); | ||
const tsConfig = | ||
projectConfig.targets?.build?.options?.tsConfig ?? | ||
joinPathFragments( | ||
projectConfig.root, | ||
projectConfig.projectType === 'application' | ||
? 'tsconfig.app.json' | ||
: 'tsconfig.lib.json' | ||
); | ||
|
||
if (tree.exists(tsConfig)) { | ||
updateTsConfigTarget(tree, tsConfig); | ||
} | ||
|
||
visitNotIgnoredFiles(tree, projectConfig.root, (filePath) => { | ||
if (!JS_TS_FILE_MATCHER.test(filePath)) { | ||
return; | ||
} | ||
|
||
installCacheModuleDeps = | ||
updateCacheManagerImport(tree, filePath) || installCacheModuleDeps; | ||
}); | ||
} | ||
|
||
await formatFiles(tree); | ||
|
||
return installCacheModuleDeps | ||
? addDependenciesToPackageJson( | ||
tree, | ||
{ | ||
'@nestjs/cache-manager': '^2.0.0', | ||
'cache-manager': '^5.2.3', | ||
}, | ||
{} | ||
) | ||
: () => {}; | ||
} | ||
|
||
async function getNestProejcts(): Promise<string[]> { | ||
const projectGraph = await createProjectGraphAsync(); | ||
|
||
return Object.entries(projectGraph.dependencies) | ||
.filter(([node, dep]) => | ||
dep.some( | ||
({ target }) => | ||
!projectGraph.externalNodes?.[node] && target === 'npm:@nestjs/common' | ||
) | ||
) | ||
.map(([projectName]) => projectName); | ||
} | ||
|
||
// change import { CacheModule } from '@nestjs/common'; | ||
// to import { CacheModule } from '@nestjs/cache-manager'; | ||
export function updateCacheManagerImport( | ||
tree: Tree, | ||
filePath: string | ||
): boolean { | ||
const content = tree.read(filePath, 'utf-8'); | ||
|
||
const updated = tsquery.replace( | ||
content, | ||
importMatch, | ||
|
||
(node: ImportDeclaration | VariableStatement) => { | ||
const text = node.getText(); | ||
return `${text.replace('CacheModule', '')}\n${ | ||
isVariableStatement(node) | ||
? "const { CacheModule } = require('@nestjs/cache-manager')" | ||
: "import { CacheModule } from '@nestjs/cache-manager';" | ||
}`; | ||
} | ||
); | ||
|
||
if (updated !== content) { | ||
tree.write(filePath, updated); | ||
return true; | ||
} | ||
} | ||
|
||
export function updateTsConfigTarget(tree: Tree, tsConfigPath: string) { | ||
updateJson(tree, tsConfigPath, (json) => { | ||
if (!json.compilerOptions.target) { | ||
return; | ||
} | ||
|
||
const normalizedTargetName = json.compilerOptions.target.toUpperCase(); | ||
// es6 isn't apart of the ScriptTarget enum but is a valid tsconfig target in json file | ||
const existingTarget = | ||
normalizedTargetName === 'ES6' | ||
? ScriptTarget.ES2015 | ||
: (ScriptTarget[normalizedTargetName] as unknown as ScriptTarget); | ||
|
||
if ( | ||
existingTarget < ScriptTarget.ES2021 || | ||
json.compilerOptions.target === 'ES6' | ||
) { | ||
json.compilerOptions.target = 'es2021'; | ||
} | ||
|
||
return json; | ||
}); | ||
} | ||
|
||
export default updateNestJs10; |
242 changes: 242 additions & 0 deletions
242
packages/nest/src/migrations/update-16-4-0-cache-manager/nestjs-10.spec.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,242 @@ | ||
import { | ||
ProjectConfiguration, | ||
ProjectGraph, | ||
Tree, | ||
addProjectConfiguration, | ||
readJson, | ||
} from '@nx/devkit'; | ||
import { | ||
updateNestJs10, | ||
updateCacheManagerImport, | ||
updateTsConfigTarget, | ||
} from './nestjs-10-updates'; | ||
import { createTreeWithEmptyWorkspace } from 'nx/src/devkit-testing-exports'; | ||
|
||
let projectGraph: ProjectGraph; | ||
jest.mock('@nx/devkit', () => ({ | ||
...jest.requireActual('@nx/devkit'), | ||
createProjectGraphAsync: () => Promise.resolve(projectGraph), | ||
})); | ||
describe('nestjs 10 migration changes', () => { | ||
let tree: Tree; | ||
beforeEach(() => { | ||
tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' }); | ||
|
||
jest.resetAllMocks(); | ||
}); | ||
|
||
it('should update nestjs project', async () => { | ||
tree.write( | ||
'apps/app1/main.ts', | ||
` | ||
/** | ||
* This is not a production server yet! | ||
* This is only a minimal backend to get started. | ||
*/ | ||
import { Logger } from '@nestjs/common'; | ||
import { NestFactory } from '@nestjs/core'; | ||
import { CacheModule } from '@nestjs/common'; | ||
import { AppModule } from './app/app.module'; | ||
async function bootstrap() { | ||
const app = await NestFactory.create(AppModule); | ||
const globalPrefix = 'api'; | ||
app.setGlobalPrefix(globalPrefix); | ||
const port = process.env.PORT || 3000; | ||
await app.listen(port); | ||
Logger.log('🚀 Application is running on: http://localhost:' + port + '/' + globalPrefix); | ||
} | ||
bootstrap(); | ||
` | ||
); | ||
|
||
tree.write( | ||
'apps/app1/tsconfig.app.json', | ||
JSON.stringify({ | ||
extends: './tsconfig.json', | ||
compilerOptions: { | ||
outDir: '../../dist/out-tsc', | ||
module: 'commonjs', | ||
types: ['node'], | ||
emitDecoratorMetadata: true, | ||
target: 'es2015', | ||
}, | ||
exclude: ['jest.config.ts', 'src/**/*.spec.ts', 'src/**/*.test.ts'], | ||
include: ['src/**/*.ts'], | ||
}) | ||
); | ||
addProject( | ||
tree, | ||
'app1', | ||
{ | ||
root: 'apps/app1', | ||
targets: { | ||
build: { | ||
executor: '@nx/webpack:webpack', | ||
options: { | ||
tsConfig: 'apps/app1/tsconfig.app.json', | ||
}, | ||
}, | ||
}, | ||
}, | ||
['npm:@nestjs/common'] | ||
); | ||
|
||
await updateNestJs10(tree); | ||
|
||
expect(readJson(tree, 'package.json').dependencies).toMatchInlineSnapshot(` | ||
{ | ||
"@nestjs/cache-manager": "^2.0.0", | ||
"cache-manager": "^5.2.3", | ||
} | ||
`); | ||
expect( | ||
readJson(tree, 'apps/app1/tsconfig.app.json').compilerOptions.target | ||
).toEqual('es2021'); | ||
expect(tree.read('apps/app1/main.ts', 'utf-8')).toContain( | ||
"import { CacheModule } from '@nestjs/cache-manager';" | ||
); | ||
}); | ||
|
||
it('should work with non buildable lib', async () => { | ||
tree.write( | ||
'libs/lib1/src/lib/lib1.module.ts', | ||
` | ||
import { Module, CacheModule } from '@nestjs/common'; | ||
@Module({ | ||
controllers: [], | ||
providers: [], | ||
exports: [], | ||
imports: [CacheModule.register()], | ||
}) | ||
export class LibOneModule {} | ||
` | ||
); | ||
|
||
tree.write( | ||
'libs/lib1/tsconfig.lib.json', | ||
JSON.stringify({ | ||
extends: './tsconfig.json', | ||
compilerOptions: { | ||
outDir: '../../dist/out-tsc', | ||
module: 'commonjs', | ||
types: ['node'], | ||
emitDecoratorMetadata: true, | ||
target: 'es6', | ||
}, | ||
exclude: ['jest.config.ts', 'src/**/*.spec.ts', 'src/**/*.test.ts'], | ||
include: ['src/**/*.ts'], | ||
}) | ||
); | ||
addProject( | ||
tree, | ||
'app1', | ||
{ | ||
root: 'libs/lib1', | ||
targets: {}, | ||
}, | ||
['npm:@nestjs/common'] | ||
); | ||
|
||
await updateNestJs10(tree); | ||
|
||
expect(readJson(tree, 'package.json').dependencies).toMatchInlineSnapshot(` | ||
{ | ||
"@nestjs/cache-manager": "^2.0.0", | ||
"cache-manager": "^5.2.3", | ||
} | ||
`); | ||
expect( | ||
readJson(tree, 'libs/lib1/tsconfig.lib.json').compilerOptions.target | ||
).toEqual('es2021'); | ||
expect(tree.read('libs/lib1/src/lib/lib1.module.ts', 'utf-8')).toContain( | ||
"import { CacheModule } from '@nestjs/cache-manager';" | ||
); | ||
}); | ||
|
||
it('should update cache module import', () => { | ||
tree.write( | ||
'main.ts', | ||
` | ||
import { Module, CacheModule } from '@nestjs/common'; | ||
const { Module, CacheModule } = require('@nestjs/common'); | ||
` | ||
); | ||
const actual = updateCacheManagerImport(tree, 'main.ts'); | ||
|
||
expect(tree.read('main.ts', 'utf-8')).toMatchInlineSnapshot(` | ||
" | ||
import { Module, } from '@nestjs/common'; | ||
import { CacheModule } from '@nestjs/cache-manager'; | ||
const { Module, } = require('@nestjs/common'); | ||
const { CacheModule } = require('@nestjs/cache-manager') | ||
" | ||
`); | ||
expect(actual).toBe(true); | ||
}); | ||
|
||
it('should NOT update cache module imports', () => { | ||
tree.write( | ||
'main.ts', | ||
` | ||
import { AnotherModule } from '@nestjs/common'; | ||
const { AnotherModule } = require('@nestjs/common'); | ||
` | ||
); | ||
const actual = updateCacheManagerImport(tree, 'main.ts'); | ||
|
||
expect(tree.read('main.ts', 'utf-8')).toMatchInlineSnapshot(` | ||
" | ||
import { AnotherModule } from '@nestjs/common'; | ||
const { AnotherModule } = require('@nestjs/common'); | ||
" | ||
`); | ||
expect(actual).toBeUndefined(); | ||
}); | ||
|
||
it('should update script target', () => { | ||
tree.write( | ||
'tsconfig.json', | ||
JSON.stringify({ compilerOptions: { target: 'es6' } }) | ||
); | ||
updateTsConfigTarget(tree, 'tsconfig.json'); | ||
expect(readJson(tree, 'tsconfig.json').compilerOptions.target).toBe( | ||
'es2021' | ||
); | ||
}); | ||
|
||
it('should NOT update script if over es2021', () => { | ||
tree.write( | ||
'tsconfig.json', | ||
JSON.stringify({ compilerOptions: { target: 'es2022' } }) | ||
); | ||
updateTsConfigTarget(tree, 'tsconfig.json'); | ||
expect(readJson(tree, 'tsconfig.json').compilerOptions.target).toBe( | ||
'es2022' | ||
); | ||
}); | ||
}); | ||
|
||
function addProject( | ||
tree: Tree, | ||
projectName: string, | ||
config: ProjectConfiguration, | ||
dependencies: string[] | ||
): void { | ||
projectGraph = { | ||
dependencies: { | ||
[projectName]: dependencies.map((d) => ({ | ||
source: projectName, | ||
target: d, | ||
type: 'static', | ||
})), | ||
}, | ||
nodes: { | ||
[projectName]: { data: config, name: projectName, type: 'app' }, | ||
}, | ||
}; | ||
addProjectConfiguration(tree, projectName, config); | ||
} |