Skip to content

Commit 0f2cb44

Browse files
committed
Ruby: add flow summary for Enumerable#sole
1 parent 5517cfa commit 0f2cb44

File tree

3 files changed

+44
-0
lines changed

3 files changed

+44
-0
lines changed

ruby/ql/lib/codeql/ruby/frameworks/ActiveSupport.qll

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -421,6 +421,16 @@ module ActiveSupport {
421421
preservesValue = true
422422
}
423423
}
424+
425+
private class SoleSummary extends SimpleSummarizedCallable {
426+
SoleSummary() { this = "sole" }
427+
428+
override predicate propagatesFlowExt(string input, string output, boolean preservesValue) {
429+
input = "Argument[self].Element[0]" and
430+
output = "ReturnValue" and
431+
preservesValue = true
432+
}
433+
}
424434
}
425435
}
426436

ruby/ql/test/library-tests/frameworks/active_support/ActiveSupportDataFlow.expected

Lines changed: 23 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
failures
2+
| hash_extensions.rb:126:10:126:19 | call to sole | Unexpected result: hasValueFlow=b |
23
edges
34
| active_support.rb:10:9:10:18 | call to source : | active_support.rb:11:10:11:10 | x : |
45
| active_support.rb:11:10:11:10 | x : | active_support.rb:11:10:11:19 | call to at |
@@ -410,6 +411,14 @@ edges
410411
| hash_extensions.rb:115:10:115:33 | call to pluck [element, element 1] : | hash_extensions.rb:115:10:115:36 | ...[...] [element 1] : |
411412
| hash_extensions.rb:115:10:115:36 | ...[...] [element 1] : | hash_extensions.rb:115:10:115:39 | ...[...] |
412413
| hash_extensions.rb:115:10:115:36 | ...[...] [element 1] : | hash_extensions.rb:115:10:115:39 | ...[...] |
414+
| hash_extensions.rb:122:15:122:25 | call to source : | hash_extensions.rb:125:10:125:15 | single [element 0] : |
415+
| hash_extensions.rb:122:15:122:25 | call to source : | hash_extensions.rb:125:10:125:15 | single [element 0] : |
416+
| hash_extensions.rb:123:14:123:24 | call to source : | hash_extensions.rb:126:10:126:14 | multi [element 0] : |
417+
| hash_extensions.rb:123:14:123:24 | call to source : | hash_extensions.rb:126:10:126:14 | multi [element 0] : |
418+
| hash_extensions.rb:125:10:125:15 | single [element 0] : | hash_extensions.rb:125:10:125:20 | call to sole |
419+
| hash_extensions.rb:125:10:125:15 | single [element 0] : | hash_extensions.rb:125:10:125:20 | call to sole |
420+
| hash_extensions.rb:126:10:126:14 | multi [element 0] : | hash_extensions.rb:126:10:126:19 | call to sole |
421+
| hash_extensions.rb:126:10:126:14 | multi [element 0] : | hash_extensions.rb:126:10:126:19 | call to sole |
413422
nodes
414423
| active_support.rb:10:9:10:18 | call to source : | semmle.label | call to source : |
415424
| active_support.rb:11:10:11:10 | x : | semmle.label | x : |
@@ -904,6 +913,18 @@ nodes
904913
| hash_extensions.rb:115:10:115:36 | ...[...] [element 1] : | semmle.label | ...[...] [element 1] : |
905914
| hash_extensions.rb:115:10:115:39 | ...[...] | semmle.label | ...[...] |
906915
| hash_extensions.rb:115:10:115:39 | ...[...] | semmle.label | ...[...] |
916+
| hash_extensions.rb:122:15:122:25 | call to source : | semmle.label | call to source : |
917+
| hash_extensions.rb:122:15:122:25 | call to source : | semmle.label | call to source : |
918+
| hash_extensions.rb:123:14:123:24 | call to source : | semmle.label | call to source : |
919+
| hash_extensions.rb:123:14:123:24 | call to source : | semmle.label | call to source : |
920+
| hash_extensions.rb:125:10:125:15 | single [element 0] : | semmle.label | single [element 0] : |
921+
| hash_extensions.rb:125:10:125:15 | single [element 0] : | semmle.label | single [element 0] : |
922+
| hash_extensions.rb:125:10:125:20 | call to sole | semmle.label | call to sole |
923+
| hash_extensions.rb:125:10:125:20 | call to sole | semmle.label | call to sole |
924+
| hash_extensions.rb:126:10:126:14 | multi [element 0] : | semmle.label | multi [element 0] : |
925+
| hash_extensions.rb:126:10:126:14 | multi [element 0] : | semmle.label | multi [element 0] : |
926+
| hash_extensions.rb:126:10:126:19 | call to sole | semmle.label | call to sole |
927+
| hash_extensions.rb:126:10:126:19 | call to sole | semmle.label | call to sole |
907928
subpaths
908929
#select
909930
| active_support.rb:182:10:182:13 | ...[...] | active_support.rb:180:10:180:17 | call to source : | active_support.rb:182:10:182:13 | ...[...] | $@ | active_support.rb:180:10:180:17 | call to source : | call to source : |
@@ -964,3 +985,5 @@ subpaths
964985
| hash_extensions.rb:114:10:114:39 | ...[...] | hash_extensions.rb:110:84:110:99 | call to source : | hash_extensions.rb:114:10:114:39 | ...[...] | $@ | hash_extensions.rb:110:84:110:99 | call to source : | call to source : |
965986
| hash_extensions.rb:115:10:115:39 | ...[...] | hash_extensions.rb:110:21:110:31 | call to source : | hash_extensions.rb:115:10:115:39 | ...[...] | $@ | hash_extensions.rb:110:21:110:31 | call to source : | call to source : |
966987
| hash_extensions.rb:115:10:115:39 | ...[...] | hash_extensions.rb:110:65:110:75 | call to source : | hash_extensions.rb:115:10:115:39 | ...[...] | $@ | hash_extensions.rb:110:65:110:75 | call to source : | call to source : |
988+
| hash_extensions.rb:125:10:125:20 | call to sole | hash_extensions.rb:122:15:122:25 | call to source : | hash_extensions.rb:125:10:125:20 | call to sole | $@ | hash_extensions.rb:122:15:122:25 | call to source : | call to source : |
989+
| hash_extensions.rb:126:10:126:19 | call to sole | hash_extensions.rb:123:14:123:24 | call to source : | hash_extensions.rb:126:10:126:19 | call to sole | $@ | hash_extensions.rb:123:14:123:24 | call to source : | call to source : |

ruby/ql/test/library-tests/frameworks/active_support/hash_extensions.rb

Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -116,3 +116,14 @@ def m_pluck(i)
116116
end
117117

118118
m_pluck(0)
119+
120+
def m_sole
121+
empty = []
122+
single = [source("a")]
123+
multi = [source("b"), source("c")]
124+
sink(empty.sole)
125+
sink(single.sole) # $ hasValueFlow=a
126+
sink(multi.sole) # TODO: model that 'sole' does not return if the receiver has multiple elements
127+
end
128+
129+
m_sole()

0 commit comments

Comments
 (0)