Line: 1 to 1 | ||||||||
---|---|---|---|---|---|---|---|---|
Execute After Put | ||||||||
Added: | ||||||||
> > | ||||||||
ContextAn API specific design rule for thetyRuBa.tdbc package. | ||||||||
Line: 51 to 53 | ||||||||
Definition | ||||||||
Changed: | ||||||||
< < | JQuery | |||||||
> > | JQuery | |||||||
This rule cannot be expressed in JQuery as is because it requires some dataflow. | ||||||||
Line: 81 to 83 | ||||||||
NOT( calls(?methodContext, PreparedInsert.execute(), ?) ) | ||||||||
Changed: | ||||||||
< < | MetaL | |||||||
> > | MetaL | |||||||
Yes, it can be expressed, but I don't know how. Somebody fill this in? | ||||||||
Changed: | ||||||||
< < | AspectJ | |||||||
> > | AspectJ | |||||||
Probably cannot be expressed in AspectJ. Perhaps it is possible to implement a dynamic checker for this, but it is complicated (need to track all instances put into during calling of a method and then check that they are all executed by the time the method exits). | ||||||||
Changed: | ||||||||
< < | Hypothethical Wishful Thinking | |||||||
> > | HypotheticaL | |||||||
Version 1: Makes use of explicit PCD that specify both calls must be on same instance and within the same method context. |
Line: 1 to 1 | ||||||||
---|---|---|---|---|---|---|---|---|
Changed: | ||||||||
< < |
| |||||||
> > |
| |||||||
Execute After PutContext | ||||||||
Line: 8 to 8 | ||||||||
Description | ||||||||
Changed: | ||||||||
< < | Calls to PreparedInsert.put*(..) should always be followed by a call to PreparedInsert.executeInsert()= within the same method context. | |||||||
> > | Calls to PreparedInsert.put*(..) should always be followed by a call to PreparedInsert.executeInsert() within the same method context. | |||||||
Rationale | ||||||||
Line: 18 to 18 | ||||||||
The interesting property (see DesignRules) we are addressing with this design rule is: did we not forget anywhere to call the execute method after putting values into a PreparedInsert .
Adopting the design rule makes it feasible to check this property manually, or with a relatively | ||||||||
Changed: | ||||||||
< < | straightforward intraprocedural analysis. The desing rule accomplishes this feat by restricting the scope of the analysis required to verify the property. | |||||||
> > | straightforward intraprocedural analysis, because adherence to the desing rule limits the scope of the required analysis. | |||||||
Example | ||||||||
Changed: | ||||||||
< < | The following code contains a copy-paste bug: | |||||||
> > | The following code contains a copy-paste bug: salaryFact never gets executed whereas employeeFact is executed twice. | |||||||
class Manager { | ||||||||
Line: 47 to 47 | ||||||||
} | ||||||||
Changed: | ||||||||
< < | This bug could be detected by a checker for our design rule using simple intraprocedural
dataflow analysis:
The instance in salaryFact never gets executed after the calls to put and putInt . | |||||||
> > | This bug could be detected by a checker for our design rule using simple intraprocedural analysis. | |||||||
Definition |
Line: 1 to 1 | ||||||||
---|---|---|---|---|---|---|---|---|
Added: | ||||||||
> > |
Execute After PutContextAn API specific design rule for thetyRuBa.tdbc package.
DescriptionCalls toPreparedInsert.put*(..) should always be followed by a call to PreparedInsert.executeInsert()= within the same method context.
RationalePutting values into aPreparedInsert is done with the intention of executing the
PreparedInsert to add facts to a TyRuBa database.
The interesting property (see DesignRules) we are addressing with this design rule is: did we not forget anywhere to call the execute method after putting values into a PreparedInsert .
Adopting the design rule makes it feasible to check this property manually, or with a relatively
straightforward intraprocedural analysis. The desing rule accomplishes this feat by restricting the scope of the analysis required to verify the property.
ExampleThe following code contains a copy-paste bug:class Manager { PreparedInsert employeeFact = null; // initialized elsewhere PreparedInsert salaryFact = null; // initialized elsewhere ... void addEmployee(String managerID, String employeeID, int employeeSalary) { ... employeeFact.put("manager", managerID); employeeFact.put("employee", employeeID); employeeFact.executeInsert(); salaryFact.put("employee", employeeID); salaryFact.putInt("salary", salary); employeeFact.executeInsert(); } }This bug could be detected by a checker for our design rule using simple intraprocedural dataflow analysis: The instance in salaryFact never gets executed after the calls to put and putInt .
DefinitionJQueryThis rule cannot be expressed in JQuery as is because it requires some dataflow. A weaker version of the rule can be checked: all methods that callPreparedInsert.put*(..) must also call PreparedInsert.executeInsert().
This weaker version will find some violations of the stronger design rule, but it would not
catch the bug in the example above.
The following JQuery query finds violations of the weaker rule:
calls(?methodContext,?put,?), re_name(?put,/^put/), method(?PreparedInsert,?put), name(?PreparedInsert,PreparedInsert), NOT( EXISTS ?execute : calls(?methodContext, ?execute, ?), method(?PreparedInsert,?execute), name(?execute,execute) )Note that this would look a lot nicer if we had a nicer pattern syntax to match the method PreparedInsert.execute() and PreparedInsert.put*(..) signatures with an AspectJ like
syntax:
calls(?methodContext, PreparedInsert.put*(..), ?), NOT( calls(?methodContext, PreparedInsert.execute(), ?) ) MetaLYes, it can be expressed, but I don't know how. Somebody fill this in?AspectJProbably cannot be expressed in AspectJ. Perhaps it is possible to implement a dynamic checker for this, but it is complicated (need to track all instances put into during calling of a method and then check that they are all executed by the time the method exits).Hypothethical Wishful ThinkingVersion 1: Makes use of explicit PCD that specify both calls must be on same instance and within the same method context.forall calls(PreparedInsert.put*(..)) && targetInstance(?this) && withincode(?methodContext) ) always followedby( PreparedInsert.executeInsert() && targetInstance(?this) && withincode(?methodContext) )Version 2: Assumes that when a rule talks about calls to non static methods we are implicitly assuming that calls are on the same instance. forall calls(PreparedInsert.put*(..)) && withincode(?methodContext) ) always followedby( PreparedInsert.executeInsert() && withincode(?methodContext) )Version 3: Assumes as in version 2 and also makes similar asumption for the calling context (withincode). forall calls(PreparedInsert.put*(..)) always followedby( PreparedInsert.executeInsert() ) RuminationsLooking at this example and in particular the different versions in the hypothethical syntax we observe some interesting ideas: LTL? Forall, always, exists, never etc. seem to be useful. Do we need LTL or something like that? Implicit context assumptions: It looks like a useful trick to simplify rules by implicitly assuming that multiple parts of the rule are implicitly asuming they are talking about same joinpoint and the same context, unless explicitly stating otherwise. This makes this rule look much more like the natural formulation in English. This trick is in fact also used by AspectJ where a pointcut always implicitly talks about a single joinpoint (thisjoinpoint). The hypothethical syntax above does that as well and in version three generalizes that to all context. Principle: make rules that are harder to verify harder to express. The assumption of implicit, limited methodContext scope does that. It seems like a good principle:
|