Skip to content
Snippets Groups Projects
unittest_analyze.py 24.8 KiB
Newer Older
# copyright 2004-2010 LOGILAB S.A. (Paris, FRANCE), all rights reserved.
# contact http://www.logilab.fr/ -- mailto:contact@logilab.fr
#
# This file is part of rql.
#
# rql is free software: you can redistribute it and/or modify it under the
# terms of the GNU Lesser General Public License as published by the Free
# Software Foundation, either version 2.1 of the License, or (at your option)
# any later version.
#
# rql is distributed in the hope that it will be useful, but WITHOUT ANY
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
# A PARTICULAR PURPOSE.  See the GNU Lesser General Public License for more
# details.
#
# You should have received a copy of the GNU Lesser General Public License along
# with rql. If not, see <http://www.gnu.org/licenses/>.
Sylvain Thénault's avatar
Sylvain Thénault committed
from logilab.common.testlib import TestCase, unittest_main, mock_object as mock
Nicolas Chauvat's avatar
Nicolas Chauvat committed

from rql import RQLHelper, TypeResolverException

Sylvain's avatar
Sylvain committed
FINAL_ETYPES = ('String', 'Boolean', 'Int', 'Float', 'Date', 'Datetime')

class ERSchema(object):
Sylvain's avatar
Sylvain committed

    def __cmp__(self, other):
        other = getattr(other, 'type', other)
        return cmp(self.type, other)
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
    def __hash__(self):
        return hash(self.type)
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
    def __str__(self):
        return self.type
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed

class RelationSchema(ERSchema):
Sylvain Thénault's avatar
Sylvain Thénault committed
    def __init__(self, assoc_types, symmetric=False, card=None):
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        self.assoc_types = assoc_types
        self.subj_types = [e_type[0] for e_type in assoc_types]
        d = {}
        for e_type, dest_types in assoc_types:
            for e_type in dest_types:
                d[e_type] = 1
        self.obj_types = d.keys()
Sylvain Thénault's avatar
Sylvain Thénault committed
        self.symmetric = symmetric
        self.inlined = False
Sylvain Thénault's avatar
Sylvain Thénault committed
        self.rdefs = {}
        for subjtype, dest_types in self.assoc_types:
            for objtype in dest_types:
                self.rdefs[(subjtype, objtype)] = mock(subject=subjtype, object=objtype, cardinality=self.card)

Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
    def associations(self):
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        return self.assoc_types
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
    def subjects(self, etype=None):
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        return self.subj_types
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
    def objects(self, etype=None):
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        return self.obj_types

Sylvain's avatar
Sylvain committed
        return self.obj_types[0] in FINAL_ETYPES
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
class EntitySchema(ERSchema):
    def __init__(self, type, specialized_by=None):
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        self.type = type
        self._specialized_by = specialized_by or ()
        self.final = self.type in FINAL_ETYPES
Sylvain Thénault's avatar
Sylvain Thénault committed

    def specialized_by(self):
        return self._specialized_by
Sylvain Thénault's avatar
Sylvain Thénault committed

class DummySchema(object):

    def __init__(self):
        self._types = {}
