Skip to content

Commit fe9b23d

Browse files
committed
JS: Move nodeRefersToModule into NameResolution
1 parent c8dbe53 commit fe9b23d

File tree

2 files changed

+75
-72
lines changed

2 files changed

+75
-72
lines changed

javascript/ql/lib/semmle/javascript/internal/NameResolution.qll

+75
Original file line numberDiff line numberDiff line change
@@ -319,4 +319,79 @@ module NameResolution {
319319
* Gets a node to which the given module flows.
320320
*/
321321
predicate trackModule = ValueFlow::TrackNode<ModuleLike>::track/1;
322+
323+
/**
324+
* Holds if `moduleName` appears to start with a package name, as opposed to a relative file import.
325+
*/
326+
bindingset[moduleName]
327+
private predicate isExternalModuleName(string moduleName) {
328+
not moduleName.regexpMatch("^(\\.|/).*")
329+
}
330+
331+
bindingset[name]
332+
private string normalizeModuleName(string name) {
333+
result =
334+
name.regexpReplaceAll("^node:", "")
335+
.regexpReplaceAll("\\.[jt]sx?$", "")
336+
.regexpReplaceAll("/(index)?$", "")
337+
}
338+
339+
/** Appends a name onto a qualified name */
340+
bindingset[a, b]
341+
string append(string a, string b) {
342+
if b = "default"
343+
then result = a
344+
else (
345+
(if a = "" or b = "" then result = a + b else result = a + "." + b) and
346+
result.length() < 100
347+
)
348+
}
349+
350+
/**
351+
* Holds if `node` is a reference to the given module, or a qualified name rooted in that module.
352+
*
353+
* If `qualifiedName` is empty, `node` refers to the module itself.
354+
*
355+
* If `mod` is the string `"global"`, `node` refers to a global access path.
356+
*
357+
* Unlike `trackModule`, this is intended to track uses of external packages.
358+
*/
359+
predicate nodeRefersToModule(Node node, string mod, string qualifiedName) {
360+
exists(PathExpr path |
361+
path = any(Import imprt).getImportedPath() or
362+
path = any(ReExportDeclaration e).getImportedPath()
363+
|
364+
node = path and
365+
mod = normalizeModuleName(path.getValue()) and
366+
isExternalModuleName(mod) and
367+
qualifiedName = ""
368+
)
369+
or
370+
mod = "global" and
371+
exists(LocalNamespaceAccess access |
372+
node = access and
373+
not exists(access.getLocalNamespaceName()) and
374+
access.getName() = qualifiedName
375+
)
376+
or
377+
// Additionally track through bulk re-exports (`export * from 'mod`).
378+
// These are normally handled by 'exportAs' which supports various shadowing rules,
379+
// but has no effect when the ultimate re-exported module is not resolved to a Module.
380+
// We propagate external module refs through bulk re-exports and ignore shadowing rules.
381+
exists(BulkReExportDeclaration reExport |
382+
nodeRefersToModule(reExport.getImportedPath(), mod, qualifiedName) and
383+
node = reExport.getContainer()
384+
)
385+
or
386+
exists(Node mid |
387+
nodeRefersToModule(mid, mod, qualifiedName) and
388+
ValueFlow::step(mid, node)
389+
)
390+
or
391+
exists(Node mid, string prefix, string step |
392+
nodeRefersToModule(mid, mod, prefix) and
393+
readStep(mid, step, node) and
394+
qualifiedName = append(prefix, step)
395+
)
396+
}
322397
}

javascript/ql/lib/semmle/javascript/internal/UnderlyingTypes.qll

-72
Original file line numberDiff line numberDiff line change
@@ -9,68 +9,6 @@ private import semmle.javascript.internal.NameResolution::NameResolution
99
* Provides name resolution and propagates type information.
1010
*/
1111
module UnderlyingTypes {
12-
/**
13-
* Holds if `moduleName` appears to start with a package name, as opposed to a relative file import.
14-
*/
15-
bindingset[moduleName]
16-
private predicate isExternalModuleName(string moduleName) {
17-
not moduleName.regexpMatch("^(\\.|/).*")
18-
}
19-
20-
bindingset[name]
21-
private string normalizeModuleName(string name) {
22-
result =
23-
name.regexpReplaceAll("^node:", "")
24-
.regexpReplaceAll("\\.[jt]sx?$", "")
25-
.regexpReplaceAll("/(index)?$", "")
26-
}
27-
28-
/**
29-
* Holds if `node` is a reference to the given module, or a qualified name rooted in that module.
30-
*
31-
* If `qualifiedName` is empty, `node` refers to the module itself.
32-
*
33-
* If `mod` is the string `"global"`, `node` refers to a global access path.
34-
*/
35-
predicate nodeRefersToModule(Node node, string mod, string qualifiedName) {
36-
exists(PathExpr path |
37-
path = any(Import imprt).getImportedPath() or
38-
path = any(ReExportDeclaration e).getImportedPath()
39-
|
40-
node = path and
41-
mod = normalizeModuleName(path.getValue()) and
42-
isExternalModuleName(mod) and
43-
qualifiedName = ""
44-
)
45-
or
46-
mod = "global" and
47-
exists(LocalNamespaceAccess access |
48-
node = access and
49-
not exists(access.getLocalNamespaceName()) and
50-
access.getName() = qualifiedName
51-
)
52-
or
53-
// Additionally track through bulk re-exports (`export * from 'mod`).
54-
// These are normally handled by 'exportAs' which supports various shadowing rules,
55-
// but has no effect when the ultimate re-exported module is not resolved to a Module.
56-
// We propagate external module refs through bulk re-exports and ignore shadowing rules.
57-
exists(BulkReExportDeclaration reExport |
58-
nodeRefersToModule(reExport.getImportedPath(), mod, qualifiedName) and
59-
node = reExport.getContainer()
60-
)
61-
or
62-
exists(Node mid |
63-
nodeRefersToModule(mid, mod, qualifiedName) and
64-
ValueFlow::step(mid, node)
65-
)
66-
or
67-
exists(Node mid, string prefix, string step |
68-
nodeRefersToModule(mid, mod, prefix) and
69-
readStep(mid, step, node) and
70-
qualifiedName = append(prefix, step)
71-
)
72-
}
73-
7412
private predicate subtypeStep(Node node1, Node node2) {
7513
exists(ClassOrInterface cls |
7614
(
@@ -133,16 +71,6 @@ module UnderlyingTypes {
13371
)
13472
}
13573

136-
bindingset[a, b]
137-
private string append(string a, string b) {
138-
if b = "default"
139-
then result = a
140-
else (
141-
(if a = "" or b = "" then result = a + b else result = a + "." + b) and
142-
result.length() < 100
143-
)
144-
}
145-
14674
predicate nodeHasUnderlyingType(Node node, string mod, string name) {
14775
nodeRefersToModule(node, mod, name)
14876
or

0 commit comments

Comments
 (0)