Ahh ok so example 1 should deliver all subtype1 instances that have subvalue1 <> 100 and all instances of other types, correct? That would require putting negations to atoms and introduce a rule like: "Negated predicates e.g. treat(e as SubType1).subvalue1 <> 100 should be rendered like (type(e) not in (SubType1) or subvalue1 <> 100). Apparently I have forgotten to think about negations. The nullness checks of subtype properties e.g. treat(e as SubType1).subvalue1 is null should be implemented like type(e) = SubType1 and subvalue1 is null? That makes it a bit more complicated as the treat could be somewhere within a more complex expression like in e.g. case when ... then treat(e as SubType1).subvalue1 else treat(e as SubType2).subvalue2 end is null but it should be doable. In the end, your approach to simply translate treat(e as SubType1).subvalue1 OPERATOR ... to type(e) = SubType1 and subvalue1 OPERATOR ... for predicates now makes much more sense. Applying that and then a predicate simplification optimization are probably simpler than my approach. Would you say this behavior should always be like that in all conditional contexts(WHERE, ON, HAVING, condition part of CASE WHEN)? Treats in the SELECT, ORDER BY and GROUP BY clauses could still rely on the NULL value produced by left outer joins and only produce the ugly case when statement for single table inheritance shared columns. |