2024-12-09 19:27:50 +01:00
|
|
|
/*
|
2024-12-10 22:06:49 +01:00
|
|
|
* INPUT: linked list of subjects
|
|
|
|
* OUTPUT: ll of events to iCal
|
|
|
|
* ll of updated subjects to db for next day
|
|
|
|
* return events_ll to caller(ui)??
|
2024-12-09 19:27:50 +01:00
|
|
|
*
|
|
|
|
*/
|
2024-12-10 22:06:49 +01:00
|
|
|
|
|
|
|
#include "planner.h" // for subject and event structs
|
2024-12-12 16:08:19 +01:00
|
|
|
#include "llist.h"
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stddef.h>
|
2024-12-11 13:03:28 +01:00
|
|
|
#include <stdio.h>
|
2024-12-11 01:08:31 +01:00
|
|
|
#include <stdlib.h>
|
2024-12-11 03:09:54 +01:00
|
|
|
#include <string.h>
|
2024-12-12 16:08:19 +01:00
|
|
|
#include <time.h>
|
2024-12-11 01:08:31 +01:00
|
|
|
|
2024-12-13 17:32:41 +01:00
|
|
|
const uint intervalLen = 45; // min
|
|
|
|
const uint pauseLen = 10; //
|
|
|
|
|
2024-12-11 13:41:31 +01:00
|
|
|
const char taskFormat[] =
|
|
|
|
"Task { %.64s = {\n created={%lu},\n deadline={%lu},\n "
|
|
|
|
"priority={%d},\n spare={%lu}\n}\n";
|
|
|
|
|
|
|
|
const char eventFormat[] =
|
2024-12-12 16:08:19 +01:00
|
|
|
"Event { %s = {\n start={%s},\n end={%s},\n spare={%lu}\n}\n";
|
|
|
|
void llistPrintT(llist *head) {
|
|
|
|
llist *c = head;
|
|
|
|
while (c != NULL) {
|
|
|
|
printTask((Task *)c->data);
|
|
|
|
c = c->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// print Event to stdout
|
|
|
|
void printEvent(Event *s) {
|
|
|
|
char st[26];
|
|
|
|
char e[26];
|
|
|
|
ctime_r(&s->plannedStartTime, st);
|
|
|
|
ctime_r(&s->plannedEndTime, e);
|
|
|
|
printf(eventFormat, s->task->name, st, e, s->spare);
|
2024-12-13 18:07:35 +01:00
|
|
|
// printTask(s->task);
|
2024-12-12 16:08:19 +01:00
|
|
|
}
|
|
|
|
void llistPrintE(llist *head) {
|
|
|
|
llist *c = head;
|
|
|
|
while (c != NULL) {
|
|
|
|
printEvent((Event *)c->data);
|
|
|
|
c = c->next;
|
|
|
|
}
|
|
|
|
}
|
2024-12-12 16:24:19 +01:00
|
|
|
void freeEvent(Event *e) {
|
|
|
|
free(e->task);
|
|
|
|
free(e);
|
|
|
|
}
|
|
|
|
void llistFreeE(llist *head) {
|
|
|
|
while (head != NULL) {
|
|
|
|
free((Event *)head->data);
|
|
|
|
llist *tmp = head;
|
|
|
|
head = head->next;
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void llistFreeT(llist *head) {
|
|
|
|
while (head != NULL) {
|
|
|
|
free(head->data);
|
|
|
|
llist *tmp = head;
|
|
|
|
head = head->next;
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
}
|
2024-12-12 16:08:19 +01:00
|
|
|
void planLog(char *msg, bool e) {
|
|
|
|
if (!e)
|
|
|
|
printf("[Planner][log] %s\n", msg);
|
|
|
|
else
|
|
|
|
fprintf(stderr, "==== [Planner][Err] %s\n", msg);
|
|
|
|
}
|
2024-12-11 13:41:31 +01:00
|
|
|
|
2024-12-11 01:08:31 +01:00
|
|
|
Event *newEvent(Task *t, time_t s, time_t e, uint64_t sp) {
|
|
|
|
Event *r = (Event *)malloc(sizeof(Event));
|
|
|
|
if (r != NULL) {
|
|
|
|
r->task = t;
|
2024-12-12 16:08:19 +01:00
|
|
|
r->plannedStartTime = s;
|
2024-12-11 01:08:31 +01:00
|
|
|
r->plannedEndTime = e;
|
|
|
|
r->spare = sp;
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
2024-12-11 13:41:31 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* creat new task with name n created c deadline d priority p spare sp
|
|
|
|
*/
|
2024-12-11 01:08:31 +01:00
|
|
|
Task *newTask(char *n, time_t c, time_t d, int p, uint64_t sp) {
|
|
|
|
Task *r = (Task *)malloc(sizeof(Task));
|
|
|
|
if (r != NULL) {
|
|
|
|
r->created = c;
|
|
|
|
r->deadline = d;
|
|
|
|
r->priority = p;
|
|
|
|
r->spare = sp;
|
|
|
|
r->name = n;
|
|
|
|
}
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
2024-12-11 03:09:54 +01:00
|
|
|
|
2024-12-11 14:06:29 +01:00
|
|
|
/*
|
|
|
|
* create deepCopy of task
|
|
|
|
* !! r->name is malloced
|
|
|
|
*/
|
|
|
|
Task *copyTask(Task *t) {
|
|
|
|
Task *r = malloc(sizeof(Task));
|
|
|
|
if (r != NULL) {
|
|
|
|
r->name = strdup(t->name);
|
|
|
|
r->created = t->created;
|
|
|
|
r->deadline = t->deadline;
|
|
|
|
r->priority = t->priority;
|
|
|
|
r->spare = t->spare;
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2024-12-11 13:03:28 +01:00
|
|
|
void printTask(Task *s) {
|
|
|
|
printf(taskFormat, s->name, s->created, s->deadline, s->priority, s->spare);
|
|
|
|
}
|
|
|
|
|
2024-12-11 13:41:31 +01:00
|
|
|
// return string representation of task
|
|
|
|
// NULL on failure
|
|
|
|
// free string after use
|
|
|
|
char *taskToStr(Task *t) {
|
|
|
|
char *r = malloc(sizeof(char) * strlen(taskFormat) + 0x200);
|
|
|
|
if (r != NULL) {
|
|
|
|
sprintf(r, taskFormat, t->name, t->created, t->deadline, t->priority,
|
|
|
|
t->spare);
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2024-12-11 03:09:54 +01:00
|
|
|
// for llist
|
2024-12-11 13:59:57 +01:00
|
|
|
|
|
|
|
// compare task by name
|
2024-12-12 16:08:19 +01:00
|
|
|
int cmpTaskN(const void *a, const void *b) {
|
2024-12-11 11:27:00 +01:00
|
|
|
Task *aa = (Task *)a;
|
|
|
|
Task *bb = (Task *)b;
|
|
|
|
return strcmp(aa->name, bb->name);
|
|
|
|
}
|
2024-12-11 13:59:57 +01:00
|
|
|
|
|
|
|
// compare task by priority
|
2024-12-12 16:08:19 +01:00
|
|
|
int cmpTaskP(const void *a, const void *b) {
|
2024-12-11 13:59:57 +01:00
|
|
|
Task *aa = (Task *)a;
|
|
|
|
Task *bb = (Task *)b;
|
2024-12-12 16:08:19 +01:00
|
|
|
return bb->priority - aa->priority;
|
2024-12-11 13:59:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// cmp event by task name
|
2024-12-12 16:08:19 +01:00
|
|
|
int cmpEvent(const void *a, const void *b) {
|
2024-12-11 11:27:00 +01:00
|
|
|
Event *aa = (Event *)a;
|
|
|
|
Event *bb = (Event *)b;
|
2024-12-11 13:59:57 +01:00
|
|
|
return cmpTaskN(aa->task, bb->task);
|
2024-12-11 11:27:00 +01:00
|
|
|
}
|
2024-12-12 16:08:19 +01:00
|
|
|
|
2024-12-13 17:32:41 +01:00
|
|
|
llist *genPlan(llist *head, time_t timeAvail) {
|
2024-12-12 16:08:19 +01:00
|
|
|
|
|
|
|
// map llist to pointer arr & sort by priority
|
|
|
|
// second arr sorted bby name
|
|
|
|
int lLen = llistLen(head);
|
|
|
|
planLog("genpla: got llist\n", false);
|
|
|
|
printf("len: %d\n", lLen);
|
|
|
|
llistPrintT(head);
|
|
|
|
|
2024-12-13 17:32:41 +01:00
|
|
|
// Task *sortedNames = calloc(lLen, sizeof(Task));
|
2024-12-12 16:08:19 +01:00
|
|
|
Task *sortedPrio = calloc(lLen, sizeof(Task));
|
2024-12-13 17:32:41 +01:00
|
|
|
if (/*sortedNames == NULL ||*/ sortedPrio == NULL) {
|
2024-12-12 16:08:19 +01:00
|
|
|
planLog("gen plan : calloc failed!!\n", true);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// add Tasks from llist to arr
|
|
|
|
llist *c = head;
|
|
|
|
for (int i = 0; c != NULL; i++) {
|
2024-12-13 17:32:41 +01:00
|
|
|
// sortedNames[i] = *copyTask(c->data);
|
|
|
|
// sortedPrio[i] = *copyTask(c->data);
|
|
|
|
// sortedNames[i] = *(Task *)c->data;
|
2024-12-12 16:08:19 +01:00
|
|
|
sortedPrio[i] = *(Task *)c->data;
|
|
|
|
c = c->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
// sort
|
2024-12-13 17:32:41 +01:00
|
|
|
// qsort(sortedNames, lLen, sizeof(Task), cmpTaskN);
|
2024-12-12 16:08:19 +01:00
|
|
|
qsort(sortedPrio, lLen, sizeof(Task), cmpTaskP);
|
|
|
|
|
2024-12-13 17:32:41 +01:00
|
|
|
// // test print
|
|
|
|
// planLog("sortendName", 0);
|
|
|
|
// for (int i = 0; i < lLen; i++) {
|
|
|
|
// printTask(sortedNames + i);
|
|
|
|
// }
|
2024-12-12 16:08:19 +01:00
|
|
|
|
|
|
|
planLog("sortendPrio", 0);
|
|
|
|
for (int i = 0; i < lLen; i++) {
|
|
|
|
printTask(sortedPrio + i);
|
|
|
|
}
|
|
|
|
|
|
|
|
planLog("creating eventList", false);
|
|
|
|
|
2024-12-13 17:32:41 +01:00
|
|
|
// genertate plan basen on priorities and available time
|
2024-12-12 16:08:19 +01:00
|
|
|
time_t now = time(NULL);
|
|
|
|
struct tm *lc = localtime(&now);
|
|
|
|
llist *events_ll = llistNew(NULL, cmpEvent);
|
2024-12-13 18:07:35 +01:00
|
|
|
if (events_ll == NULL) {
|
|
|
|
exit(1);
|
|
|
|
}
|
2024-12-13 17:32:41 +01:00
|
|
|
// time_t avail = mktime(timeAvail);
|
|
|
|
time_t start, end;
|
|
|
|
// (mktime(lc) < mktime(timeAvail))
|
|
|
|
do {
|
|
|
|
start = mktime(lc); // start now
|
|
|
|
lc->tm_min += intervalLen;
|
|
|
|
end = mktime(lc); //
|
2024-12-12 16:08:19 +01:00
|
|
|
|
2024-12-13 18:07:35 +01:00
|
|
|
llist *tmp = llistGet(head, sortedPrio);
|
|
|
|
Event *c = NULL;
|
|
|
|
if (tmp != NULL) {
|
|
|
|
c = newEvent(tmp->data, start, end,
|
|
|
|
0); // use elem with wighest priority
|
|
|
|
}
|
2024-12-12 16:08:19 +01:00
|
|
|
llistAppend(events_ll, c);
|
2024-12-13 18:07:35 +01:00
|
|
|
// printEvent(c);
|
|
|
|
// printEvent((Event *)((events_ll)->next)->data);
|
2024-12-13 17:32:41 +01:00
|
|
|
|
|
|
|
// decrement priority of first elem and resort list
|
|
|
|
(*sortedPrio).priority -= 1;
|
|
|
|
qsort(sortedPrio, lLen, sizeof(Task), cmpTaskP);
|
|
|
|
|
|
|
|
lc->tm_min += pauseLen; // add pause
|
|
|
|
|
|
|
|
} while (mktime(lc) < timeAvail);
|
|
|
|
|
|
|
|
// for (int i = 0; i < lLen; i++) {
|
|
|
|
// time_t start = mktime(lc);
|
|
|
|
//
|
|
|
|
// printf("start:: %s\n", ctime(&start));
|
|
|
|
// lc->tm_min += 45;
|
|
|
|
// time_t end = mktime(lc);
|
|
|
|
// printf("end:: %s\n", ctime(&end));
|
|
|
|
//
|
|
|
|
// Event *c = newEvent(sortedPrio + i, start, end, 0);
|
|
|
|
// printEvent(c);
|
|
|
|
// lc->tm_min += 15;
|
|
|
|
// llistAppend(events_ll, c);
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
|
|
|
|
// free empty head
|
|
|
|
llist *tmp = events_ll;
|
|
|
|
events_ll = events_ll->next;
|
|
|
|
tmp->next = NULL;
|
|
|
|
llistFreeE(tmp);
|
|
|
|
llistPrintE(events_ll);
|
2024-12-12 16:08:19 +01:00
|
|
|
|
2024-12-13 18:07:35 +01:00
|
|
|
// update prioriteis in original llist
|
|
|
|
for (int i = 0; i < lLen; i++) {
|
|
|
|
llist *tmp = llistGet(head, sortedPrio + i);
|
|
|
|
if (tmp != NULL) {
|
|
|
|
((Task *)(tmp->data))->priority = sortedPrio[i].priority;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(sortedPrio);
|
2024-12-12 16:08:19 +01:00
|
|
|
|
|
|
|
planLog("END GEN PLAN", 0);
|
2024-12-13 18:07:35 +01:00
|
|
|
// llistFreeT(head);
|
2024-12-13 17:32:41 +01:00
|
|
|
// send updated tasks to db for storage
|
2024-12-12 16:08:19 +01:00
|
|
|
|
|
|
|
return events_ll;
|
|
|
|
}
|