Skip to content

Commit 19656b8

Browse files
authored
feat: support tsconfig paths and package-relative imports [sc-22644] (#1006)
* feat: async dependency collector * refactor: switch to a sync implementation to avoid changes to constructs * refactor: remove unnecessary try/catch * feat: look up original TS sources when dealing with folders as modules * fix: include package.json when resolving module folders * feat: let parser know how tsconfig paths were resolved for a file This optionally allows the parser to massage the file structure into an alternate output format. * feat: add jsconfig.json support * fix: supported module check was accidentally negated * feat: bundle relevant tsconfig/jsconfig.json files These files are currently not utilized by the backend but they might be in the near future. * feat: cache all source files and set up unique IDs for later dedup purposes * feat: add tests for tsconfig behavior * feat: more complete support for imports with extensions * chore: remove unused code * fix: remove mistakenly implemented useless package-relative path support * fix: remove unused variable * fix: remove unused variable * feat: cache common dependencies within a Session Shares a common Parser (or Parsers, one per runtime) within a Session and avoids unnecessary AST walks for filePaths the parser has already seen when parsing other entrypoints. Share PackageFilesResolver so that its file caches can be shared within the same Parser (which is now also shared within the Session), and cache its result per filePath to avoid duplicate work. Helps use cases where there are multiple entrypoints that share common libraries. * fix: remove allowImportingTsExtensions support Does not play well during a nested lookup when we're looking up a path that we've already resolved to a candidate with a .ts extension earlier in the process. Not a super useful feature anyway. * chore: empty out package-lock.json in fixtures, keep file * feat: support `index.json` which apparently works
1 parent 5dbc83f commit 19656b8

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

48 files changed

+1400
-111
lines changed

packages/cli/src/constructs/api-check.ts

+1-5
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ import { Check, CheckProps } from './check'
33
import { HttpHeader } from './http-header'
44
import { Session } from './project'
55
import { QueryParam } from './query-param'
6-
import { Parser } from '../services/check-parser/parser'
76
import { pathToPosix } from '../services/util'
87
import { printDeprecationWarning } from '../reporters/util'
98
import { Content, Entrypoint } from './construct'
@@ -352,10 +351,7 @@ export class ApiCheck extends Check {
352351
if (!runtime) {
353352
throw new Error(`${runtimeId} is not supported`)
354353
}
355-
const parser = new Parser({
356-
supportedNpmModules: Object.keys(runtime.dependencies),
357-
checkUnsupportedModules: Session.verifyRuntimeDependencies,
358-
})
354+
const parser = Session.getParser(runtime)
359355
const parsed = parser.parse(absoluteEntrypoint)
360356
// Maybe we can get the parsed deps with the content immediately
361357

packages/cli/src/constructs/browser-check.ts

+1-5
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import * as path from 'path'
22
import { Check, CheckProps } from './check'
33
import { Session } from './project'
4-
import { Parser } from '../services/check-parser/parser'
54
import { CheckConfigDefaults } from '../services/checkly-config-loader'
65
import { pathToPosix } from '../services/util'
76
import { Content, Entrypoint } from './construct'
@@ -119,10 +118,7 @@ export class BrowserCheck extends Check {
119118
if (!runtime) {
120119
throw new Error(`${runtimeId} is not supported`)
121120
}
122-
const parser = new Parser({
123-
supportedNpmModules: Object.keys(runtime.dependencies),
124-
checkUnsupportedModules: Session.verifyRuntimeDependencies,
125-
})
121+
const parser = Session.getParser(runtime)
126122
const parsed = parser.parse(entry)
127123
// Maybe we can get the parsed deps with the content immediately
128124

packages/cli/src/constructs/multi-step-check.ts

+1-5
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import * as path from 'path'
22
import { Check, CheckProps } from './check'
33
import { Session } from './project'
4-
import { Parser } from '../services/check-parser/parser'
54
import { CheckConfigDefaults } from '../services/checkly-config-loader'
65
import { pathToPosix } from '../services/util'
76
import { Content, Entrypoint } from './construct'
@@ -104,10 +103,7 @@ export class MultiStepCheck extends Check {
104103
if (!runtime) {
105104
throw new Error(`${runtimeId} is not supported`)
106105
}
107-
const parser = new Parser({
108-
supportedNpmModules: Object.keys(runtime.dependencies),
109-
checkUnsupportedModules: Session.verifyRuntimeDependencies,
110-
})
106+
const parser = Session.getParser(runtime)
111107
const parsed = parser.parse(entry)
112108
// Maybe we can get the parsed deps with the content immediately
113109

packages/cli/src/constructs/project.ts

+18
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import * as api from '../rest/api'
22
import { CheckConfigDefaults } from '../services/checkly-config-loader'
3+
import { Parser } from '../services/check-parser/parser'
34
import { Construct } from './construct'
45
import { ValidationError } from './validator-error'
56

@@ -147,6 +148,7 @@ export class Session {
147148
static loadingChecklyConfigFile: boolean
148149
static checklyConfigFileConstructs?: Construct[]
149150
static privateLocations: PrivateLocationApi[]
151+
static parsers = new Map<string, Parser>()
150152

151153
static registerConstruct (construct: Construct) {
152154
if (Session.project) {
@@ -191,4 +193,20 @@ export class Session {
191193
}
192194
return Session.availableRuntimes[effectiveRuntimeId]
193195
}
196+
197+
static getParser (runtime: Runtime): Parser {
198+
const cachedParser = Session.parsers.get(runtime.name)
199+
if (cachedParser !== undefined) {
200+
return cachedParser
201+
}
202+
203+
const parser = new Parser({
204+
supportedNpmModules: Object.keys(runtime.dependencies),
205+
checkUnsupportedModules: Session.verifyRuntimeDependencies,
206+
})
207+
208+
Session.parsers.set(runtime.name, parser)
209+
210+
return parser
211+
}
194212
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 'hello'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 'world'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
import { } from './dep1'
2+
import { } from './dep2.js'
3+
import { } from './dep3.js'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 'hello'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 'world'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
import { } from './dep1'
2+
import { } from './dep1.ts'
3+
import { } from './dep1.js'
4+
import { } from './dep2'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value1 = 'value1'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value2 = 'value2'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value3 = 'value3'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
export { value1 } from './dep1.ts'
2+
export { value2 } from './dep2.js'
3+
export { value3 } from './dep3'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
{
2+
"compilerOptions": {
3+
"target": "ES2022",
4+
"module": "NodeNext",
5+
"moduleResolution": "nodenext",
6+
"esModuleInterop": true,
7+
"baseUrl": ".",
8+
"noEmit": true,
9+
"allowImportingTsExtensions": true
10+
}
11+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
node_modules/
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
v22.12.0
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
import { defineConfig } from 'checkly'
2+
3+
const config = defineConfig({
4+
projectName: 'TSConfig Paths Sample Project',
5+
logicalId: 'tsconfig-paths-sample-project',
6+
checks: {
7+
frequency: 10,
8+
locations: ['us-east-1'],
9+
tags: ['mac'],
10+
runtimeId: '2024.09',
11+
checkMatch: '**/__checks__/**/*.check.ts',
12+
browserChecks: {
13+
testMatch: '**/__checks__/**/*.spec.ts',
14+
},
15+
},
16+
cli: {
17+
runLocation: 'us-east-1',
18+
reporters: ['list'],
19+
},
20+
})
21+
22+
export default config
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 'file1'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 'file2'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export { value } from './file2'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 'file2'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
export { value as value1 } from './file1'
2+
export { value as value2 } from './file2'
3+
export { value as value3 } from './folder/file1'

packages/cli/src/services/check-parser/__tests__/check-parser-fixtures/tsconfig-paths-sample-project/lib1/package-lock.json

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
{
2+
"name": "@internal/lib",
3+
"main": "dist/index.js",
4+
"scripts": {
5+
"clean": "rimraf ./dist",
6+
"prepare": "npm run clean && tsc --build"
7+
},
8+
"devDependencies": {
9+
"rimraf": "^6.0.1",
10+
"typescript": "^5.7.2"
11+
}
12+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
{
2+
"exclude": [
3+
"dist",
4+
"node_modules"
5+
],
6+
"include": [
7+
"./**/*.ts"
8+
],
9+
"compilerOptions": {
10+
"target": "ES2022",
11+
"module": "NodeNext",
12+
"moduleResolution": "nodenext",
13+
"outDir": "dist",
14+
"declaration": true,
15+
"sourceMap": true,
16+
"declarationMap": true
17+
}
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const name = 'lib2'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 11517
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
{
2+
"compilerOptions": {
3+
"target": "ES2022",
4+
"module": "NodeNext",
5+
"moduleResolution": "nodenext",
6+
"esModuleInterop": true,
7+
"baseUrl": "."
8+
}
9+
}

packages/cli/src/services/check-parser/__tests__/check-parser-fixtures/tsconfig-paths-sample-project/package-lock.json

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
{
2+
"name": "tsconfig-paths-sample-project",
3+
"dependencies": {
4+
"checkly": "^4.15.0"
5+
},
6+
"devDependencies": {
7+
"@playwright/test": "^1.49.1",
8+
"typescript": "^5.7.2"
9+
}
10+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
import { value1 } from '@internal/lib1'
2+
import { value } from '@/foo/bar'
3+
import { name } from 'lib2'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
{
2+
"compilerOptions": {
3+
"target": "ES2022",
4+
"module": "NodeNext",
5+
"moduleResolution": "nodenext",
6+
"esModuleInterop": true,
7+
"baseUrl": ".",
8+
"paths": {
9+
"@internal/lib1": [
10+
"./lib1"
11+
],
12+
"@/*": [
13+
"./lib3/*"
14+
]
15+
}
16+
}
17+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const value = 'nothing here'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
{
2+
"compilerOptions": {
3+
"target": "ES2022",
4+
"module": "NodeNext",
5+
"moduleResolution": "nodenext",
6+
"esModuleInterop": true,
7+
"baseUrl": ".",
8+
"paths": {
9+
"@internal/lib1": [
10+
"./lib1"
11+
],
12+
"@/*": [
13+
"./lib3/*"
14+
]
15+
}
16+
}
17+
}

packages/cli/src/services/check-parser/__tests__/check-parser.spec.ts

+75
Original file line numberDiff line numberDiff line change
@@ -127,6 +127,81 @@ describe('dependency-parser - parser()', () => {
127127
])
128128
})
129129

130+
it('should parse typescript dependencies using tsconfig', () => {
131+
const toAbsolutePath = (...filepath: string[]) => path.join(__dirname, 'check-parser-fixtures', 'tsconfig-paths-sample-project', ...filepath)
132+
const parser = new Parser({
133+
supportedNpmModules: defaultNpmModules,
134+
})
135+
const { dependencies } = parser.parse(toAbsolutePath('src', 'entrypoint.ts'))
136+
expect(dependencies.map(d => d.filePath).sort()).toEqual([
137+
toAbsolutePath('lib1', 'file1.ts'),
138+
toAbsolutePath('lib1', 'file2.ts'),
139+
toAbsolutePath('lib1', 'folder', 'file1.ts'),
140+
toAbsolutePath('lib1', 'folder', 'file2.ts'),
141+
toAbsolutePath('lib1', 'index.ts'),
142+
toAbsolutePath('lib1', 'package.json'),
143+
toAbsolutePath('lib1', 'tsconfig.json'),
144+
toAbsolutePath('lib2', 'index.ts'),
145+
toAbsolutePath('lib3', 'foo', 'bar.ts'),
146+
toAbsolutePath('tsconfig.json'),
147+
])
148+
})
149+
150+
it('should not include tsconfig if not needed', () => {
151+
const toAbsolutePath = (...filepath: string[]) => path.join(__dirname, 'check-parser-fixtures', 'tsconfig-paths-unused', ...filepath)
152+
const parser = new Parser({
153+
supportedNpmModules: defaultNpmModules,
154+
})
155+
const { dependencies } = parser.parse(toAbsolutePath('src', 'entrypoint.ts'))
156+
expect(dependencies.map(d => d.filePath).sort()).toEqual([])
157+
})
158+
159+
it('should support importing ts extensions if allowed', () => {
160+
const toAbsolutePath = (...filepath: string[]) => path.join(__dirname, 'check-parser-fixtures', 'tsconfig-allow-importing-ts-extensions', ...filepath)
161+
const parser = new Parser({
162+
supportedNpmModules: defaultNpmModules,
163+
})
164+
const { dependencies } = parser.parse(toAbsolutePath('src', 'entrypoint.ts'))
165+
expect(dependencies.map(d => d.filePath).sort()).toEqual([
166+
toAbsolutePath('src', 'dep1.ts'),
167+
toAbsolutePath('src', 'dep2.ts'),
168+
toAbsolutePath('src', 'dep3.ts'),
169+
])
170+
})
171+
172+
it('should not import TS files from a JS file', () => {
173+
const toAbsolutePath = (...filepath: string[]) => path.join(__dirname, 'check-parser-fixtures', 'no-import-ts-from-js', ...filepath)
174+
const parser = new Parser({
175+
supportedNpmModules: defaultNpmModules,
176+
})
177+
expect.assertions(1)
178+
try {
179+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
180+
const { dependencies } = parser.parse(toAbsolutePath('entrypoint.js'))
181+
} catch (err) {
182+
expect(err).toMatchObject({
183+
missingFiles: [
184+
toAbsolutePath('dep1'),
185+
toAbsolutePath('dep1.ts'),
186+
toAbsolutePath('dep1.js'),
187+
],
188+
})
189+
}
190+
})
191+
192+
it('should import JS files from a TS file', () => {
193+
const toAbsolutePath = (...filepath: string[]) => path.join(__dirname, 'check-parser-fixtures', 'import-js-from-ts', ...filepath)
194+
const parser = new Parser({
195+
supportedNpmModules: defaultNpmModules,
196+
})
197+
const { dependencies } = parser.parse(toAbsolutePath('entrypoint.ts'))
198+
expect(dependencies.map(d => d.filePath).sort()).toEqual([
199+
toAbsolutePath('dep1.js'),
200+
toAbsolutePath('dep2.js'),
201+
toAbsolutePath('dep3.ts'),
202+
])
203+
})
204+
130205
it('should handle ES Modules', () => {
131206
const toAbsolutePath = (...filepath: string[]) => path.join(__dirname, 'check-parser-fixtures', 'esmodules-example', ...filepath)
132207
const parser = new Parser({

0 commit comments

Comments
 (0)