From 98ed86c767dd9a765d77ec449d8c6008c970b1bc Mon Sep 17 00:00:00 2001 From: Anastasia Diseth <58883418+anadis504@users.noreply.github.com> Date: Mon, 3 Jun 2024 12:05:24 +0300 Subject: [PATCH] cluster the data columns by the exercise_name (and module) (#62) --- data-parser/main.py | 391 ++++++++++++++++++++++++++++++-------------- 1 file changed, 266 insertions(+), 125 deletions(-) diff --git a/data-parser/main.py b/data-parser/main.py index 1714f1c..8b0936b 100644 --- a/data-parser/main.py +++ b/data-parser/main.py @@ -1,5 +1,5 @@ import polars as pl -import json +import json from collections.abc import Iterable from collections import ChainMap from datetime import datetime @@ -7,6 +7,7 @@ from os import listdir from os.path import isfile, join + # Function for flattening an array of arbitrary nested arrays or objects def flatten(xs): result = [] @@ -17,6 +18,7 @@ def flatten(xs): result.append(x) return result + # Check that the ./data folder exsists try: dir = os.listdir("./data") @@ -28,136 +30,217 @@ def flatten(xs): userdetail_files = [] exercisetasks_files = [] user_consents_files = [] -datafiles = [f for f in listdir('./data/') if isfile(join('./data/', f))] +datafiles = [f for f in listdir("./data/") if isfile(join("./data/", f))] course_names = set() for f in datafiles: - [course_name, file_name] = f.split(' - ', 1) - #course_names.add(course_name) - if 'Submissions' in file_name: + [course_name, file_name] = f.split(" - ", 1) + # course_names.add(course_name) + if "Submissions" in file_name: submission_files.append(f) - elif 'Exercise tasks' in file_name: + elif "Exercise tasks" in file_name: exercisetasks_files.append(f) - elif 'User Details' in file_name: + elif "User Details" in file_name: userdetail_files.append(f) - elif 'User Consents' in file_name: + elif "User Consents" in file_name: user_consents_files.append(f) # Sort the files by date, most recent first -submission_files = sorted(submission_files, key=lambda x: (x.split(' ')[-1]), reverse=True) -exercisetasks_files = sorted(exercisetasks_files, key=lambda x: (x.split(' ')[-1]), reverse=True) -userdetail_files = sorted(userdetail_files, key=lambda x: (x.split(' ')[-1]), reverse=True) -user_consents_files = sorted(user_consents_files, key=lambda x: (x.split(' ')[-1]), reverse=True) +submission_files = sorted( + submission_files, key=lambda x: (x.split(" ")[-1]), reverse=True +) +exercisetasks_files = sorted( + exercisetasks_files, key=lambda x: (x.split(" ")[-1]), reverse=True +) +userdetail_files = sorted( + userdetail_files, key=lambda x: (x.split(" ")[-1]), reverse=True +) +user_consents_files = sorted( + user_consents_files, key=lambda x: (x.split(" ")[-1]), reverse=True +) -[course_name, file_name] = exercisetasks_files[0].split(' - ', 1) +[course_name, file_name] = exercisetasks_files[0].split(" - ", 1) course_names.add(course_name) -[course_name, file_name] = submission_files[0].split(' - ', 1) +[course_name, file_name] = submission_files[0].split(" - ", 1) course_names.add(course_name) -[course_name, file_name] = userdetail_files[0].split(' - ', 1) +[course_name, file_name] = userdetail_files[0].split(" - ", 1) course_names.add(course_name) -[course_name, file_name] = user_consents_files[0].split(' - ', 1) +[course_name, file_name] = user_consents_files[0].split(" - ", 1) course_names.add(course_name) try: - exercise_tasks = pl.read_csv(join('./data/', exercisetasks_files[0])) + exercise_tasks = pl.read_csv(join("./data/", exercisetasks_files[0])) except OSError as error: print(error) try: - submissions = (pl.read_csv(join('./data/', submission_files[0])) - # remove outdated format - .filter(pl.col('created_at') >= '2023-05-22')) + submissions = ( + pl.read_csv(join("./data/", submission_files[0])) + # remove outdated format + .filter(pl.col("created_at") >= "2023-05-22") + ) except OSError as error: print(error) try: - user_details = (pl.read_csv(join('./data/', userdetail_files[0])) - .select(['user_id', pl.concat_str(pl.col('^.*_name$').fill_null('').alias('name'), separator=' '), 'email'])) + user_details = pl.read_csv(join("./data/", userdetail_files[0])).select( + [ + "user_id", + pl.concat_str( + pl.col("^.*_name$").fill_null("").alias("name"), separator=" " + ), + "email", + ] + ) except OSError as error: print(error) try: - user_consents = pl.read_csv(join('./data/', user_consents_files[0])).select(['course_id', 'question','user_id','research_consent']) - + user_consents = pl.read_csv(join("./data/", user_consents_files[0])).select( + ["course_id", "question", "user_id", "research_consent"] + ) + except OSError as error: print(error) - -cleaned_subs = (submissions - .join(user_details.select(pl.exclude('created_at')), on='user_id', how='left') - .join(exercise_tasks.select(['id', 'exercise_type', 'exercise_name', 'course_module_name']), left_on='exercise_task_id', right_on='id', how='left') - .filter(pl.col('exercise_type') == 'dogs-factorial-analysis-survey') - .drop(['course_instance_id', 'score_given','exercise_type']) - .sort('created_at', descending=True) - .unique(subset=['exercise_task_id', 'user_id'], keep='first') - ) +exercise_tasks = exercise_tasks.sort(["course_module_name", "exercise_name"]) + +cleaned_subs = ( + submissions.join( + user_details.select(pl.exclude("created_at")), on="user_id", how="left" + ) + .join( + exercise_tasks.select( + ["id", "exercise_type", "exercise_name", "course_module_name"] + ), + left_on="exercise_task_id", + right_on="id", + how="left", + ) + .filter(pl.col("exercise_type") == "dogs-factorial-analysis-survey") + .drop(["course_instance_id", "score_given", "exercise_type"]) + .sort("created_at", descending=True) + .unique(subset=["exercise_task_id", "user_id"], keep="first") + .sort("course_module_name", "exercise_name") +) user_consents = user_consents.with_columns( - pl.col('question').str.replace_all(',', '').str.replace('Hyväksyn että ', '').str.slice(0, 40) + pl.col("question") + .str.replace_all(",", "") + .str.replace("Hyväksyn että ", "") + .str.slice(0, 40) ) -user_consents = user_consents.pivot(index="user_id", columns="question", values="research_consent", aggregate_function=None) +user_consents = user_consents.pivot( + index="user_id", + columns="question", + values="research_consent", + aggregate_function=None, +) -user_details = user_details.join(user_consents, how='left', on='user_id') +user_details = user_details.join(user_consents, how="left", on="user_id") -course_modules = exercise_tasks.get_column('course_module_name').unique() +course_modules = exercise_tasks.get_column("course_module_name").unique() module_user_details_indexes = dict() module_user_details = [] for index, mod in enumerate(course_modules): df = user_details.clone() - df = df.with_columns(pl.lit(mod).alias('course_module_name')) + df = df.with_columns(pl.lit(mod).alias("course_module_name")) module_user_details.append(df) module_user_details_indexes[mod] = index -#user_details = user_details.join(course_modules, how="cross") +# user_details = user_details.join(course_modules, how="cross") # The map of private-specs: { exercise_task_id : { private_spec } } -exercise_tasks_map = dict([(x[0], json.loads(x[1])) for x in exercise_tasks.select(['id','private_spec','exercise_type']).rows() if 'factorial' in x[2]]) +exercise_tasks_map = dict( + [ + (x[0], json.loads(x[1])) + for x in exercise_tasks.select(["id", "private_spec", "exercise_type"]).rows() + if "factorial" in x[2] + ] +) # Formatting the private_specs to needed fields for exstracting submission info keys_to_delete = [] -for k,v in exercise_tasks_map.items(): - -# factorial survey type: { -# id, -# options: [{id, name, value}], -# questions: [{id, questionLabel}], -# type: 'factorial' -# } - if v['type'] == 'factorial': - exercise_tasks_map[k] = dict([(key, val) for key, val in dict(v).items() if key in ['id', 'type', 'questions', 'options']]) - exercise_tasks_map[k]['questions'] = ([ - dict([(key, val) for key,val in dict(quest).items() if key not in ['question', 'mandatory']]) - for quest in exercise_tasks_map[k]['questions'] if quest['questionLabel'] not in 'info' - ]) - exercise_tasks_map[k]['exercise_name'] = exercise_tasks.select(['exercise_name','id']).row(by_predicate=(pl.col('id') == k))[0] - exercise_tasks_map[k]['course_module_name'] = exercise_tasks.select(['course_module_name','id']).row(by_predicate=(pl.col('id') == k))[0] - -# non-factorial survey type: { -# id, -# content: [{surveyItemId, options, questionLabel, answer-type}], -# type: 'non-factorial' -# } +for k, v in exercise_tasks_map.items(): + + # factorial survey type: { + # id, + # options: [{id, name, value}], + # questions: [{id, questionLabel}], + # type: 'factorial' + # } + if v["type"] == "factorial": + exercise_tasks_map[k] = dict( + [ + (key, val) + for key, val in dict(v).items() + if key in ["id", "type", "questions", "options"] + ] + ) + exercise_tasks_map[k]["questions"] = [ + dict( + [ + (key, val) + for key, val in dict(quest).items() + if key not in ["question", "mandatory"] + ] + ) + for quest in exercise_tasks_map[k]["questions"] + if quest["questionLabel"] not in "info" + ] + exercise_tasks_map[k]["exercise_name"] = exercise_tasks.select( + ["exercise_name", "id"] + ).row(by_predicate=(pl.col("id") == k))[0] + exercise_tasks_map[k]["course_module_name"] = exercise_tasks.select( + ["course_module_name", "id"] + ).row(by_predicate=(pl.col("id") == k))[0] + + # non-factorial survey type: { + # id, + # content: [{surveyItemId, options, questionLabel, answer-type}], + # type: 'non-factorial' + # } else: - exercise_tasks_map[k] = dict([(un, val) for un, val in dict(v).items() if un in ['id', 'type', 'content']]) - content = exercise_tasks_map[k]['content'] - - content = ([ - dict([(key, value ) for key, value in dict(item).items() if key in ['id', 'answer', 'question']]) - for item in content if item['question']['questionLabel'] not in ['info']]) - - for idx,item in enumerate(content): - newItem = {'surveyItemId': item['id'], - 'options': item['answer']['options'], - 'answer-type': item['answer']['type'], - 'questionLabel': item['question']['questionLabel']} + exercise_tasks_map[k] = dict( + [ + (un, val) + for un, val in dict(v).items() + if un in ["id", "type", "content"] + ] + ) + content = exercise_tasks_map[k]["content"] + + content = [ + dict( + [ + (key, value) + for key, value in dict(item).items() + if key in ["id", "answer", "question"] + ] + ) + for item in content + if item["question"]["questionLabel"] not in ["info"] + ] + + for idx, item in enumerate(content): + newItem = { + "surveyItemId": item["id"], + "options": item["answer"]["options"], + "answer-type": item["answer"]["type"], + "questionLabel": item["question"]["questionLabel"], + } content[idx] = newItem if content: - exercise_tasks_map[k]['content'] = content - exercise_tasks_map[k]['exercise_name'] = exercise_tasks.select(['exercise_name','id']).row(by_predicate=(pl.col('id') == k))[0] - exercise_tasks_map[k]['course_module_name'] = exercise_tasks.select(['course_module_name','id']).row(by_predicate=(pl.col('id') == k))[0] + exercise_tasks_map[k]["content"] = content + exercise_tasks_map[k]["exercise_name"] = exercise_tasks.select( + ["exercise_name", "id"] + ).row(by_predicate=(pl.col("id") == k))[0] + exercise_tasks_map[k]["course_module_name"] = exercise_tasks.select( + ["course_module_name", "id"] + ).row(by_predicate=(pl.col("id") == k))[0] else: keys_to_delete.append(k) @@ -166,51 +249,79 @@ def flatten(xs): del exercise_tasks_map[key] # Building additional columns to final dataframe exercise tasks at a time -for k,v in exercise_tasks_map.items(): - submissions_data = cleaned_subs.filter(pl.col('exercise_task_id') == k).select('user_id','data_json', 'created_at').rows() - current_module = v['course_module_name'] +for k, v in exercise_tasks_map.items(): + submissions_data = ( + cleaned_subs.filter(pl.col("exercise_task_id") == k) + .select("user_id", "data_json", "created_at") + .rows() + ) + current_module = v["course_module_name"] ind = module_user_details_indexes[current_module] - user_submissions, col_labels, typed_col_labels = [],[],{} + user_submissions, col_labels, typed_col_labels = [], [], {} - if v['type'] == 'factorial': - col_labels = [lab['questionLabel'] for lab in v['questions']] - col_labels.append('user_id') - exercise_name_label_for_timestamp = v['exercise_name'] - col_labels.append('course_module_name') + if v["type"] == "factorial": + col_labels = [lab["questionLabel"] for lab in v["questions"]] + col_labels.append("user_id") + exercise_name_label_for_timestamp = v["exercise_name"] + col_labels.append("course_module_name") - typed_col_labels = dict(ChainMap(*[{lab['questionLabel']: pl.Int8} for lab in v['questions']])) + typed_col_labels = dict( + ChainMap(*[{lab["questionLabel"]: pl.Int8} for lab in v["questions"]]) + ) if exercise_name_label_for_timestamp not in module_user_details[ind].columns: - col_labels.append(exercise_name_label_for_timestamp) - typed_col_labels[exercise_name_label_for_timestamp] = pl.Utf8 - typed_col_labels['user_id'] = pl.Utf8 - typed_col_labels['course_module_name'] = pl.Utf8 + col_labels.append(exercise_name_label_for_timestamp) + typed_col_labels[exercise_name_label_for_timestamp] = pl.Utf8 + typed_col_labels["user_id"] = pl.Utf8 + typed_col_labels["course_module_name"] = pl.Utf8 + + options = dict([(option["id"], option["value"]) for option in v["options"]]) - options = dict([(option['id'], option['value']) for option in v['options']]) - for row in submissions_data: row = [row[0], json.loads(row[1]), row[2]] user_answers = dict(row[1]) - submission = dict(ChainMap(*[{item['questionLabel']: options.get(item.get('chosenOptionId'))} for item in user_answers.get('answeredQuestions')])) - - submission['user_id'] = row[0] - submission['course_module_name'] = v['course_module_name'] - exercise_name_label_for_timestamp = v['exercise_name'] - if exercise_name_label_for_timestamp not in module_user_details[ind].columns: - submission[exercise_name_label_for_timestamp] = row[2][0:19] # create_at field + submission = dict( + ChainMap( + *[ + {item["questionLabel"]: options.get(item.get("chosenOptionId"))} + for item in user_answers.get("answeredQuestions") + ] + ) + ) + + submission["user_id"] = row[0] + submission["course_module_name"] = v["course_module_name"] + exercise_name_label_for_timestamp = v["exercise_name"] + if ( + exercise_name_label_for_timestamp + not in module_user_details[ind].columns + ): + submission[exercise_name_label_for_timestamp] = row[2][ + 0:19 + ] # create_at field user_submissions.append(submission) - else: + else: # Columns are "questionLabel", unless it is mulptiple-choice question, # then columns are "questionLabel option" per option. - col_labels = [[' '.join([item['questionLabel'], option.split(';', 1)[0].strip()]) for option in item['options']] if item['answer-type'] == 'multiple-choice' else item['questionLabel'] for item in v['content']] + col_labels = [ + ( + [ + " ".join([item["questionLabel"], option.split(";", 1)[0].strip()]) + for option in item["options"] + ] + if item["answer-type"] == "multiple-choice" + else item["questionLabel"] + ) + for item in v["content"] + ] col_labels = flatten(col_labels) - col_labels.append('user_id') - exercise_name_label_for_timestamp = v['exercise_name'] + col_labels.append("user_id") + exercise_name_label_for_timestamp = v["exercise_name"] if exercise_name_label_for_timestamp not in module_user_details[ind].columns: - col_labels.append(exercise_name_label_for_timestamp) - col_labels.append('course_module_name') + col_labels.append(exercise_name_label_for_timestamp) + col_labels.append("course_module_name") typed_col_labels = dict(ChainMap(*[{col: pl.Utf8} for col in col_labels])) @@ -218,47 +329,77 @@ def flatten(xs): row = [row[0], json.loads(row[1]), row[2]] user_answer = dict(row[1]) - user_answer = dict([(answeredQ.get('questionLabel'), answeredQ.get('answer')) for answeredQ in user_answer.get('answeredQuestions')]) + user_answer = dict( + [ + (answeredQ.get("questionLabel"), answeredQ.get("answer")) + for answeredQ in user_answer.get("answeredQuestions") + ] + ) submission = [] for col in col_labels: - questionAnswer = user_answer.get(col.split(' ', 1)[0], None) + questionAnswer = user_answer.get(col.split(" ", 1)[0], None) # The question is not multipple-choice - if len(col.split(' ', 1)) == 1: + if len(col.split(" ", 1)) == 1: submission.append({col: questionAnswer if questionAnswer else None}) # The question is multipple-choice else: - # If user has not answered this question, all cloumns belonging to this question are NaN, + # If user has not answered this question, all cloumns belonging to this question are NaN, # otherwise "1" indicates option is checked else "0". - submission.append({col: (1 if col.split(' ', 1)[1] in [x.split(';', 1)[0].strip() for x in questionAnswer] else 0) if questionAnswer else None}) + submission.append( + { + col: ( + ( + 1 + if col.split(" ", 1)[1] + in [ + x.split(";", 1)[0].strip() + for x in questionAnswer + ] + else 0 + ) + if questionAnswer + else None + ) + } + ) submission = dict(ChainMap(*submission)) - - submission['user_id']= row[0] - submission['course_module_name'] = v['course_module_name'] - if exercise_name_label_for_timestamp not in module_user_details[ind].columns: - submission[exercise_name_label_for_timestamp] = row[2][0:19] # create_at field + + submission["user_id"] = row[0] + submission["course_module_name"] = v["course_module_name"] + if ( + exercise_name_label_for_timestamp + not in module_user_details[ind].columns + ): + submission[exercise_name_label_for_timestamp] = row[2][ + 0:19 + ] # create_at field user_submissions.append(submission) data = user_submissions if user_submissions else [[None for _ in col_labels]] - additional_cols = pl.DataFrame(data, schema=typed_col_labels).select(col_labels) - - module_user_details[ind] = module_user_details[ind].join(additional_cols, how='left', on=['user_id','course_module_name']) + module_user_details[ind] = module_user_details[ind].join( + additional_cols, how="left", on=["user_id", "course_module_name"] + ) try: os.mkdir("./parsed-outputs") except OSError as error: if error.errno == 17: pass - else: print(error) - + else: + print(error) + user_details = pl.concat(module_user_details, how="diagonal") -dt = datetime.now().strftime('%d-%m-%Y %H:%M:%S') -course_name = '-'.join(course_names) -filename = f'./parsed-outputs/Survey_data-{course_name}-{dt}.csv' -user_details.write_csv(filename, has_header=True, quote='"', null_value='', separator=';') +dt = datetime.now().strftime("%d-%m-%Y %H:%M:%S") +course_name = "-".join(course_names) +filename = f"./parsed-outputs/Survey_data-{course_name}-{dt}.csv" +user_details.write_csv( + filename, has_header=True, quote='"', null_value="", separator=";" +) +