Example #1
0
def planning_view(request):
    pools = list(Pool.all())
    poolid2idx = dict()
    i = 0
    for pool in pools:
        poolid2idx[pool._id] = i
        i += 1
    events = list()
    # TODO reduce number of queries
    for e in Event.all_in_future():
        ei = {  'name': e.name,
                'date': str(e.date.date()),
                'kind': e.kind,
            'vacancies': dict()}
        for idx in poolid2idx.values():
            ei['vacancies'][idx] = list()
        for v in e.vacancies():
            ei['vacancies'][poolid2idx[v.pool_id]].append({
                'begin': v.begin,
                'begin_time': v.begin_time,
                'end_time': v.end_time,
                'assignee': v.assignee.get_user().humanName
                        if v.assignee else "?"})
        for idx in poolid2idx.values():
            ei['vacancies'][idx].sort(key=lambda x: x['begin'])
        events.append(ei)
    events.sort(key=lambda x: x['date'])
    return render_to_response('planning/overview.html',
            {'events': events,
             'pools': pools,
             'poolcount': len(pools)},
            context_instance=RequestContext(request))
Example #2
0
def planning_template(request, poolname):
    pool = Pool.by_name(poolname)
    events = list()
    # TODO reduce number of queries
    for e in Event.all_in_future():
        vacancies = list(e.vacancies(pool))
        if not vacancies:
            continue
        ei = {'name': e.name, 'date': e.date, 'vacancies': list()}
        shifts = dict()
        for v in vacancies:
            if v.begin not in shifts:
                shifts[v.begin] = {
                    'name': v.name,
                    'begin': v.begin,
                    'begin_time': v.begin_time,
                    'end_time': v.end_time,
                    'assignees': list()
                }
            shifts[v.begin]['assignees'].append(v.assignee)
        ei['vacancies'] = [
            x[1] for x in sorted(shifts.items(), key=lambda x: x[0])
        ]
        events.append(ei)
    events.sort(key=lambda x: x['date'])
    return render(request, 'planning/template.html', {'events': events})
Example #3
0
def planning_view(request):
    period = 'now'
    if request.GET.get('year') in {'past1', 'past2'}:
        period = request.GET['year']
    pools = list(Pool.all())
    poolid2index = dict()
    poolids = set()
    for pool in pools:
        poolids.add(_id(pool))
    # TODO reduce number of queries
    current = date_to_midnight(now() - datetime.timedelta(days=1))
    past1year = date_to_midnight(now() - datetime.timedelta(days=356))
    past2year = date_to_midnight(now() - datetime.timedelta(days=356*2))
    if period == 'now':
        start = current
        end = None
    elif period == 'past1':
        start = past1year
        end = current
    elif period == 'past2':
        start = past2year
        end = past1year
    event_entities = list(Event.all_in_period(start, end))
    used_pools = set()
    for e in event_entities:
        for v in e.vacancies():
            used_pools.add(v.pool_id)
    pools_tpl = []
    i = 0
    for pool in pools:
        if _id(pool) not in used_pools:
            continue
        poolid2index[pool._id] = i
        pools_tpl.append(pool)
        i += 1
    events = list()
    for e in event_entities:
        ei = {'id': _id(e),
              'name': e.name,
              'datetime': e.date,
              'kind': e.kind,
              'vacancies': dict()}
        for index in poolid2index.values():
            ei['vacancies'][index] = list()
        for v in e.vacancies():
            ei['vacancies'][poolid2index[v.pool_id]].append({
                'begin': v.begin,
                'begin_time': v.begin_time,
                'end_time': v.end_time,
                'assignee': v.assignee.humanName
                if v.assignee else "?"})
        for index in poolid2index.values():
            ei['vacancies'][index].sort(key=lambda x: x['begin'])
        events.append(ei)
    events.sort(key=lambda x: x['datetime'])
    return render_to_response('planning/overview.html',
                              {'events': events,
                               'pools': pools_tpl,
                               'period': period},
                              context_instance=RequestContext(request))