Sylvain Thénault's avatar
Sylvain Thénault committed
        for etype in ['String', 'Boolean', 'Int', 'Float', 'Date', 'Datetime',
                      'Eetype', 'Person', 'Company', 'Address', 'Student']:
            self._types[etype] = EntitySchema(etype)
        self._types['Person']._specialized_by = [self._types['Student']]
            'eid' : RelationSchema( ( ('Person', ('Int',) ),
                                      ('Student', ('Int',) ),
                                      ('Company', ('Int',) ),
                                      ('Address', ('Int',) ),
                                      ('Eetype', ('Int',) ),
                                      )
                                    ),
            'creation_date' : RelationSchema( ( ('Person', ('Datetime',) ),
                                                ('Student', ('Datetime',) ),
                                                ('Company', ('Datetime',) ),
                                                ('Address', ('Datetime',) ),
                                                ('Eetype', ('Datetime',) ),
                                                )
                                    ),
            'name' : RelationSchema( ( ('Person', ('String',) ),
                                       ('Student', ('String',) ),
                                       ('Company', ('String',) ),
Nicolas Chauvat's avatar
Nicolas Chauvat committed
                                      )
                                    ),
            'firstname' : RelationSchema( ( ('Person', ('String',) ),
                                            ('Student', ('String',) ),
                                           )
                                    ),
            'work_for' : RelationSchema( ( ('Person', ('Company',) ),
                                           ('Student', ('Company',) ),
            'is' : RelationSchema( ( ('Person', ('Eetype',) ),
                                     ('Student', ('Eetype',) ),
                                     ('Company', ('Eetype',) ),
                                     ('Address', ('Eetype',) ),
                                     ('Eetype', ('Eetype',) ),
Nicolas Chauvat's avatar
Nicolas Chauvat committed
                                     )
                                   ),
            'is_instance_of' : RelationSchema( ( ('Person', ('Eetype',) ),
                                              ('Student', ('Eetype',) ),
                                              ('Company', ('Eetype',) ),
                                              ('Address', ('Eetype',) ),
                                              ('Eetype', ('Eetype',) ),
            'connait' : RelationSchema( (('Person', ('Person',) ),
                                         ('Student', ('Person',) ),
                                         ('Student', ('Student',) ),
                                         ('Person', ('Student',) ),
Sylvain Thénault's avatar
Sylvain Thénault committed
                                        symmetric=True),
            'located' : RelationSchema( ( ('Person', ('Address',) ),
                                          ('Student', ('Address',) ),
                                          ('Company', ('Address',) ),
                                         )
                                       ),
            'owned_by' : RelationSchema( ( ('Person', ('Person',) ),
                                           ('Student', ('Person',) ),
                                           ('Company', ('Person',) ),
                                           ('Eetype', ('Person',) ),
                                           )
                                         ),
            'identity' : RelationSchema( ( ('Person', ('Person',) ),
                                           ('Student', ('Student',) ),
                                           ('Company', ('Company',) ),
                                           ('Address', ('Address',) ),
                                           ('Eetype', ('Eetype',) ),
                                      )
                                    ),
            'number' : RelationSchema( ( ('Person', ('Int',) ),
                                         ('Student', ('Int',) ),
                                         ('Company', ('Float',) ),
                                      )
                                    ),
        for rtype, rschema in self._relations.iteritems():
            rschema.type = rtype
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
    def entities(self):
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        return self._types.values()
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def relations(self):
        return self._relations.keys()

    def has_entity(self, e_type):
        return self._types.has_key(e_type)
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def has_relation(self, r_type):
        return self._relations.has_key(r_type)
Sylvain Thénault's avatar
Sylvain Thénault committed

    def __contains__(self, ertype):
        return self.has_entity(ertype) or self.has_relation(ertype)
Sylvain Thénault's avatar
Sylvain Thénault committed

    def rschema(self, r_type):
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        return self._relations[r_type]
    def eschema(self, e_type):
        return self._types[e_type]
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed
UNRESOLVABLE_QUERIES = (
    'Person X WHERE Y work_for X',
    'Person X WHERE X work_for Y, Y is Address',
    'Insert Company X : X name "toto", X work_for Y WHERE Y name "logilab"',
    )

DEBUG = 0
Sylvain Thenault's avatar
Sylvain Thenault committed
ALL_SOLS = [{'X': 'Address'}, {'X': 'Company'},
            {'X': 'Eetype'}, {'X': 'Person'}, {'X': 'Student'}]
Nicolas Chauvat's avatar
Nicolas Chauvat committed

class AnalyzerClassTest(TestCase):
Nicolas Chauvat's avatar
Nicolas Chauvat committed
    """check wrong queries arre correctly detected
    """
Sylvain's avatar
Sylvain committed
    def _type_from_eid(self, eid):
        return self.eids.get(eid, 'Person')
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def setUp(self):
Sylvain's avatar
Sylvain committed
        self.helper = RQLHelper(DummySchema(), {'eid': self._type_from_eid})
Sylvain Thenault's avatar
Sylvain Thenault committed

    def test_raise(self):
        for rql in UNRESOLVABLE_QUERIES:
            if DEBUG:
                print rql
            node = self.helper.parse(rql)
            self.assertRaises(TypeResolverException,
                              self.helper.compute_solutions, node, debug=DEBUG)
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def test_base_1(self):
        node = self.helper.parse('Any X')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'X': 'Address'},
                                {'X': 'Company'},
                                {'X': 'Eetype'},
                                {'X': 'Person'},
                                {'X': 'Student'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def test_base_2(self):
        node = self.helper.parse('Person X')
Sylvain's avatar
Sylvain committed
        # check constant type of the is relation inserted
        self.assertEqual(node.children[0].where.children[1].children[0].type,
Sylvain's avatar
Sylvain committed
                         'etype')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = node.children[0].solutions
        self.assertEqual(sols, [{'X': 'Person'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

    def test_base_3(self):
        node = self.helper.parse('Any X WHERE X eid 1')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = node.children[0].solutions
        self.assertEqual(sols, [{'X': 'Person'}])
Sylvain's avatar
Sylvain committed
        self.helper.simplify(node)
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = node.children[0].solutions
        self.assertEqual(sols, [{}])
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def test_base_guess_1(self):
        node = self.helper.parse('Person X WHERE X work_for Y')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'X': 'Person', 'Y': 'Company'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def test_base_guess_2(self):
        node = self.helper.parse('Any X WHERE X name "Logilab"')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'X': 'Company'}, {'X': 'Person'}, {'X': 'Student'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

    def test_non_regr_no_final_type(self):
        """https://www.logilab.net/elo/ticket/9042"""
        node = self.helper.parse('Any X WHERE X creation_date > ((2009 - 4) - 16)')
        self.helper.compute_solutions(node, debug=DEBUG)
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'X': 'Address'},
                                {'X': 'Company'},
                                {'X': 'Eetype'},
                                {'X': 'Person'},
                                {'X': 'Student'}])

    def test_is_instance_of_1(self):
        node = self.helper.parse('Any X WHERE X is_instance_of Person')
        # check constant type of the is relation inserted
        self.assertEqual(node.children[0].where.children[1].children[0].type,
                         'etype')
        self.helper.compute_solutions(node, debug=DEBUG)
        sols = node.children[0].solutions
        self.assertEqual(sols, [{'X': 'Person'}, {'X': 'Student'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

    def test_is_instance_of_2(self):
        node = self.helper.parse('Any X WHERE X is_instance_of Student')
        # check constant type of the is relation inserted
        self.assertEqual(node.children[0].where.children[1].children[0].type,
                         'etype')
        self.helper.compute_solutions(node, debug=DEBUG)
        sols = node.children[0].solutions
        self.assertEqual(sols, [{'X': 'Student'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

    def test_is_query(self):
        node = self.helper.parse('Any T WHERE X name "logilab", X is T')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'X': 'Company', 'T': 'Eetype'},
                                {'X': 'Person', 'T': 'Eetype'},
                                {'X': 'Student', 'T': 'Eetype'}])
Nicolas Chauvat's avatar
Nicolas Chauvat committed

    def test_is_query_const(self):
        node = self.helper.parse('Any X WHERE X is T, T eid 10')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'X': 'Address', 'T': 'Eetype'},
                                {'X': 'Company', 'T': 'Eetype'},
Sylvain Thénault's avatar
Sylvain Thénault committed
                                {'X': 'Eetype', 'T': 'Eetype'},
                                {'X': 'Person', 'T': 'Eetype'},
                                {'X': 'Student', 'T': 'Eetype'}])
Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def test_not(self):
        node = self.helper.parse('Any X WHERE NOT X is Person')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        expected = ALL_SOLS[:]
        expected.remove({'X': 'Person'})
Sylvain Thenault's avatar
Sylvain Thenault committed
        self.assertEqual(sols, expected)
Nicolas Chauvat's avatar
Nicolas Chauvat committed

    def test_uid_func_mapping(self):
        h = self.helper
        def type_from_uid(name):
            self.assertEquals(name, "Logilab")
            return 'Company'
        uid_func_mapping = {'name': type_from_uid}
        # constant as rhs of the uid relation
        node = h.parse('Any X WHERE X name "Logilab"')
        h.compute_solutions(node, uid_func_mapping, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEquals(sols, [{'X': 'Company'}])
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        # variable as rhs of the uid relation
        node = h.parse('Any N WHERE X name N')
        h.compute_solutions(node, uid_func_mapping, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEquals(sols, [{'X': 'Company', 'N': 'String'},
                                 {'X': 'Person', 'N': 'String'},
                                 {'X': 'Student', 'N': 'String'}])
Nicolas Chauvat's avatar
Nicolas Chauvat committed
        # substitute as rhs of the uid relation
        node = h.parse('Any X WHERE X name %(company)s')
        h.compute_solutions(node, uid_func_mapping, {'company': 'Logilab'},
Sylvain Thenault's avatar
Sylvain Thenault committed
                        debug=DEBUG)
        sols = sorted(node.children[0].solutions)
        self.assertEquals(sols, [{'X': 'Company'}])
Nicolas Chauvat's avatar
Nicolas Chauvat committed

Sylvain Thénault's avatar
Sylvain Thénault committed
    def test_non_regr_subjobj1(self):
        h = self.helper
        def type_from_uid(name):
            self.assertEquals(name, "Societe")
            return 'Eetype'
        uid_func_mapping = {'name': type_from_uid}
        # constant as rhs of the uid relation
        node = h.parse('Any X WHERE X name "Societe", X is ISOBJ, ISSIBJ is X')
        h.compute_solutions(node, uid_func_mapping, debug=DEBUG)
        sols = sorted(node.children[0].solutions)
        self.assertEquals(sols, [{'X': 'Eetype', 'ISOBJ': 'Eetype', 'ISSIBJ': 'Address'},
                                 {'X': 'Eetype', 'ISOBJ': 'Eetype', 'ISSIBJ': 'Company'},
                                 {'X': 'Eetype', 'ISOBJ': 'Eetype', 'ISSIBJ': 'Eetype'},
                                 {'X': 'Eetype', 'ISOBJ': 'Eetype', 'ISSIBJ': 'Person'},
                                 {'X': 'Eetype', 'ISOBJ': 'Eetype', 'ISSIBJ': 'Student'}])

Sylvain Thénault's avatar
Sylvain Thénault committed
    def test_non_regr_subjobj2(self):
        h = self.helper
        def type_from_uid(name):
            self.assertEquals(name, "Societe")
            return 'Eetype'
        uid_func_mapping = {'name': type_from_uid}
        node = h.parse('Any X WHERE X name "Societe", X is ISOBJ, ISSUBJ is X, X is_instance_of ISIOOBJ, ISIOSUBJ is_instance_of X')
        h.compute_solutions(node, uid_func_mapping, debug=DEBUG)
        select = node.children[0]
        sols = sorted(select.solutions)
        self.assertEquals(len(sols), 25)
        def var_sols(var):
            s = set()
            for sol in sols:
                s.add(sol.get(var))
            return s
        self.assertEquals(var_sols('X'), set(('Eetype',)))
        self.assertEquals(var_sols('X'), select.defined_vars['X'].stinfo['possibletypes'])
        self.assertEquals(var_sols('ISSUBJ'), set(('Address', 'Company', 'Eetype', 'Person', 'Student')))
        self.assertEquals(var_sols('ISSUBJ'), select.defined_vars['ISSUBJ'].stinfo['possibletypes'])
        self.assertEquals(var_sols('ISOBJ'), set(('Eetype',)))
        self.assertEquals(var_sols('ISOBJ'), select.defined_vars['ISOBJ'].stinfo['possibletypes'])
        self.assertEquals(var_sols('ISIOSUBJ'), set(('Address', 'Company', 'Eetype', 'Person', 'Student')))
        self.assertEquals(var_sols('ISIOSUBJ'), select.defined_vars['ISIOSUBJ'].stinfo['possibletypes'])
        self.assertEquals(var_sols('ISIOOBJ'), set(('Eetype',)))
        self.assertEquals(var_sols('ISIOOBJ'), select.defined_vars['ISIOOBJ'].stinfo['possibletypes'])

    def test_unusableuid_func_mapping(self):
        h = self.helper
        def type_from_uid(name):
            self.assertEquals(name, "Logilab")
            return 'Company'
        uid_func_mapping = {'name': type_from_uid}
        node = h.parse('Any X WHERE NOT X name %(company)s')
        h.compute_solutions(node, uid_func_mapping, {'company': 'Logilab'},
                            debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEquals(sols, ALL_SOLS)
        node = h.parse('Any X WHERE X name > %(company)s')
        h.compute_solutions(node, uid_func_mapping, {'company': 'Logilab'},
                            debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEquals(sols, ALL_SOLS)
Sylvain Thénault's avatar
Sylvain Thénault committed


Nicolas Chauvat's avatar
Nicolas Chauvat committed
    def test_base_guess_3(self):
        node = self.helper.parse('Any Z GROUPBY Z WHERE X name Z')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'X': 'Company', 'Z': 'String'},
                                {'X': 'Person', 'Z': 'String'},
                                {'X': 'Student', 'Z': 'String'}])
Nicolas Chauvat's avatar
Nicolas Chauvat committed

    def test_var_name(self):
        node = self.helper.parse('Any E1 GROUPBY E1 WHERE E2 is Person, E2 name E1')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'E2': 'Person', 'E1': 'String'}])
Nicolas Chauvat's avatar
Nicolas Chauvat committed

Sylvain's avatar
Sylvain committed
    def test_relation_eid(self):
        node = self.helper.parse('Any E2 WHERE E2 work_for E1, E2 eid 2')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'E1': 'Company', 'E2': 'Person'}])
