2018-02-14 16:43:28 +01:00
|
|
|
#! /usr/bin/env nix-shell
|
2024-01-08 15:44:28 +01:00
|
|
|
#!nix-shell -i python3 -p python3 python3Packages.pyyaml -I nixpkgs=flake:nixpkgs
|
|
|
|
# pip install pyyaml pyscipopt pyright
|
2018-02-14 16:43:28 +01:00
|
|
|
import sys
|
|
|
|
import yaml
|
|
|
|
import re
|
2024-01-08 15:44:28 +01:00
|
|
|
import argparse
|
2024-04-15 09:54:15 +02:00
|
|
|
from collections import OrderedDict, defaultdict
|
2024-01-08 15:44:28 +01:00
|
|
|
from pyscipopt import Model, quicksum
|
2024-01-08 12:06:10 +01:00
|
|
|
from typing import Any, Tuple, TypeVar
|
|
|
|
from dataclasses import dataclass, field
|
2024-04-15 10:22:47 +02:00
|
|
|
from tabulate import tabulate
|
2022-11-09 18:06:10 +01:00
|
|
|
conf = yaml.safe_load(open('config.yaml', 'r'))
|
2024-04-15 09:49:59 +02:00
|
|
|
|
|
|
|
DEFAULT_CONFIG = {
|
2024-04-15 10:22:47 +02:00
|
|
|
"max_load_person": 6,
|
|
|
|
"day_names": [f"dag {str(i)}" for i in range(conf['config']['days'])],
|
2024-04-15 09:49:59 +02:00
|
|
|
}
|
|
|
|
config = DEFAULT_CONFIG | conf['config']
|
2019-02-18 11:36:35 +01:00
|
|
|
config['ignore'].append('')
|
2024-04-15 10:22:47 +02:00
|
|
|
assert(len(config['day_names']) == config['days'])
|
2024-01-08 12:06:10 +01:00
|
|
|
|
2024-01-08 15:44:28 +01:00
|
|
|
QUADRATIC = False
|
|
|
|
|
2024-01-08 12:06:10 +01:00
|
|
|
@dataclass
|
|
|
|
class TaskConfig:
|
|
|
|
personen: list[str]
|
|
|
|
workload: int
|
|
|
|
req: list[int]
|
2024-04-15 09:54:15 +02:00
|
|
|
name: str
|
2024-01-08 12:06:10 +01:00
|
|
|
hardcode: list[str] | None = None
|
|
|
|
lookup: list[str] = field(default_factory=list)
|
|
|
|
|
2024-04-15 09:54:15 +02:00
|
|
|
tasks: dict[str, TaskConfig] = OrderedDict({k: TaskConfig(**({"name": k} | t)) for k, t in conf['tasks'].items()})
|
2024-01-08 12:06:10 +01:00
|
|
|
X = TypeVar("X")
|
|
|
|
def index(x: dict[X, Any]) -> dict[X, int]:
|
|
|
|
return {v: k for k, v in enumerate(x)}
|
2019-02-18 11:36:35 +01:00
|
|
|
daily_workloads = \
|
2024-01-08 12:06:10 +01:00
|
|
|
[sum(task.workload * task.req[d] for task in tasks.values()) for d in range(config['days'])]
|
|
|
|
ALL_DAYS: set[int] = set(range(config['days']))
|
2018-02-14 16:43:28 +01:00
|
|
|
class Person(object):
|
2024-04-15 09:54:15 +02:00
|
|
|
def __init__(self, name: str, conf={"dagen":ALL_DAYS}):
|
|
|
|
self.name = name
|
2024-01-08 15:44:28 +01:00
|
|
|
self.can: set[str] = set()
|
|
|
|
self.loves: set[str] = set()
|
|
|
|
self.hates: set[str] = set()
|
|
|
|
self.does: set[Tuple[int, str]] = set() # hardcoded
|
2018-02-14 16:43:28 +01:00
|
|
|
self.has_prefs = False
|
|
|
|
self.conf = conf
|
2019-02-18 11:36:35 +01:00
|
|
|
self.conf['dagen'] = set(conf['dagen'])
|
2018-02-14 16:43:28 +01:00
|
|
|
def vrolijkheid(self):
|
|
|
|
res = config['days'] - len(self.does)
|
2024-01-08 12:06:10 +01:00
|
|
|
for (_,t) in self.does:
|
2018-02-14 16:43:28 +01:00
|
|
|
if t in self.loves:
|
|
|
|
res += config['weights']['likes']
|
|
|
|
if t in self.hates:
|
|
|
|
res -= config['weights']['hates']
|
|
|
|
return res
|
|
|
|
def workload(self, tasks):
|
2024-01-08 15:44:28 +01:00
|
|
|
return sum(tasks[t].workload for (_,t) in self.does)
|
2019-02-18 11:36:35 +01:00
|
|
|
def cost(self, num_people):
|
|
|
|
return round(sum((daily_workloads[d] for d in self.conf['dagen'])) / num_people)
|
|
|
|
# probabilistic round: int(math.floor(x + random.random()))
|
2024-01-08 12:06:10 +01:00
|
|
|
def read_people(conf_ppl) -> dict[str, Person]:
|
2018-02-14 16:43:28 +01:00
|
|
|
people = OrderedDict()
|
|
|
|
for x in conf_ppl:
|
2019-02-18 11:36:35 +01:00
|
|
|
val = {"dagen": ALL_DAYS}
|
2024-01-08 15:44:28 +01:00
|
|
|
if isinstance(x, dict):
|
2018-02-14 16:43:28 +01:00
|
|
|
x,val = x.popitem()
|
2024-04-15 09:54:15 +02:00
|
|
|
people[x.lower()] = Person(x, val)
|
2018-02-14 16:43:28 +01:00
|
|
|
return people
|
|
|
|
# deal with loves/hates
|
2024-01-08 12:06:10 +01:00
|
|
|
def make_task_lut(tasks: dict[str, TaskConfig]):
|
2024-04-15 09:54:15 +02:00
|
|
|
task_lut = defaultdict(set)
|
2018-02-14 16:43:28 +01:00
|
|
|
for t, taskconf in tasks.items():
|
2024-01-08 12:06:10 +01:00
|
|
|
for lookup in taskconf.lookup:
|
2024-04-15 09:54:15 +02:00
|
|
|
task_lut[lookup] |= {t.lower()}
|
2018-02-14 16:43:28 +01:00
|
|
|
task_re = re.compile(config['task_re'])
|
|
|
|
def lookup_tasks(tasks):
|
2024-04-15 09:54:15 +02:00
|
|
|
return set.union(*(task_lut[x.strip()] for x in task_re.split(tasks) if x not in config['ignore']))
|
2018-02-14 16:43:28 +01:00
|
|
|
return lookup_tasks
|
|
|
|
def read_prefs(pref_file, tasks, people):
|
|
|
|
lookup_tasks = make_task_lut(tasks)
|
|
|
|
# read the wiki corvee table
|
|
|
|
for [name, loves, hates] in ((q.strip().lower() for q in x.split('\t')) for x in pref_file):
|
|
|
|
p = people[name]
|
|
|
|
p.has_prefs = True
|
|
|
|
p.loves |= set(lookup_tasks(loves))
|
|
|
|
p.hates |= set(lookup_tasks(hates))
|
|
|
|
for name, p in people.items():
|
|
|
|
if not p.has_prefs:
|
|
|
|
print("warning: no preferences for", name, file=sys.stderr)
|
|
|
|
# deal with capability and hardcode
|
2024-01-08 12:06:10 +01:00
|
|
|
def set_capabilities(tasks: dict[str, TaskConfig], people: dict[str, Person]):
|
|
|
|
for (task,conf) in tasks.items():
|
|
|
|
if conf.personen == 'iedereen':
|
2018-02-14 16:43:28 +01:00
|
|
|
for p in people.values():
|
|
|
|
p.can.add(task)
|
2024-01-08 12:06:10 +01:00
|
|
|
elif conf.personen == 'liefhebbers':
|
2018-02-14 16:43:28 +01:00
|
|
|
for p in people.values():
|
|
|
|
if task in p.loves:
|
|
|
|
p.can.add(task)
|
|
|
|
else:
|
2024-01-08 12:06:10 +01:00
|
|
|
for p in conf.personen:
|
2018-02-14 16:43:28 +01:00
|
|
|
people[p.lower()].can.add(task)
|
2024-01-08 12:06:10 +01:00
|
|
|
if conf.hardcode is not None:
|
|
|
|
for day, pers in enumerate(conf.hardcode):
|
2024-04-15 09:56:23 +02:00
|
|
|
if pers:
|
|
|
|
people[pers.lower()].does.add((day, task))
|
2024-01-08 15:44:28 +01:00
|
|
|
|
2024-04-15 10:22:47 +02:00
|
|
|
def write_tasks(people: dict[str, Person], tasks: dict[str, TaskConfig], file=sys.stdout):
|
|
|
|
headers = ["wie"] + config['day_names']
|
|
|
|
if not args.simple:
|
|
|
|
headers += ["workload", "vrolijkheid"]
|
|
|
|
tabl = []
|
|
|
|
for p in people.values():
|
|
|
|
days = [[] for _ in range(config['days'])]
|
2018-02-14 16:43:28 +01:00
|
|
|
for (d,t) in p.does:
|
2019-02-18 11:36:35 +01:00
|
|
|
days[d].append((t, t in p.loves, t in p.hates))
|
2024-04-15 10:22:47 +02:00
|
|
|
if not args.simple:
|
|
|
|
def q(w):
|
|
|
|
return ",".join([tasks[t].name + (" :)" if love else "") + (" :(" if hate else "") for (t,love,hate) in w])
|
|
|
|
else:
|
|
|
|
def q(w):
|
|
|
|
return ",".join([tasks[t].name for (t,_,_) in w])
|
|
|
|
row = [p.name, *map(q, days)]
|
|
|
|
if not args.simple:
|
|
|
|
row += [p.workload(tasks), p.vrolijkheid()]
|
|
|
|
tabl.append(row)
|
|
|
|
print(tabulate(tabl, headers=headers, tablefmt=args.output_format), file=file)
|
2024-01-08 12:06:10 +01:00
|
|
|
|
2024-01-08 15:44:28 +01:00
|
|
|
def scipsol(people: dict[str, Person], tasks: dict[str, TaskConfig]):
|
|
|
|
max_loads: dict[Tuple[str, int], int] = {}
|
|
|
|
for i,p in people.items():
|
|
|
|
if 'max_load' in p.conf: # max_load override for Pol
|
|
|
|
for d,load in enumerate(p.conf['max_load']):
|
|
|
|
max_loads[(i,d)] = load
|
|
|
|
# filter days that the person does not exist
|
|
|
|
for d in ALL_DAYS - p.conf['dagen']:
|
|
|
|
max_loads[(i,d)] = 0
|
|
|
|
|
|
|
|
m = Model()
|
|
|
|
does = {}
|
|
|
|
happiness = []
|
|
|
|
stdevs = []
|
|
|
|
errors = []
|
|
|
|
for pname, person in people.items():
|
|
|
|
workloads = []
|
|
|
|
p_error = m.addVar(vtype="I", name=f"{pname}_error", lb=0, ub=None)
|
|
|
|
for d in ALL_DAYS:
|
|
|
|
pdt = []
|
|
|
|
for task in tasks:
|
|
|
|
var = m.addVar(vtype="B", name=f"{pname}_does_{task}@{d}")
|
|
|
|
pdt.append(var)
|
|
|
|
does[(pname, d, task)] = var
|
|
|
|
# a person only does what (s)he is capable of
|
|
|
|
if task not in person.can:
|
|
|
|
m.addCons(var == 0)
|
|
|
|
workloads.append(var * tasks[task].workload)
|
|
|
|
for task in person.loves:
|
|
|
|
happiness.append(does[(pname, d, task)] * config['weights']['likes'])
|
|
|
|
for task in person.hates:
|
|
|
|
happiness.append(does[(pname, d, task)] * (config['weights']['hates'] * -1))
|
|
|
|
|
|
|
|
# max_load_person: a person only has one task per day at most
|
|
|
|
m.addCons(quicksum(pdt) <= max_loads.get((pname, d), 1))
|
|
|
|
|
|
|
|
m.addCons(p_error >= person.cost(len(people)) - quicksum(workloads))
|
|
|
|
m.addCons(p_error >= quicksum(workloads) - person.cost(len(people)))
|
|
|
|
errors.append(p_error)
|
|
|
|
|
|
|
|
stdevs.append((person.cost(len(people)) - quicksum(workloads)) ** 2)
|
|
|
|
|
|
|
|
# min_load_person: person has at least 1 task
|
|
|
|
m.addCons(quicksum([does[(pname, d, task)] for d in ALL_DAYS for task in tasks]) >= 1)
|
|
|
|
|
|
|
|
# duplicate_jobs: a person does not perform the same job on all days
|
|
|
|
for task in tasks:
|
|
|
|
m.addCons(quicksum(does[(pname, d, task)] for d in ALL_DAYS) <= len(ALL_DAYS) - 1)
|
|
|
|
|
|
|
|
# max_load_person_total
|
2024-04-15 09:49:59 +02:00
|
|
|
m.addCons(quicksum([does[(pname, d, task)] * tasks[task].workload for d in ALL_DAYS for task in tasks]) <= config['max_load_person'])
|
2024-01-08 15:44:28 +01:00
|
|
|
|
|
|
|
# hardcode constraint
|
|
|
|
for d, task in person.does:
|
|
|
|
m.addCons(does[(pname, d, task)] == 1)
|
|
|
|
|
|
|
|
|
|
|
|
# all_allocated: each task in allocated
|
|
|
|
for j in tasks:
|
|
|
|
for d in ALL_DAYS:
|
|
|
|
m.addCons(quicksum(does[(p, d, j)] for p in people) == tasks[j].req[d])
|
|
|
|
|
|
|
|
objective = m.addVar(name="objvar", vtype="C", lb=None, ub=None)
|
|
|
|
m.setObjective(objective, "maximize")
|
|
|
|
if QUADRATIC:
|
|
|
|
m.addCons(objective <= quicksum(happiness) - quicksum(stdevs) / 2)
|
|
|
|
else:
|
|
|
|
if args.max_total_error is not None:
|
|
|
|
m.addCons(quicksum(errors) <= args.max_total_error)
|
|
|
|
m.addCons(objective <= quicksum(happiness) - quicksum(errors))
|
|
|
|
m.setObjective(objective, "maximize")
|
|
|
|
m.solveConcurrent()
|
|
|
|
for pname, person in people.items():
|
|
|
|
for d in ALL_DAYS:
|
|
|
|
for task in tasks:
|
|
|
|
if m.getVal(does[(pname, d, task)]):
|
|
|
|
person.does.add((d, task))
|
|
|
|
write_tasks(people, tasks)
|
|
|
|
|
|
|
|
print("Totale vrolijkheid:", sum(p.vrolijkheid() for p in people.values()))
|
|
|
|
print("workload deviation:", m.getVal(quicksum(stdevs)))
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument("-q", "--quadratic", action="store_true")
|
2024-04-15 10:22:47 +02:00
|
|
|
parser.add_argument("--simple", action="store_true", help="hide workload and happiness")
|
2024-01-08 15:44:28 +01:00
|
|
|
parser.add_argument("--max_total_error", type=int, default=None)
|
2024-04-15 10:22:47 +02:00
|
|
|
parser.add_argument("--output-format", default="mediawiki", help="`tabulate` output format")
|
2024-01-08 15:44:28 +01:00
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
QUADRATIC = args.quadratic
|
|
|
|
|
2018-02-14 16:43:28 +01:00
|
|
|
people = read_people(conf['people'])
|
|
|
|
with open('prefs_table', 'r') as pref_file:
|
|
|
|
read_prefs(pref_file, tasks, people)
|
|
|
|
set_capabilities(tasks, people)
|
2024-01-08 15:44:28 +01:00
|
|
|
scipsol(people, tasks)
|