Example #4
0
def event_create(request):
    if not may_manage_planning(request.user):
        raise PermissionDenied
    if request.method == 'POST':
        form = EventCreateForm(request.POST)
        if form.is_valid():
            fd = form.cleaned_data
            day = date_to_dt(fd['date'])
            e = Event({
                'name': fd['name'],
                'date': day,
                'kind': fd['template']
            })
            e.save()
            for poolname, periods in templates[fd['template']].items():
                pool = Pool.by_name(poolname)
                for period in periods:
                    begin_date = day + datetime.timedelta(seconds=period[0][0])
                    end_date = day + datetime.timedelta(seconds=period[1][0])
                    v = Vacancy({
                        'name': period[2],
                        'event': _id(e),
                        'begin': (begin_date, period[0][1]),
                        'end': (end_date, period[1][1]),
                        'pool': _id(pool),
                        'assignee': None,
                        'reminder_needed': True,
                    })
                    v.save()
            return HttpResponseRedirect(
                reverse('planning-event-edit', args=(e._id, )))
    else:
        form = EventCreateForm()
    return render(request, 'planning/event_create.html', {'form': form})
Example #5
0
def planning_template(request, poolname):
    pool = Pool.by_name(poolname)
    events = list()
    # TODO reduce number of queries
    for e in Event.all_in_future():
        vacancies = list(e.vacancies(pool))
        if not vacancies:
            continue
        ei = {  'name': e.name,
                'date': e.date,
            'vacancies': list()}
        shifts = dict()
        for v in vacancies:
            if not v.begin in shifts:
                shifts[v.begin] = {
                    'name': v.name,
                    'begin': v.begin,
                    'begin_time': v.begin_time,
                    'end_time': v.end_time,
                    'assignees': list()}
            shifts[v.begin]['assignees'].append(v.assignee)
        ei['vacancies'] = map(lambda x: x[1], sorted(shifts.items(),
            key=lambda x: x[0]))
        events.append(ei)
    events.sort(key=lambda x: x['date'])
    return render_to_response('planning/template.html', {'events': events},
            context_instance=RequestContext(request))
Example #6
0
def event_create(request):
    if request.method == 'POST':
        form = EventCreateForm(request.POST)
        if form.is_valid():
            fd = form.cleaned_data
            day = date_to_dt(fd['date'])
            e = Event({
                'name': fd['name'],
                'date': day,
                'kind': fd['template']})
            e.save()
            for poolname, periods in templates[fd['template']].items():
                pool = Pool.by_name(poolname)
                for period in periods:
                    begin_date = day + datetime.timedelta(seconds=period[0][0])
                    end_date = day + datetime.timedelta(seconds=period[1][0])
                    v = Vacancy({
                        'name': period[2],
                        'event': _id(e),
                        'begin': (begin_date, period[0][1]),
                        'end': (end_date, period[1][1]) ,
                        'pool': _id(pool),
                        'assignee': None,
                        'reminder_needed': True,
                    })
                    v.save()
            return HttpResponseRedirect(reverse('planning-event-edit',
                args=(e._id,)))
    else:
        form = EventCreateForm()
    return render_to_response('planning/event_create.html', {'form': form},
            context_instance=RequestContext(request))
Example #7
0
def planning_poollist(request):
    if not may_manage_planning(request.user):
        # There's no planning you can change anyway, so what are you doing
        # here?
        raise PermissionDenied
    pools = list(Pool.all())
    return render(request, 'planning/pools.html', {'pools': pools})
Example #8
0
def planning_template(request, poolname):
    locale.setlocale(locale.LC_ALL, 'nl_NL')
    pool = Pool.by_name(poolname)
    events = list()
    # TODO reduce number of queries
    for e in Event.all_in_future():
        vacancies = list(e.vacancies(pool))
        if not vacancies:
            continue
        ei = {  'name': e.name,
                'date': str(e.date.date()),
            'vacancies': list()}
        ei['description'] = e.date.strftime('%A %d %B')
        if e.name != 'Borrel':
            ei['description'] = '%s (%s)' % (ei['description'], ei['name'])
        shifts = dict()
        for v in vacancies:
            if not v.begin in shifts:
                shifts[v.begin] = {
                    'name': v.name,
                    'begin': v.begin,
                    'begin_time': v.begin_time,
                    'end_time': v.end_time,
                    'assignees': list()}
            shifts[v.begin]['assignees'].append(v.assignee)
        ei['vacancies'] = map(lambda x: x[1], sorted(shifts.items(),
            key=lambda x: x[0]))
        events.append(ei)
    events.sort(key=lambda x: x['date'])
    return render_to_response('planning/template.html', {'events': events},
            context_instance=RequestContext(request))