Sylvain's avatar
Sylvain committed
        self.helper.simplify(node)
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'E1': 'Company'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
        node = self.helper.parse('Any E1 WHERE E2 work_for E1, E2 eid 2')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'E1': 'Company', 'E2': 'Person'}])
Sylvain's avatar
Sylvain committed
        self.helper.simplify(node)
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'E1': 'Company'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain Thénault's avatar
Sylvain Thénault committed
    def test_not_symmetric_relation_eid(self):
Sylvain's avatar
Sylvain committed
        node = self.helper.parse('Any P WHERE X eid 0, NOT X connait P')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'P': 'Person', 'X': 'Person'},
                                {'P': 'Student', 'X': 'Person'}])
Sylvain's avatar
Sylvain committed
        self.helper.simplify(node)
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'P': 'Person'}, {'P': 'Student'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
    def test_union(self):
        node = self.helper.parse('(Any P WHERE X eid 0, X is Person, NOT X connait P) UNION (Any E1 WHERE E2 work_for E1, E2 eid 2)')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'P': 'Person', 'X': 'Person'}, {'P': 'Student', 'X': 'Person'}])
        sols = sorted(node.children[1].solutions)
        self.assertEqual(sols, [{'E1': 'Company', 'E2': 'Person'}])
Sylvain's avatar
Sylvain committed
        self.helper.simplify(node)
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'P': 'Person'}, {'P': 'Student'}],)
        sols = sorted(node.children[1].solutions)
        self.assertEqual(sols, [{'E1': 'Company'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain's avatar
Sylvain committed
    def test_exists(self):
        node = self.helper.parse("Any X WHERE X firstname 'lulu',"
                                 "EXISTS (X owned_by U, U name 'lulufanclub' OR U name 'managers');")
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'X': 'Person',
                                 'U': 'Person'},
                                {'X': 'Student',
Sylvain Thenault's avatar
Sylvain Thenault committed
                                 'U': 'Person'}])
