forked from Yaltik/golem
227 lines
12 KiB
Python
227 lines
12 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright 2018 Youssef El Ouahby <youssef@yaltik.com>
|
|
# Copyright 2018 Fabien Bourgeois <fabien@yaltik.com>
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Affero General Public License as
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
# License, or (at your option) any later version.
|
|
#
|
|
# This program 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 Affero General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
""" GOLEM Resource Reservation """
|
|
|
|
from math import modf
|
|
from datetime import timedelta
|
|
from odoo import models, fields, api, _
|
|
from odoo.exceptions import ValidationError
|
|
|
|
|
|
class GolemResourceReservation(models.Model):
|
|
""" GOLEM Resource Reservation Model """
|
|
_name = 'golem.resource.reservation'
|
|
_description = 'GOLEM Reservation Model'
|
|
_inherit = 'mail.thread'
|
|
_order = 'day_start desc, hour_start asc'
|
|
|
|
name = fields.Char(compute='_compute_name', store=True)
|
|
# TODO: handle multiple days reservation
|
|
date_start = fields.Datetime('Start date', required=True,
|
|
index=True, readonly=True,
|
|
states={'draft': [('readonly', False)]})
|
|
date_stop = fields.Datetime('Stop date', required=True,
|
|
index=True, readonly=True,
|
|
states={'draft': [('readonly', False)]})
|
|
day_start = fields.Date(compute='_compute_day_hour_start', store=True)
|
|
hour_start = fields.Float(compute='_compute_day_hour_start', store=True)
|
|
resource_id = fields.Many2one('golem.resource', required=True, index=True,
|
|
string='Resource', readonly=True,
|
|
track_visibility='onchange',
|
|
states={'draft': [('readonly', False)]})
|
|
resource_avaibility_start = fields.Date(related='resource_id.avaibility_start')
|
|
resource_avaibility_stop = fields.Date(related='resource_id.avaibility_stop')
|
|
resource_avaibility_24_7 = fields.Boolean(related='resource_id.availibility_24_7')
|
|
resource_timetable_ids = fields.One2many(related='resource_id.timetable_ids')
|
|
|
|
note = fields.Text(help='Notes, optional subject for the reservation, reason')
|
|
|
|
user_id = fields.Many2one('res.users', required=True, index=True, readonly=True,
|
|
string='User', default=lambda self: self.env.user,
|
|
states={'draft': [('readonly', False)]})
|
|
partner_id = fields.Many2one('res.partner', string='On behalf of',
|
|
required=True, index=True, readonly=True,
|
|
track_visibility='onchange',
|
|
states={'draft': [('readonly', False)]})
|
|
state = fields.Selection([('canceled', 'Canceled'),
|
|
('draft', 'Draft'),
|
|
('confirmed', 'Confirmed'),
|
|
('validated', 'Validated'),
|
|
('rejected', 'Rejected')],
|
|
default='draft', track_visibility='onchange')
|
|
|
|
rejection_reason = fields.Text(readonly=True, track_visibility='onchange')
|
|
|
|
@api.depends('resource_id', 'date_start')
|
|
def _compute_name(self):
|
|
""" Computes reservation name """
|
|
for reservation in self:
|
|
reservation.name = u'{}/{}'.format(reservation.resource_id.name,
|
|
reservation.date_start)
|
|
|
|
@api.depends('date_start')
|
|
def _compute_day_hour_start(self):
|
|
""" Computes Day and Hour Start : for better sorting """
|
|
for reservation in self:
|
|
if reservation.date_start:
|
|
date_start = fields.Datetime.from_string(reservation.date_start)
|
|
reservation.day_start = date_start.date().isoformat()
|
|
reservation.hour_start = date_start.hour + date_start.minute / 60.0
|
|
|
|
@api.onchange('date_start')
|
|
def onchange_date_start(self):
|
|
""" Propose automatically stop hour after start hour had been filled """
|
|
for reservation in self:
|
|
if reservation.date_start:
|
|
start = fields.Datetime.from_string(reservation.date_start)
|
|
duration = timedelta(hours=1)
|
|
reservation.date_stop = start + duration
|
|
|
|
|
|
@api.constrains('date_start', 'date_stop')
|
|
def _check_date_consistency(self):
|
|
""" Checks date consistency """
|
|
for reservation in self:
|
|
if reservation.date_stop <= reservation.date_start:
|
|
raise ValidationError(_('Stop date should be after start date'))
|
|
|
|
@api.multi
|
|
def state_draft(self):
|
|
""" Status to draft """
|
|
self.write({'state': 'draft'})
|
|
|
|
@api.multi
|
|
def state_confirm(self):
|
|
""" Confirms reservation, or validates it if not workflow is involved """
|
|
for reservation in self:
|
|
# Needed, for constraint checking
|
|
reservation.state = 'confirmed'
|
|
if not reservation.resource_id.validation_required:
|
|
reservation.state = 'validated'
|
|
|
|
|
|
@api.multi
|
|
def state_canceled(self):
|
|
""" Status to cancel """
|
|
self.write({'state': 'canceled'})
|
|
|
|
@api.multi
|
|
def state_validated(self):
|
|
""" Status to validated """
|
|
self.write({'state': 'validated'})
|
|
|
|
@api.multi
|
|
def state_rejected(self):
|
|
""" Wizard call for reservation reject """
|
|
self.ensure_one()
|
|
reservation_id = self[0]
|
|
return {'name' : _('Please enter the rejection reason'),
|
|
'type' : 'ir.actions.act_window',
|
|
'res_model' : 'golem.reservation.rejection.wizard',
|
|
'context': {'default_reservation_id': reservation_id.id},
|
|
'view_mode': 'form',
|
|
'target': 'new'}
|
|
|
|
|
|
@api.constrains('state')
|
|
def check_access(self):
|
|
""" Checks access when state is updated """
|
|
reservation = self[0]
|
|
if reservation.state in ('rejected', 'validated'):
|
|
if not self.env.user.has_group('golem_base.group_golem_manager'):
|
|
verr = _('You do not have permissions to validate or reject a reservation.')
|
|
raise ValidationError(verr)
|
|
|
|
@api.constrains('state')
|
|
def check_confirmed(self):
|
|
""" Check date coherence on reservation confirmation """
|
|
for reservation in self:
|
|
if reservation.state == 'confirmed':
|
|
# Check is reservation is not taking place out of the resource avaibility period
|
|
if reservation.date_start < reservation.resource_id.avaibility_start or \
|
|
reservation.date_stop > reservation.resource_id.avaibility_stop:
|
|
verr = _('Not allowed, the resource is not available in '
|
|
'this period, please choose another périod before '
|
|
'confirming')
|
|
raise ValidationError(verr)
|
|
#check if the resource hasn't a total availibility
|
|
if not reservation.resource_id.availibility_24_7:
|
|
# Check if reservation is not taking place out the avaibility timetables
|
|
date_start = fields.Datetime.from_string(reservation.date_start)
|
|
date_stop = fields.Datetime.from_string(reservation.date_stop)
|
|
reservation_period = [date_start + timedelta(days=x) for x in range(
|
|
(date_stop - date_start).days + 1)]
|
|
for reservation_day in reservation_period:
|
|
is_day_allowed = False
|
|
for timetable in reservation.resource_id.timetable_ids:
|
|
# Check for the time according to resource timetable avaibility
|
|
#date = fields.Datetime.from_string(reservation_day)
|
|
if int(timetable.weekday) == reservation_day.weekday():
|
|
is_day_allowed = True
|
|
#only check if the day hasn't a 24 availibility
|
|
if not timetable.availibility_24:
|
|
reservation_day_date = reservation_day.date()
|
|
day_start = date_start.date()
|
|
day_stop = date_stop.date()
|
|
if reservation_day_date == day_start and \
|
|
reservation_day_date == day_stop:
|
|
hour_start = date_start.hour + date_start.minute / 60.0
|
|
hour_stop = date_stop.hour + date_stop.minute / 60.0
|
|
elif reservation_day_date == day_start:
|
|
hour_start = date_start.hour + date_start.minute / 60.0
|
|
hour_stop = 23.98 # Just before 23:59
|
|
elif reservation_day_date == day_stop:
|
|
hour_start = 0.0
|
|
hour_stop = date_stop.hour + date_stop.minute / 60.0
|
|
else:
|
|
#if the day is not a start nor stop it
|
|
#should be covered on all day
|
|
#strange, as availibility_24 is not True
|
|
hour_start = 0.0
|
|
hour_stop = 23.98
|
|
|
|
if is_day_allowed and (hour_start < timetable.time_start or \
|
|
hour_stop > timetable.time_stop):
|
|
verr = _('Not allowed, the resource is not available '
|
|
'during this period, please choose another '
|
|
'time before confirming.')
|
|
raise ValidationError(verr)
|
|
if not is_day_allowed:
|
|
verr = _('Not allowed, the resource is not available '
|
|
'this day : {}. Please choose another '
|
|
'date.'.format(reservation_day.strftime('%A')))
|
|
raise ValidationError(verr)
|
|
# Check if the resource is already taken during this period
|
|
# PERF : check for res that can be in conflict,
|
|
# do not iterate over all reservations
|
|
domain = [('resource_id', '=', reservation.resource_id.id),
|
|
('date_start', '<=', reservation.date_stop),
|
|
('date_stop', '>=', reservation.date_start),
|
|
('state', 'in', ('confirmed', 'validated')),
|
|
('id', '!=', reservation.id)]
|
|
reservations = self.env['golem.resource.reservation'].search(domain)
|
|
for other_res in reservations:
|
|
if (other_res.date_start < reservation.date_start < other_res.date_stop) or \
|
|
(other_res.date_start < reservation.date_stop < other_res.date_stop):
|
|
verr = _('Not allowed, the resource is already taken '
|
|
'during this period : from {} to {} this day, '
|
|
'please choose another périod before confirming.')
|
|
raise ValidationError(verr.format(other_res.date_start,
|
|
other_res.date_stop))
|