Example #9
0
def planning_poollist(request):
    if not may_manage_planning(request.user):
        # There's no planning you can change anyway, so what are you doing here?
        raise PermissionDenied
    pools = list(Pool.all())
    return render_to_response('planning/pools.html',
            {'pools': pools},
            context_instance=RequestContext(request))
Example #10
0
def planning_poollist(request):
    if not may_manage_planning(request.user):
        # There's no planning you can change anyway, so what are you doing
        # here?
        raise PermissionDenied
    pools = list(Pool.all())
    return render(request, 'planning/pools.html',
                  {'pools': pools})
Example #11
0
def event_edit(request, eventid):
    if not may_manage_planning(request.user):
        raise PermissionDenied
    avform = None
    e = Event.by_id(eventid)
    if e is None:
        raise Http404
    if request.method == 'POST':
        if request.POST['action'] == 'remove_event':
            for vacancy in e.vacancies():
                vacancy.delete()
            e.delete()
            return HttpResponseRedirect(reverse('planning-poollist'))
        elif request.POST['action'] == 'remove_vacancy':
            Vacancy.by_id(_id(request.POST['vacancy_id'])).delete()
            return HttpResponseRedirect(reverse('planning-event-edit',
                                                args=(eventid,)))
        elif request.POST['action'] == 'add_vacancy':
            avform = AddVacancyForm(request.POST)
            if avform.is_valid():
                fd = avform.cleaned_data
                (begin_hour, begin_minute) = map(int, fd['begin'].split(':'))
                (end_hour, end_minute) = map(int, fd['end'].split(':'))
                begin_offset = hm2s(begin_hour, begin_minute)
                end_offset = hm2s(end_hour, end_minute)
                begin_date = e.date + datetime.timedelta(seconds=begin_offset)
                end_date = e.date + datetime.timedelta(seconds=end_offset)
                v = Vacancy({
                    'name': fd['name'],
                    'event': _id(e),
                    'begin': (begin_date,
                              fd['begin_is_approximate'] == "True"),
                    'end': (end_date, fd['end_is_approximate'] == "True"),
                    'pool': _id(fd['pool']),
                    'assignee': None,
                    'reminder_needed': True,
                })
                v.save()
                return HttpResponseRedirect(reverse('planning-event-edit',
                                                    args=(eventid,)))
    if avform is None:
        avform = AddVacancyForm()
    pools = dict()
    for p in Pool.all():
        pools[_id(p)] = p
    vacancies = list()
    for v in e.vacancies():
        v.poolname = pools[v.pool_id].name
        v.assignee_text = str(v.assignee.name) if v.assignee else "-"
        v.vid = str(v._id)
        vacancies.append(v)
    vacancies.sort(key=lambda x: str(x.pool_id) + str(x.begin))
    return render(
        request,
        'planning/event_edit.html',
        {'name': e.name, 'kind': e.kind, 'date': e.date.date(),
         'avform': avform, 'vacancies': vacancies},
    )
