-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcreate_koha_csv.py
executable file
·326 lines (281 loc) · 10.6 KB
/
create_koha_csv.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
#!/usr/bin/env python
"""
Usage:
create_koha_csv.py <prox_report.csv> --end <YYYY-MM-DD>
Options:
--end=DATE last day of the semester in YYYY-MM-DD format
Converts JSON data from Workday into Koha patron import CSV. We run it once per
semester just prior start. A couple things should be manually checked:
- Ensure mappings haven't changed (see koha_mappings.py and new-programs.sh)
- Look up last day of the semester for --end flag
The JSON files from Workday come from the integration-success bucket and the
script expects them to be in the root and keep their names, but you can
override them with the following environment variables:
STUDENT_DATA=student_data.json
PRECOLLEGE_DATA=student_pre_college_data.json
EMPLOYEE_DATA=employee_data.json
OUTPUT_FILE=patron_bulk_import.csv
"""
import csv
from datetime import date, timedelta
import json
import os
from docopt import docopt
from termcolor import colored
from koha_mappings import category, fac_depts, stu_major
from patron_update import create_prox_map
from workday.models import Employee, Student
from workday.utils import get_entries
today = date.today()
files = {
"student": os.environ.get("STUDENT_DATA", "student_data.json"),
"precollege": os.environ.get("PRECOLLEGE_DATA", "student_pre_college_data.json"),
"employee": os.environ.get("EMPLOYEE_DATA", "employee_data.json"),
"output": os.environ.get("OUTPUT_FILE", f"patron_bulk_import.csv"),
}
def warn(string) -> None:
print(colored("Warning: " + string, "red"))
def is_exception(user: Employee | Student) -> bool:
exceptions = ["deborahstein", "sraffeld"]
if user.username in exceptions:
return True
return False
def make_student_row(student) -> dict | None:
student = Student(**student)
if is_exception(student):
return None
# some students don't have CCA emails, skip them
# one student record in Summer 2021 lacked a last_name
if student.inst_email is None or student.last_name is None:
return None
patron = {
"branchcode": "SF",
"categorycode": category[student.academic_level],
# fill in Prox number if we have it, or default to UID
"cardnumber": prox_map.get(student.universal_id, student.universal_id).strip(),
"dateenrolled": today.isoformat(),
"dateexpiry": args["--end"],
"email": student.inst_email,
"firstname": student.first_name,
"patron_attributes": "UNIVID:{},STUID:{}".format(
student.universal_id, student.student_id
),
# "phone": student.get("phone", ""),
"surname": student.last_name,
"userid": student.username,
}
# note for pre-college and skip student major calculation
if student.academic_level == "Pre-College":
patron["borrowernotes"] = f"Pre-college {today.year}"
else:
# handle student major (additional patron attribute)
major = None
if student.primary_program in stu_major:
major = str(stu_major[student.primary_program])
patron["patron_attributes"] += ",STUDENTMAJ:{}".format(major)
else:
for program in student.programs:
if program["program"] in stu_major:
major = str(stu_major[program["program"]])
patron["patron_attributes"] += ",STUDENTMAJ:{}".format(major)
break
# we couldn't find a major, print a warning
if major is None:
warn(
f"""Unable to parse major for student {student.username}
Primary program: {student.primary_program}
Program credentials: {student.programs}"""
)
return patron
def expiration_date(person: Employee) -> str:
"""Calculate patron expiration date based on personnel data and the last
day of the semester.
Parameters
----------
person : dict
Dict of user data. "etype" and "future_etype" are most important here.
Returns
-------
str (in YYYY-MM-DD format)
The appropriate expiration date as an ISO-8601 date string. For faculty
added during Fall, this is Jan 31 of the next year. For faculty added
during Spring, this is May 31 of the current year. For staff, it is the
last day of the last month of the impending semester.
"""
# there are 3 etypes: Staff, Instructors, Faculty. Sometimes we do not have
# an etype but _do_ have a "future_etype".
etype = person.etype or person.etype_future
if not etype:
warn(
(
"Employee {} does not have an etype nor a etype_future. They "
"will be assigned the Staff expiration date.".format(person.username)
)
)
etype = "Staff"
d: date = date.fromisoformat(args["--end"])
if etype == "Instructors":
# go into next month then subtract the number of days from next month
next_mo = d.replace(day=28) + timedelta(days=4)
return str(next_mo - timedelta(days=next_mo.day))
elif etype == "Staff":
# one year from now
return str(today.replace(year=today.year + 1))
else:
# implies faculty
# Spring => May 31
if d.month == 5:
return str(d.replace(day=31))
# Summer => Aug 31
elif d.month == 8:
return str(d.replace(day=31))
# Fall => Jan 31 of the following year
elif d.month == 12:
return str(d.replace(year=d.year + 1, month=1, day=31))
else:
warn(
f"""End date {args['--end']} is not in May, August, or December so it does not map to a typical semester. Faculty accounts will given the Staff expiration date of one year."""
)
return str(today.replace(year=today.year + 1))
pass
def make_employee_row(person) -> dict | None:
person = Employee(**person)
if is_exception(person):
return None
# skip inactive, people w/o emails, & the one random record for a student
if (
not person.active_status
or not person.work_email
or person.etype in ("Contingent Employees/Contractors", "Students")
):
return None
# create a hybrid program/department field
# some people have neither (tend to be adjuncts or special programs staff)
prodep = None
if person.program:
prodep = person.program
elif person.department:
prodep = person.department
elif person.job_profile in fac_depts:
prodep = person.job_profile
# skip inactive special programs faculty
if person.job_profile == "Special Programs Instructor (inactive)":
return None
# skip contingent employees
if person.is_contingent == "1":
return None
# we assume etype=Instructors => special programs faculty
if (
person.etype == "Instructors"
and person.job_profile
not in (
"Atelier Instructor",
"Special Programs Instructor",
"YASP & Atelier Youth Programs Instructor",
)
and person.job_profile not in fac_depts
):
warn(
(
"Instructor {} is not a Special Programs Instructor, check " "record."
).format(person.username)
)
patron = {
"branchcode": "SF",
"categorycode": category.get(person.etype or person.etype_future or "Staff"),
# fill in Prox number if we have it, or default to UID
"cardnumber": prox_map.get(person.universal_id, person.universal_id).strip(),
"dateenrolled": today.isoformat(),
"dateexpiry": expiration_date(person),
"email": person.work_email,
"firstname": person.first_name,
"patron_attributes": "UNIVID:" + person.universal_id,
"phone": person.work_phone,
"surname": person.last_name,
"userid": person.username,
}
# handle faculty/staff department (additional patron attribute)
if prodep and prodep in fac_depts:
code = str(fac_depts[prodep])
patron["patron_attributes"] += ",FACDEPT:{}".format(code)
elif prodep:
# there's a non-empty program/department value we haven't accounted for
warn(
"""No mapping in koha_mappings.fac_depts for faculty/staff prodep
"{}", see patron {}""".format(
prodep, person.username
)
)
if prodep is None:
warn(
"Employee {} has no academic program or department:".format(person.username)
)
print(person)
return patron
def file_exists(fn) -> bool:
if not os.path.exists(fn):
warn(f'Did not find "{fn}" file')
return False
return True
def proc_students(pc=False) -> None:
if pc:
file = files["precollege"]
prefix = " pre-college "
else:
file = files["student"]
prefix = " "
if file_exists(file):
print(f"Adding{prefix}students to Koha patron CSV.")
with open(file, "r") as file:
students = get_entries(json.load(file))
with open(files["output"], "a") as output:
writer = csv.DictWriter(output, fieldnames=koha_fields)
for stu in students:
row = make_student_row(stu)
if row:
writer.writerow(row)
def proc_staff() -> None:
if file_exists(files["employee"]):
print("Adding Faculty/Staff to Koha patron CSV.")
with open(files["employee"], "r") as file:
employees = get_entries(json.load(file))
# open in append mode & don't add header row
with open(files["output"], "a") as output:
writer = csv.DictWriter(output, fieldnames=koha_fields)
for employee in employees:
row = make_employee_row(employee)
if row:
writer.writerow(row)
def main() -> None:
# write header row
with open(files["output"], "w+") as output:
writer = csv.DictWriter(output, fieldnames=koha_fields)
writer.writeheader()
proc_students()
proc_students(pc=True)
proc_staff()
print(
"Done! Upload the CSV at "
"https://library-staff.cca.edu/cgi-bin/koha/tools/import_borrowers.pl"
)
if __name__ == "__main__":
args = docopt(__doc__, version="Create Koha CSV 1.0") # type: ignore
PROX_FILE = args["<prox_report.csv>"]
if not file_exists(PROX_FILE):
exit(1)
prox_map = create_prox_map(PROX_FILE)
koha_fields = [
"branchcode",
"cardnumber",
"categorycode",
"dateenrolled",
"dateexpiry",
"email",
"firstname",
"patron_attributes",
"surname",
"userid",
"phone",
"borrowernotes",
]
main()