Skip to content
housing.py 5.19 KiB
Newer Older
Romain Bignon's avatar
Romain Bignon committed
# -*- coding: utf-8 -*-

# Copyright(C) 2012 Romain Bignon
#
# This file is part of weboob.
#
# weboob is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
Romain Bignon's avatar
Romain Bignon committed
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# weboob 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
Romain Bignon's avatar
Romain Bignon committed
# along with weboob. If not, see <http://www.gnu.org/licenses/>.


Florent's avatar
Florent committed
from .base import Capability, BaseObject, Field, IntField, DecimalField, \
                  StringField, BytesField, Enum, EnumField, UserError
from .date import DateField
__all__ = [
    'CapHousing', 'Housing', 'Query', 'City', 'UTILITIES', 'ENERGY_CLASS',
    'POSTS_TYPES', 'ADVERT_TYPES', 'HOUSE_TYPES', 'TypeNotSupported',
    'HousingPhoto',
]
class TypeNotSupported(UserError):
    """
    Raised when query type is not supported
    """

    def __init__(self,
                 msg='This type of house is not supported by this module'):
        super(TypeNotSupported, self).__init__(msg)
class HousingPhoto(BaseObject):
    """
    Photo of a housing.
    """
    data =      BytesField('Data of photo')

    def __init__(self, url):
        super(HousingPhoto, self).__init__(url.split('/')[-1], url)
Romain Bignon's avatar
Romain Bignon committed

    def __iscomplete__(self):
        return self.data

    def __unicode__(self):
Romain Bignon's avatar
Romain Bignon committed
        return self.url

    def __repr__(self):
        return '<HousingPhoto %r data=%do url=%r>' % (self.id, len(self.data) if self.data else 0, self.url)
Romain Bignon's avatar
Romain Bignon committed

    INCLUDED = u'C.C.'
    EXCLUDED = u'H.C.'
    UNKNOWN = u''
class ENERGY_CLASS(Enum):
    A = u'A'
    B = u'B'
    C = u'C'
    D = u'D'
    E = u'E'
    F = u'F'
    G = u'G'


class POSTS_TYPES(Enum):
    RENT=u'RENT'
    SALE = u'SALE'
    SHARING = u'SHARING'
    FURNISHED_RENT = u'FURNISHED_RENT'
    VIAGER = u'VIAGER'


class ADVERT_TYPES(Enum):
    PROFESSIONAL = u'Professional'
    PERSONAL = u'Personal'


class HOUSE_TYPES(Enum):
    APART = u'Apartment'
    HOUSE = u'House'
    PARKING = u'Parking'
    LAND = u'Land'
    OTHER = u'Other'
    UNKNOWN = u'Unknown'
class Housing(BaseObject):
    type = EnumField('Type of housing (rent, sale, sharing)',
                     POSTS_TYPES)
    advert_type = EnumField('Type of advert (professional or personal)',
                            ADVERT_TYPES)
    house_type = EnumField(u'Type of house (apartment, house, parking, …)',
                           HOUSE_TYPES)
Phyks (Lucas Verney)'s avatar
Phyks (Lucas Verney) committed
    title = StringField('Title of housing')
    area = DecimalField('Area of housing, in m2')
    cost = DecimalField('Cost of housing')
    price_per_meter = DecimalField('Price per meter ratio')
Phyks (Lucas Verney)'s avatar
Phyks (Lucas Verney) committed
    currency = StringField('Currency of cost')
    utilities = EnumField('Utilities included or not', UTILITIES)
Phyks (Lucas Verney)'s avatar
Phyks (Lucas Verney) committed
    date = DateField('Date when the housing has been published')
    location = StringField('Location of housing')
    station = StringField('What metro/bus station next to housing')
    text = StringField('Text of the housing')
    phone = StringField('Phone number to contact')
    photos = Field('List of photos', list)
    rooms = DecimalField('Number of rooms')
    bedrooms = DecimalField('Number of bedrooms')
    details = Field('Key/values of details', dict)
    DPE = EnumField('DPE (Energy Performance Certificate)', ENERGY_CLASS)
    GES = EnumField('GES (Greenhouse Gas Emissions)', ENERGY_CLASS)
Bezleputh's avatar
Bezleputh committed

class Query(BaseObject):
    type = EnumField('Type of housing to find (POSTS_TYPES constants)',
                     POSTS_TYPES)
    cities = Field('List of cities to search in', list, tuple)
    area_min = IntField('Minimal area (in m2)')
    area_max = IntField('Maximal area (in m2)')
    cost_min = IntField('Minimal cost')
    cost_max = IntField('Maximal cost')
    nb_rooms = IntField('Number of rooms')
    house_types = Field('List of house types', list, tuple, default=list(HOUSE_TYPES))
    advert_types = Field('List of advert types to filter on', list, tuple,
class City(BaseObject):
Phyks (Lucas Verney)'s avatar
Phyks (Lucas Verney) committed
    name = StringField('Name of city')
Florent's avatar
Florent committed
class CapHousing(Capability):
    """
    Capability of websites to search housings.
    """
Romain Bignon's avatar
Romain Bignon committed
    def search_housings(self, query):
        """
        Search housings.

        :param query: search query
        :type query: :class:`Query`
        :rtype: iter[:class:`Housing`]
        """
Romain Bignon's avatar
Romain Bignon committed
        raise NotImplementedError()

    def get_housing(self, housing):
        """
        Get an housing from an ID.

        :param housing: ID of the housing
        :type housing: str
        :rtype: :class:`Housing` or None if not found.
        """
Romain Bignon's avatar
Romain Bignon committed
        raise NotImplementedError()

    def search_city(self, pattern):
        """
        Search a city from a pattern.

        :param pattern: pattern to search
        :type pattern: str
        :rtype: iter[:class:`City`]
        """
Romain Bignon's avatar
Romain Bignon committed
        raise NotImplementedError()