Example #12
0
def planning_view(request):
    if 'lookbehind' in request.GET:
        lookbehind = int(request.GET['lookbehind'])
    else:
        lookbehind = 1
    pools = list(Pool.all())
    poolid2index = dict()
    poolids = set()
    for pool in pools:
        poolids.add(_id(pool))
    # TODO reduce number of queries
    event_entities = list(Event.all_since_datetime(date_to_midnight(now())
            - datetime.timedelta(days=lookbehind)))
    used_pools = set()
    for e in event_entities:
        for v in e.vacancies():
            used_pools.add(v.pool_id)
    pools_tpl = []
    i = 0
    for pool in pools:
        if _id(pool) not in used_pools:
            continue
        poolid2index[pool._id] = i
        pools_tpl.append(pool)
        i += 1
    events = list()
    for e in event_entities:
        ei = {  'id': _id(e),
                'name': e.name,
                'datetime': e.date,
                'kind': e.kind,
            'vacancies': dict()}
        for index in poolid2index.values():
            ei['vacancies'][index] = list()
        for v in e.vacancies():
            ei['vacancies'][poolid2index[v.pool_id]].append({
                'begin': v.begin,
                'begin_time': v.begin_time,
                'end_time': v.end_time,
                'assignee': v.assignee.humanName
                        if v.assignee else "?"})
        for index in poolid2index.values():
            ei['vacancies'][index].sort(key=lambda x: x['begin'])
        events.append(ei)
    events.sort(key=lambda x: x['datetime'])
    return render_to_response('planning/overview.html',
            {'events': events,
             'pools': pools_tpl},
            context_instance=RequestContext(request))
Example #13
0
class AddVacancyForm(forms.Form):
    name = forms.CharField(label="Shiftnaam", initial="eerste dienst")
    begin = forms.RegexField(label="Begintijd",
                             initial="20:30",
                             regex=r'^[0123][0-9]:[0-5][0-9]$')
    begin_is_approximate = forms.ChoiceField(initial=False,
                                             choices=((True, "bij benadering"),
                                                      (False, "exact")))
    end = forms.RegexField(label="Eindtijd",
                           initial="23:00",
                           regex=r'^[0123][0-9]:[0-5][0-9]$')
    end_is_approximate = forms.ChoiceField(initial=False,
                                           choices=((True, "bij benadering"),
                                                    (False, "exact")))
    pool = forms.ChoiceField(label="Type",
                             choices=map(lambda x: (x._id, x.name),
                                         Pool.all()))
Example #14
0
def planning_view(request):
    if 'lookbehind' in request.GET:
        lookbehind = int(request.GET['lookbehind'])
    else:
        lookbehind = 1
    pools = list(Pool.all())
    poolid2idx = dict()
    i = 0
    for pool in pools:
        poolid2idx[pool._id] = i
        i += 1
    events = list()
    # TODO reduce number of queries
    for e in Event.all_since_datetime(
            date_to_midnight(now()) - datetime.timedelta(days=lookbehind)):
        ei = {
            'name': e.name,
            'datetime': e.date,
            'kind': e.kind,
            'vacancies': dict()
        }
        for idx in poolid2idx.values():
            ei['vacancies'][idx] = list()
        for v in e.vacancies():
            ei['vacancies'][poolid2idx[v.pool_id]].append({
                'begin':
                v.begin,
                'begin_time':
                v.begin_time,
                'end_time':
                v.end_time,
                'assignee':
                v.assignee.get_user().humanName if v.assignee else "?"
            })
        for idx in poolid2idx.values():
            ei['vacancies'][idx].sort(key=lambda x: x['begin'])
        events.append(ei)
    events.sort(key=lambda x: x['datetime'])
    return render_to_response('planning/overview.html', {
        'events': events,
        'pools': pools,
        'poolcount': len(pools)
    },
                              context_instance=RequestContext(request))
Example #15
0
def planning_view(request):
    if 'lookbehind' in request.GET:
        lookbehind = int(request.GET['lookbehind'])
    else:
        lookbehind = 1
    pools = list(Pool.all())
    poolid2idx = dict()
    i = 0
    for pool in pools:
        poolid2idx[pool._id] = i
        i += 1
    events = list()
    # TODO reduce number of queries
    for e in Event.all_since_datetime(date_to_midnight(now())
            - datetime.timedelta(days=lookbehind)):
        ei = {  'name': e.name,
                'datetime': e.date,
                'kind': e.kind,
            'vacancies': dict()}
        for idx in poolid2idx.values():
            ei['vacancies'][idx] = list()
        for v in e.vacancies():
            ei['vacancies'][poolid2idx[v.pool_id]].append({
                'begin': v.begin,
                'begin_time': v.begin_time,
                'end_time': v.end_time,
                'assignee': v.assignee.humanName
                        if v.assignee else "?"})
        for idx in poolid2idx.values():
            ei['vacancies'][idx].sort(key=lambda x: x['begin'])
        events.append(ei)
    events.sort(key=lambda x: x['datetime'])
    return render_to_response('planning/overview.html',
            {'events': events,
             'pools': pools,
             'poolcount': len(pools)},
            context_instance=RequestContext(request))