Nicolas Chauvat's avatar
Nicolas Chauvat committed

        node = self.helper.parse('Any L, Y, F WHERE Y located L '
                                 'WITH Y,F BEING ((Any X,F WHERE X is Person, X firstname F) '
                                 'UNION (Any X,F WHERE X is Company, X name F))')
        self.helper.compute_solutions(node, debug=DEBUG)
        self.assertEqual(node.children[0].with_[0].query.children[0].solutions,
                         [{'X': 'Person', 'F': 'String'}])
        self.assertEqual(node.children[0].with_[0].query.children[1].solutions,
                         [{'X': 'Company', 'F': 'String'}])
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'Y': 'Company', 'L': 'Address',
                                 'F': 'String'},
                                {'Y': 'Person', 'L': 'Address',
                                 'F': 'String'}])

    def test_subqueries_aggregat(self):
        node = self.helper.parse('Any L, SUM(X)*100/Y GROUPBY L '
                                 'WHERE X is Person, X located L '
                                 'WITH Y BEING (Any SUM(X) WHERE X is Person)')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(node.children[0].with_[0].query.children[0].solutions, [{'X': 'Person'}])
Sylvain Thenault's avatar
Sylvain Thenault committed
        self.assertEqual(node.children[0].solutions, [{'X': 'Person', 'Y': 'Person',
                                                       'L': 'Address'}])
