package endpoints import ( "arbeitszeitmessung/helper" "arbeitszeitmessung/helper/paramParser" "arbeitszeitmessung/models" "bytes" "fmt" "log" "log/slog" "net/http" "time" "github.com/Dadido3/go-typst" ) const DE_DATE string = "02.01.2006" func convertDaysToTypst(days []models.IWorkDay, u models.User) ([]typstDay, error) { var typstDays []typstDay for _, day := range days { var thisTypstDay typstDay work, pause, overtime := day.GetTimesVirtual(u, models.WorktimeBaseWeek) thisTypstDay.Date = day.Date().Format(DE_DATE) thisTypstDay.Worktime = helper.FormatDurationFill(work, true) thisTypstDay.Pausetime = helper.FormatDurationFill(pause, true) thisTypstDay.Overtime = helper.FormatDurationFill(overtime, true) thisTypstDay.IsFriday = day.Date().Weekday() == time.Friday thisTypstDay.DayParts = convertDayToTypstDayParts(day, u) typstDays = append(typstDays, thisTypstDay) } return typstDays, nil } func convertDayToTypstDayParts(day models.IWorkDay, user models.User) []typstDayPart { var typstDayParts []typstDayPart if day.IsWorkDay() { workDay, _ := day.(*models.WorkDay) for i := 0; i < len(workDay.Bookings); i += 2 { var typstDayPart typstDayPart typstDayPart.BookingFrom = workDay.Bookings[i].Timestamp.Format("15:04") typstDayPart.BookingTo = workDay.Bookings[i+1].Timestamp.Format("15:04") typstDayPart.WorkType = workDay.Bookings[i].BookingType.Name typstDayPart.IsWorkDay = true typstDayParts = append(typstDayParts, typstDayPart) } if day.IsKurzArbeit() { tsFrom, tsTo := workDay.GenerateKurzArbeitBookings(user) typstDayParts = append(typstDayParts, typstDayPart{ BookingFrom: tsFrom.Format("15:04"), BookingTo: tsTo.Format("15:04"), WorkType: "Kurzarbeit", IsWorkDay: true, }) } } else { absentDay, _ := day.(*models.Absence) typstDayParts = append(typstDayParts, typstDayPart{IsWorkDay: false, WorkType: absentDay.AbwesenheitTyp.Name}) } return typstDayParts } func renderPDF(days []typstDay, metadata typstMetadata) (bytes.Buffer, error) { var markup bytes.Buffer var output bytes.Buffer if err := typst.InjectValues(&markup, map[string]any{"meta": metadata, "days": days}); err != nil { return output, err } // Import the template and invoke the template function with the custom data. // Show is used to replace the current document with whatever content the template function in `template.typ` returns. markup.WriteString(` #import "template.typ": abrechnung #show: doc => abrechnung(meta, days) `) // Compile the prepared markup with Typst and write the result it into `output.pdf`. // f, err := os.Create("output.pdf") // if err != nil { // log.Panicf("Failed to create output file: %v.", err) // } // defer f.Close() // typstCLI := typst.CLI{} if err := typstCLI.Compile(&markup, &output, nil); err != nil { return output, err } return output, nil } func PDFCreateController(w http.ResponseWriter, r *http.Request) { helper.RequiresLogin(Session, w, r) switch r.Method { case http.MethodGet: user, err := models.GetUserFromSession(Session, r.Context()) if err != nil { log.Println("Error getting user!") return } pp := paramParser.New(r.URL.Query()) startDate := pp.ParseTimestampFallback("start_date", time.DateOnly, time.Now()) var members []models.User = make([]models.User, 0) output, err := createReports(user, members, startDate) if err != nil { slog.Warn("Could not create pdf report", slog.Any("Error", err)) } w.Header().Set("Content-type", "application/pdf") output.WriteTo(w) w.WriteHeader(200) default: http.Error(w, "Method not allowed!", http.StatusMethodNotAllowed) } } func createReports(user models.User, employes []models.User, startDate time.Time) (bytes.Buffer, error) { if startDate.Day() > 1 { startDate = startDate.AddDate(0, 0, -(startDate.Day() - 1)) } endDate := startDate.AddDate(0, 1, -1) return createEmployeReport(user, startDate, endDate) } func createEmployeReport(employee models.User, startDate, endDate time.Time) (bytes.Buffer, error) { targetHours := (employee.ArbeitszeitProWoche() / 5) * time.Duration(helper.GetWorkingDays(startDate, endDate)) workingDays := models.GetDays(employee, startDate, endDate, false) slog.Debug("Baseline Working hours", "targetHours", targetHours.Hours()) var actualHours time.Duration for _, day := range workingDays { actualHours += day.TimeWorkVirtual(employee) } worktimeBalance := actualHours - targetHours typstDays, err := convertDaysToTypst(workingDays, employee) if err != nil { log.Panicf("Failed to convert days!") } metadata := typstMetadata{ EmployeeName: fmt.Sprintf("%s %s", employee.Vorname, employee.Name), TimeRange: fmt.Sprintf("%s - %s", startDate.Format(DE_DATE), endDate.Format(DE_DATE)), Overtime: helper.FormatDurationFill(worktimeBalance, true), WorkTime: helper.FormatDurationFill(actualHours, true), OvertimeTotal: "", CurrentTimestamp: time.Now().Format("02.01.2006 - 15:04 Uhr"), } return renderPDF(typstDays, metadata) } func PDFHandler(w http.ResponseWriter, r *http.Request) { helper.RequiresLogin(Session, w, r) startDate := time.Now() if startDate.Day() > 1 { startDate = startDate.AddDate(0, 0, -(startDate.Day() - 1)) } endDate := startDate.AddDate(0, 1, -1) user, err := models.GetUserFromSession(Session, r.Context()) if err != nil { log.Println("Error getting user!") } //TODO: only accepted weeks weeks := models.GetDays(user, startDate, endDate, false) var aggregatedOvertime, aggregatedWorkTime time.Duration for _, day := range weeks { aggregatedOvertime += day.TimeOvertimeReal(user) aggregatedWorkTime += day.TimeWorkVirtual(user) } typstDays, err := convertDaysToTypst(weeks, user) if err != nil { log.Panicf("Failed to convert days!") } metadata := typstMetadata{ EmployeeName: fmt.Sprintf("%s %s", user.Vorname, user.Name), TimeRange: fmt.Sprintf("%s - %s", startDate.Format(DE_DATE), endDate.Format(DE_DATE)), Overtime: helper.FormatDurationFill(aggregatedOvertime, true), WorkTime: helper.FormatDurationFill(aggregatedWorkTime, true), OvertimeTotal: "", CurrentTimestamp: time.Now().Format("02.01.2006 - 15:04 Uhr"), } output, err := renderPDF(typstDays, metadata) if err != nil { slog.Warn("Could not create pdf report", slog.Any("Error", err)) } w.Header().Set("Content-type", "application/pdf") output.WriteTo(w) w.WriteHeader(200) } type typstMetadata struct { TimeRange string `json:"time-range"` EmployeeName string `json:"employee-name"` WorkTime string `json:"worktime"` Overtime string `json:"overtime"` OvertimeTotal string `json:"overtime-total"` CurrentTimestamp string `json:"current-timestamp"` } type typstDayPart struct { BookingFrom string `json:"booking-from"` BookingTo string `json:"booking-to"` WorkType string `json:"worktype"` IsWorkDay bool `json:"is-workday"` } type typstDay struct { Date string `json:"date"` DayParts []typstDayPart `json:"day-parts"` Worktime string `json:"worktime"` Pausetime string `json:"pausetime"` Overtime string `json:"overtime"` IsFriday bool `json:"is-weekend"` }