Example #16
0
# vim: et:sta:bs=2:sw=4:
import _import
import datetime
import sys

import kn.leden.entities as Es
from kn.settings import DT_MIN, DT_MAX
from kn.leden.mongo import _id
from kn.leden.date import now
from kn.planning.entities import Pool, Worker, Vacancy

pools = dict()
for p in Pool.all():
    pools[p.name] = p

workers = dict()
for w in Worker.all():
    workers[w.get_user()] = w

dt = now()
for w in workers.itervalues():
    del w.pools[:]
for type in ["tappers", "bestuur", "barco", "draai", "cocks"]:
    poolid = _id(pools[type])
    group = Es.by_name(type)
    relations = group.get_rrelated(None, dt, dt, True, None, None)
    for gm in relations:
        if gm["who"] in workers:
            if poolid not in workers[gm["who"]].pools:
                print "%s -> %s" % (gm["who"].name, type)
                workers[gm["who"]].pools.append(poolid)
Example #17
0
import sys

from kn.leden.mongo import _id
from kn.planning.entities import Event, Pool, Vacancy


def hm2s(hours, minutes=0):
    return (hours * 60 + minutes) * 60


begin = datetime.datetime.strptime('%s %s' % (sys.argv[1], sys.argv[2]),
                                   '%Y-%m-%d %H:%M')

e = Event({
    'name': 'Dranktelling',
    'date': datetime.datetime.combine(begin.date(), datetime.time())})
e.save()

pool = Pool.by_name('barco')

for p in [1, 2]:
    v = Vacancy({
        'name': 'Teller %d' % p,
        'event': _id(e),
        'begin': begin,
        'end': begin + datetime.timedelta(seconds=1800),
        'pool': _id(pool),
        'assignee': None})
    print(v._data)
    v.save()
Example #18
0
def planning_manage(request, poolname):
    if not may_manage_planning(request.user):
        raise PermissionDenied
    pool = Pool.by_name(poolname)
    if pool is None:
        raise Http404
    if not pool.may_manage(request.user):
        raise PermissionDenied
    # TODO reduce number of queries
    events = dict()
    for e in Event.all_in_future():
        eid = _id(e)
        vacancies = list(e.vacancies(pool=pool))
        events[eid] = {
            'vacancies': vacancies,
            'date': e.date.date(),
            'name': e.name,
            'kind': e.kind,
            'id': eid
        }
        posted = False
        events[eid]['vacancies'].sort(key=lambda v: v.begin)
        if request.method == 'POST' and _id(request.POST['eid']) == eid:
            events[eid]['form'] = ManagePlanningForm(
                request.POST, pool=pool, vacancies=events[eid]['vacancies'])
            posted = True
        else:
            events[eid]['form'] = ManagePlanningForm(
                pool=pool, vacancies=events[eid]['vacancies'])
        if posted and events[eid]['form'].is_valid():
            for vacancy in events[eid]['vacancies']:
                worker = request.POST['shift_%s' % vacancy._id]
                if worker == '':
                    vacancy.assignee = None
                    vacancy.reminder_needed = True
                else:
                    if vacancy.assignee_id is None or \
                            _id(vacancy.assignee_id) != _id(worker):
                        delta = datetime.timedelta(days=5)
                        vacancy.reminder_needed = now() + delta < e.date
                        vacancy.assignee_id = _id(worker)
                vacancy.save()
    workers = pool.workers()
    # XXX het is cooler de shift dichtstbijzijnd aan de vacancy te
    # zoeken.  Stel dat iemand over een half-jaar al is ingepland
    # dan is dat niet zo boeiend.  Terwijl hij nu geen enkele
    # bardienst meer zou krijgen
    shifts = pool.last_shifts()
    for eid in events:
        for vacancy in events[eid]['vacancies']:
            vacancy.suggestions = list()
            workers_by_score = dict()
            for worker in workers:
                score = planning_vacancy_worker_score(vacancy, worker)
                if score not in workers_by_score:
                    workers_by_score[score] = list()
                workers_by_score[score].append(worker)
            found_scores = list(workers_by_score.keys())
            found_scores.sort(reverse=True)
            for score in found_scores:
                scorers = workers_by_score[score]
                shuffle(scorers)
                scorers.sort(key=lambda x: shifts[_id(x)]
                             if shifts[_id(x)] else DT_MIN.date())
                for scorer in scorers:
                    vacancy.suggestions.append({
                        'scorer': scorer,
                        'score': score
                    })

    events = list(events.values())
    events.sort(key=lambda e: e['date'])
    return render(request, 'planning/manage.html', {
        'events': events,
        'pool': pool
    })