Sylvain's avatar
Sylvain committed

    def test_subqueries_outer_filter_type(self):
        # this kind of query may be generated by erudi's facettes box
        node = self.helper.parse('Any L, Y, F WHERE Y located L, Y is Person '
                                 'WITH Y,F BEING ((Any X,F WHERE X is Person, X firstname F) '
                                 'UNION (Any X,F WHERE X is Company, X name F))')
        self.helper.compute_solutions(node, debug=DEBUG)
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'Y': 'Person', 'L': 'Address',
                                 'F': 'String'}])
        self.assertEqual(node.children[0].with_[0].query.children[0].solutions,
                         [{'X': 'Person', 'F': 'String'}])
        # auto-simplification
        self.assertEqual(len(node.children[0].with_[0].query.children), 1)
        self.assertEquals(node.as_string(), 'Any L,Y,F WHERE Y located L, Y is Person WITH Y,F BEING (Any X,F WHERE X is Person, X firstname F)')
        self.assertEqual(node.children[0].with_[0].query.children[0].solutions,
                         [{'X': 'Person', 'F': 'String'}])

Sylvain's avatar
Sylvain committed
    def test_insert(self):
        node = self.helper.parse('INSERT Person X : X name "toto", X work_for Y WHERE Y name "logilab"')
        self.helper.compute_solutions(node, debug=DEBUG)
        sols = sorted(node.solutions)
        self.assertEqual(sols, [{'X': 'Person', 'Y': 'Company'}])

    def test_delete(self):
        node = self.helper.parse('DELETE Person X WHERE X name "toto", X work_for Y')
        self.helper.compute_solutions(node, debug=DEBUG)
        sols = sorted(node.solutions)
        self.assertEqual(sols, [{'X': 'Person', 'Y': 'Company'}])

    def test_set(self):
        node = self.helper.parse('SET X name "toto", X work_for Y WHERE Y name "logilab"')
        self.helper.compute_solutions(node, debug=DEBUG)
        sols = sorted(node.solutions)
        self.assertEqual(sols, [{'X': 'Person', 'Y': 'Company'},
                                {'X': 'Student', 'Y': 'Company'}])
Sylvain's avatar
Sylvain committed

    def test_set_mathexpr(self):
        node = self.helper.parse('SET S number N/4 WHERE P work_for S, P number N')
        self.helper.compute_solutions(node, debug=DEBUG)
        sols = sorted(node.solutions)
        self.assertEqual(sols, [{'P': 'Person', 'S': 'Company', 'N': 'Int'},
                                {'P': 'Student', 'S': 'Company', 'N': 'Int'}])

Sylvain Thénault's avatar
Sylvain Thénault committed

Sylvain Thenault's avatar
Sylvain Thenault committed
    def test_nongrer_not_u_ownedby_u(self):
        node = self.helper.parse('Any U WHERE NOT U owned_by U')
        self.helper.compute_solutions(node, debug=DEBUG)
Sylvain Thenault's avatar
Sylvain Thenault committed
        sols = sorted(node.children[0].solutions)
        self.assertEqual(sols, [{'U': 'Person'}])
Sylvain Thénault's avatar
Sylvain Thénault committed

Nicolas Chauvat's avatar
Nicolas Chauvat committed

if __name__ == '__main__':
    unittest_main()