Example #19
0
def planning_view(request):
    period = 'now'
    if request.GET.get('year') in {'past1', 'past2'}:
        period = request.GET['year']
    pools = list(Pool.all())
    poolid2index = dict()
    poolids = set()
    for pool in pools:
        poolids.add(_id(pool))
    # TODO reduce number of queries
    current = date_to_midnight(now() - datetime.timedelta(days=1))
    past1year = date_to_midnight(now() - datetime.timedelta(days=356))
    past2year = date_to_midnight(now() - datetime.timedelta(days=356 * 2))
    if period == 'now':
        start = current
        end = None
    elif period == 'past1':
        start = past1year
        end = current
    elif period == 'past2':
        start = past2year
        end = past1year
    event_entities = list(Event.all_in_period(start, end))
    used_pools = set()
    for e in event_entities:
        for v in e.vacancies():
            used_pools.add(v.pool_id)
    pools_tpl = []
    i = 0
    for pool in pools:
        if _id(pool) not in used_pools:
            continue
        poolid2index[pool._id] = i
        pools_tpl.append(pool)
        i += 1
    events = list()
    for e in event_entities:
        ei = {
            'id': _id(e),
            'name': e.name,
            'datetime': e.date,
            'kind': e.kind,
            'vacancies': dict()
        }
        for index in poolid2index.values():
            ei['vacancies'][index] = list()
        for v in e.vacancies():
            ei['vacancies'][poolid2index[v.pool_id]].append({
                'begin':
                v.begin,
                'begin_time':
                v.begin_time,
                'end_time':
                v.end_time,
                'assignee':
                v.assignee.humanName if v.assignee else "?"
            })
        for index in poolid2index.values():
            ei['vacancies'][index].sort(key=lambda x: x['begin'])
        events.append(ei)
    events.sort(key=lambda x: x['datetime'])
    return render(request, 'planning/overview.html', {
        'events': events,
        'pools': pools_tpl,
        'period': period
    })
Example #20
0
    'tappers': [[hm2s(20, 30), hm2s(23), 'eerste dienst'],
    [hm2s(23), hm2s(25), 'tweede dienst'],
    [hm2s(25), hm2s(28), 'derde dienst']],
    'bestuur': [[hm2s(20, 30), hm2s(24), 'openen'],
    [hm2s(24), hm2s(28), 'sluiten']],
    'draai': [[hm2s(20, 45), hm2s(23), 'openen'],
    [hm2s(23), hm2s(24), 'prime-time'],
    [hm2s(24), hm2s(25), 'sluiten']]}

day = datetime.datetime.strptime(sys.argv[1], '%Y-%m-%d')

e = Event({
    'name': 'Borrel',
    'date': day,
    'kind': 'borrel' })
e.save()

for _type, periods in typePeriods.items():
    pool = Pool.by_name(_type)
    for period in periods:
        v = Vacancy({
            'name': period[2],
            'event': _id(e),
            'begin': day + datetime.timedelta(seconds=period[0]),
            'end': day + datetime.timedelta(seconds=period[1]),
            'pool': _id(pool),
            'assignee': None,
        })
        print v._data
        v.save()
Example #21
0
    'tappers': [[hm2s(20, 30), hm2s(23), 'eerste dienst'],
    [hm2s(23), hm2s(25), 'tweede dienst'],
    [hm2s(25), hm2s(28), 'derde dienst']],
    'bestuur': [[hm2s(20, 30), hm2s(24), 'openen'],
    [hm2s(24), hm2s(28), 'sluiten']],
    'draai': [[hm2s(20, 45), hm2s(23), 'openen'],
    [hm2s(23), hm2s(24), 'prime-time'],
    [hm2s(24), hm2s(25), 'sluiten']]}

day = datetime.datetime.strptime(sys.argv[1], '%Y-%m-%d')

e = Event({
    'name': 'Borrel',
    'date': day,
    'kind': 'borrel' })
e.save()

for _type, periods in typePeriods.items():
    pool = Pool.by_name(_type)
    for period in periods:
        v = Vacancy({
            'name': period[2],
            'event': _id(e),
            'begin': day + datetime.timedelta(seconds=period[0]),
            'end': day + datetime.timedelta(seconds=period[1]),
            'pool': _id(pool),
            'assignee': None,
        })
        print v._data
        v.save()
Example #22
0
def planning_poollist(request):
    pools = list(Pool.all())
    return render_to_response('planning/pools.html',
            {'pools': pools},
            context_instance=RequestContext(request))
Example #23
0
def planning_manage(request, poolname):
    pool = Pool.by_name(poolname)
    if pool is None:
        raise Http404
    if not request.user.cached_groups_names & set(['secretariaat',
        pool.administrator]):
        raise PermissionDenied
    # TODO reduce number of queries
    events = dict()
    for e in Event.all_in_future():
        eid = _id(e)
        vacancies = list(e.vacancies(pool=pool))
        events[eid] = {'vacancies': vacancies, 'date': e.date.date(),
                'name': e.name, 'kind': e.kind, 'id': eid}
        posted = False
        events[eid]['vacancies'].sort(key=lambda v: v.begin)
        if request.method == 'POST' and _id(request.POST['eid']) == eid:
            events[eid]['form'] = ManagePlanningForm(request.POST, pool=pool,
                    vacancies=events[eid]['vacancies'])
            posted = True
        else:
            events[eid]['form'] = ManagePlanningForm(pool=pool,
                    vacancies=events[eid]['vacancies'])
        if posted and events[eid]['form'].is_valid():
            for vacancy in events[eid]['vacancies']:
                worker = request.POST['shift_%s' % vacancy._id]
                if worker == '':
                    vacancy.assignee = None
                    vacancy.reminder_needed = True
                else:
                    if vacancy.assignee_id == None or \
                            _id(vacancy.assignee_id) != _id(worker):
                        delta = datetime.timedelta(days=5)
                        vacancy.reminder_needed = now() + delta < e.date
                        vacancy.assignee_id = _id(worker)
                vacancy.save()
    workers = pool.workers()
    # XXX het is cooler de shift dichtstbijzijnd aan de vacancy te
    # zoeken.  Stel dat iemand over een half-jaar al is ingepland
    # dan is dat niet zo boeiend.  Terwijl hij nu geen enkele
    # bardienst meer zou krijgen
    shifts = pool.last_shifts()
    for eid in events:
        for vacancy in events[eid]['vacancies']:
            vacancy.suggestions = list()
            workers_by_score = dict()
            for worker in workers:
                score = planning_vacancy_worker_score(vacancy,
                                    worker)
                if score not in workers_by_score:
                    workers_by_score[score] = list()
                workers_by_score[score].append(worker)
            found_scores = list(workers_by_score.keys())
            found_scores.sort(reverse=True)
            for score in found_scores:
                scorers = workers_by_score[score]
                shuffle(scorers)
                scorers.sort(key=lambda x: shifts[_id(x)], cmp=cmp_None)
                for scorer in scorers:
                    vacancy.suggestions.append({'scorer': scorer, 'score': score})

    events = list(events.values())
    events.sort(key=lambda e: e['date'])
    return render_to_response('planning/manage.html',
            {'events': events, 'pool': pool},
           context_instance=RequestContext(request))
Example #24
0
# vim: et:sta:bs=2:sw=4:
import _import
import datetime
import sys

import kn.leden.entities as Es
from kn.settings import DT_MIN, DT_MAX
from kn.leden.mongo import _id
from kn.leden.date import now
from kn.planning.entities import Pool, Worker, Vacancy

pools = dict()
for p in Pool.all():
    pools[p.name] = p

workers = dict()
for w in Worker.all():
    workers[w.get_user()] = w

dt = now()
for w in workers.itervalues():
    del w.pools[:]
for type in ['tappers', 'bestuur', 'barco', 'draai', 'cocks']:
    poolid = _id(pools[type])
    group = Es.by_name(type)
    relations = group.get_rrelated(None, dt, dt, True, None, None)
    for gm in relations:
        if gm['who'] in workers:
            if poolid not in workers[gm['who']].pools:
                print '%s -> %s' % (gm['who'].name, type)
                workers[gm['who']].pools.append(poolid)
Example #25
0
def event_edit(request, eventid):
    if not may_manage_planning(request.user):
        raise PermissionDenied
    avform = None
    e = Event.by_id(eventid)
    if e is None:
        raise Http404
    if request.method == 'POST':
        if request.POST['action'] == 'remove_event':
            for vacancy in e.vacancies():
                vacancy.delete()
            e.delete()
            return HttpResponseRedirect(reverse('planning-poollist'))
        elif request.POST['action'] == 'remove_vacancy':
            Vacancy.by_id(_id(request.POST['vacancy_id'])).delete()
            return HttpResponseRedirect(
                reverse('planning-event-edit', args=(eventid, )))
        elif request.POST['action'] == 'add_vacancy':
            avform = AddVacancyForm(request.POST)
            if avform.is_valid():
                fd = avform.cleaned_data
                (begin_hour, begin_minute) = map(int, fd['begin'].split(':'))
                (end_hour, end_minute) = map(int, fd['end'].split(':'))
                begin_offset = hm2s(begin_hour, begin_minute)
                end_offset = hm2s(end_hour, end_minute)
                begin_date = e.date + datetime.timedelta(seconds=begin_offset)
                end_date = e.date + datetime.timedelta(seconds=end_offset)
                v = Vacancy({
                    'name':
                    fd['name'],
                    'event':
                    _id(e),
                    'begin':
                    (begin_date, fd['begin_is_approximate'] == "True"),
                    'end': (end_date, fd['end_is_approximate'] == "True"),
                    'pool':
                    _id(fd['pool']),
                    'assignee':
                    None,
                    'reminder_needed':
                    True,
                })
                v.save()
                return HttpResponseRedirect(
                    reverse('planning-event-edit', args=(eventid, )))
    if avform is None:
        avform = AddVacancyForm()
    pools = dict()
    for p in Pool.all():
        pools[_id(p)] = p
    vacancies = list()
    for v in e.vacancies():
        v.poolname = pools[v.pool_id].name
        v.assignee_text = str(v.assignee.name) if v.assignee else "-"
        v.vid = str(v._id)
        vacancies.append(v)
    vacancies.sort(key=lambda x: str(x.pool_id) + str(x.begin))
    return render(
        request,
        'planning/event_edit.html',
        {
            'name': e.name,
            'kind': e.kind,
            'date': e.date.date(),
            'avform': avform,
            'vacancies': vacancies
        },
    )
Example #26
0
def planning_poollist(request):
    pools = list(Pool.all())
    return render_to_response('planning/pools.html', {'pools': pools},
                              context_instance=RequestContext(request))
Example #27
0
from kn.leden.mongo import _id
from kn.settings import DT_MIN, DT_MAX
from kn.planning.entities import Pool, Worker, Event, Vacancy


def hm2s(hours, minutes=0):
    return (hours * 60 + minutes) * 60


begin = datetime.datetime.strptime('%s %s' % (sys.argv[1], sys.argv[2]),
                                   '%Y-%m-%d %H:%M')

e = Event({
    'name': 'Dranktelling',
    'date': datetime.datetime.combine(begin.date(), datetime.time())
})
e.save()

pool = Pool.by_name('barco')

for p in [1, 2]:
    v = Vacancy({
        'name': 'Teller %d' % p,
        'event': _id(e),
        'begin': begin,
        'end': begin + datetime.timedelta(seconds=1800),
        'pool': _id(pool),
        'assignee': None
    })
    print v._data
    v.save()